You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
atmega-geiger/lib/libssd1306/src/lcd/lcd_common.h

459 lines
16 KiB

/*
MIT License
Copyright (c) 2017-2019, Alexey Dynda
Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
in the Software without restriction, including without limitation the rights
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in all
copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
SOFTWARE.
*/
/**
* @file lcd_common.h global lcd settings
*/
#ifndef _LCD_COMMON_H_
#define _LCD_COMMON_H_
#include "ssd1306_hal/io.h"
#ifdef __cplusplus
extern "C" {
#endif
/**
* @defgroup LCD_INTERFACE_API OLEDs: initialization and service functions
* @{
* @brief LCD initialization and service functions
*
* @details This group contains API functions for OLED displays initialization and
* direct programming of GDRAM. This API can be used to create your own
* graphics functions.
*/
/**
* Enumeration, describing display type
*/
typedef enum
{
/** Default type of LCD display: ssd1306 oled */
LCD_TYPE_SSD1306,
/** Experimental type of LCD display: pcd8544 led */
LCD_TYPE_PCD8544,
/** Support for sh1106 OLED display */
LCD_TYPE_SH1106,
/** Default type of LCD display: ssd1331 oled */
LCD_TYPE_SSD1331,
/** User LCD type */
LCD_TYPE_CUSTOM,
} lcd_type_t;
/**
* Available lcd modes used by the library.
* LCD_MODE_SSD1306_COMPAT is compatible mode, which should be used
* with standard monochrome functions.
*/
typedef enum
{
/**
* Normal mode RGB displays. All ssd1306 monochrome direct draw
* functions do not work in this mode.
*/
LCD_MODE_NORMAL = 0,
/**
* ssd1306 compatible mode. This is special mode, that allows
* to use ssd1306 monochrome direct draw functions, but RGB
* functions will not work.
*/
LCD_MODE_SSD1306_COMPAT = 1,
} lcd_mode_t;
/**
* Structure, describing display driver configuration
*/
typedef struct
{
/** Current selected lcd display type */
lcd_type_t type;
/** Current display width */
lcduint_t width;
/** Current display height */
lcduint_t height;
/**
* @brief Sets block in RAM of lcd display controller to write data to.
*
* Sets block in RAM of lcd display controller to write data to.
* For ssd1306 it uses horizontal addressing mode, while for
* sh1106 the function uses page addressing mode.
* Width can be specified as 0, thus the library will set the right boundary to
* region of RAM block to the right column of the display.
* @param x - column (left region)
* @param y - page (top page of the block)
* @param w - width of the block in pixels to control
*
* @warning - this function initiates session (i2c or spi) and do not close it.
* To close session, please, use ssd1306_intf.stop().
*/
void (*set_block)(lcduint_t x, lcduint_t y, lcduint_t w);
/**
* Switches to the start of next RAM page for the block, specified by
* set_block().
* For ssd1306 it does nothing, while for sh1106 the function moves cursor to
* next page.
*/
void (*next_page)(void);
/**
* Sends 8 monochrome vertical pixels to OLED driver.
* @param data - byte, representing 8 pixels.
*/
void (*send_pixels1)(uint8_t data);
/**
* Sends buffer containing 8 monochrome vertical pixels, encoded in each byte.
* @param buffer - buffer containing monochrome pixels.
* @param len - length of buffer in bytes.
*/
void (*send_pixels_buffer1)(const uint8_t *buffer, uint16_t len);
/**
* @brief Sends RGB pixel encoded in 3-3-2 format to OLED driver.
* Sends RGB pixel encoded in 3-3-2 format to OLED driver.
* @param data - byte, representing RGB8 pixel.
*/
void (*send_pixels8)(uint8_t data);
/**
* @brief Sends RGB pixel encoded in 5-6-5 format to OLED driver.
* Sends RGB pixel encoded in 5-6-5 format to OLED driver.
* @param data 16-bit word, representing RGB16 pixel
*/
void (*send_pixels16)(uint16_t data);
/**
* @brief Sets library display mode for direct draw functions.
*
* Sets library display mode for direct draw functions.
* There are currently 2 modes supported: LCD_MODE_SSD1306_COMPAT and
* LCD_MODE_NORMAL. In general, ssd1306 compatible mode uses different GDRAM
* addressing mode, than normal mode, intended for using with RBG full-color functions.
*
* @param mode lcd mode to activate.
* @see LCD_MODE_SSD1306_COMPAT
* @see LCD_MODE_NORMAL
* @see lcd_mode_t
*/
void (*set_mode)(lcd_mode_t mode);
} ssd1306_lcd_t;
/**
* Structure containing callback to low level function for currently enabled display
*/
extern ssd1306_lcd_t ssd1306_lcd;
/**
* Current display height
* @deprecated Use ssd1306_lcd.height instead.
*/
#define s_displayHeight ssd1306_lcd.height
/**
* Current display width
* @deprecated Use ssd1306_lcd.width instead.
*/
#define s_displayWidth ssd1306_lcd.width
/**
* Current selected lcd display type
* @deprecated Use ssd1306_lcd.type instead.
*/
#define g_lcd_type ssd1306_lcd.type
/**
* Sends byte data to SSD1306 controller memory.
* Performs 3 operations at once: ssd1306_intf.start(), ssd1306_intf.send(), ssd1306_intf.stop();
* @param data - byte to send to the controller memory
* @note At present this function is used only in Arkanoid demo.
* @deprecated There is wide variaty of functions, that can be used for this.
*/
void ssd1306_sendData(uint8_t data) __attribute__ ((deprecated));
/**
* @brief Sets block in RAM of lcd display controller to write data to.
*
* Sets block in RAM of lcd display controller to write data to.
* For ssd1306 it uses horizontal addressing mode, while for
* sh1106 the function uses page addressing mode.
* Width can be specified as 0, thus the library will set the right boundary to
* region of RAM block to the right column of the display.
* @param x - column (left region)
* @param y - page (top page of the block)
* @param w - width of the block in pixels to control
*
* @deprecated Use ssd1306_lcd.set_block() instead.
* @warning - this function initiates session (i2c or spi) and do not close it.
* To close session, please, use ssd1306_intf.stop().
*/
#define ssd1306_setRamBlock ssd1306_lcd.set_block
/**
* Switches to the start of next RAM page for the block, specified by
* ssd1306_setRamBlock().
* For ssd1306 it does nothing, while for sh1106 the function moves cursor to
* next page.
* @deprecated Use ssd1306_lcd.next_page() instead.
*/
#define ssd1306_nextRamPage ssd1306_lcd.next_page
/**
* Sends 8 monochrome vertical pixels to OLED driver.
* @param data - byte, representing 8 pixels.
* @deprecated Use ssd1306_lcd.send_pixels1() instead.
*/
#define ssd1306_sendPixels ssd1306_lcd.send_pixels1
/**
* Sends buffer containing 8 monochrome vertical pixels, encoded in each byte.
* @param buffer - buffer containing monochrome pixels.
* @param len - length of buffer in bytes.
* @deprecated Use ssd1306_lcd.send_pixels_buffer1() instead.
*/
#define ssd1306_sendPixelsBuffer ssd1306_lcd.send_pixels_buffer1
/**
* @brief Sends RGB pixel encoded in 3-3-2 format to OLED driver.
* Sends RGB pixel encoded in 3-3-2 format to OLED driver.
* @param data - byte, representing RGB8 pixel.
* @deprecated Use ssd1306_lcd.send_pixels8() instead.
*/
#define ssd1306_sendPixel8 ssd1306_lcd.send_pixels8
/**
* @brief Sends configuration being passed to lcd display i2c/spi controller.
*
* Sends configuration being passed to lcd display i2c/spi controller.
* The data bytes are sent to lcd controller as is. In case of spi display
* this function sends cmd arguments in command mode. If lcd controller requires
* arguments to be sent in data mode, please use ssd1306_configureSpiDisplay().
*
* @param config configuration, located in flash, to send to i2c/spi controller.
* @param configSize - size of configuration data in bytes.
*/
void ssd1306_configureI2cDisplay(const uint8_t *config, uint8_t configSize);
/**
* @brief Sends configuration being passed to lcd display spi controller.
*
* Sends configuration being passed to lcd display spi controller. If data byte value
* to be sent is less than 255, then data byte is sent in command mode. If data byte
* is 0xFF, the function does't send it to controller, but switches to spi data mode,
* and next byte after will be sent in data spi mode. Then the function will switch back
* to command mode. If lcd controller requires cmd arguments to be sent in command mode,
* please use ssd1306_configureI2cDisplay().
*
* @param config configuration, located in flash, to send to i2c/spi controller.
* @param configSize - size of configuration data in bytes.
*/
void ssd1306_configureSpiDisplay(const uint8_t *config, uint8_t configSize);
/**
* @brief Sends configuration being passed to lcd display spi controller.
*
* Sends configuration being passed to lcd display spi controller. If data byte value
* to be sent is less than 255, then data byte is sent in command mode. Each command has
* additional parameter: number of arguments.
* If lcd controller requires cmd arguments to be sent in command mode,
* please use ssd1306_configureI2cDisplay().
*
* @param config configuration, located in flash, to send to i2c/spi controller.
* @param configSize - size of configuration data in bytes.
*/
void ssd1306_configureSpiDisplay2(const uint8_t *config, uint8_t configSize);
/**
* @brief Sets library display mode for direct draw functions.
*
* Sets library display mode for direct draw functions.
* There are currently 2 modes supported: LCD_MODE_SSD1306_COMPAT and
* LCD_MODE_NORMAL. In general, ssd1306 compatible mode uses different GDRAM
* addressing mode, than normal mode, intended for using with RBG full-color functions.
*
* @param mode lcd mode to activate.
* @see LCD_MODE_SSD1306_COMPAT
* @see LCD_MODE_NORMAL
* @see lcd_mode_t
*/
void ssd1306_setMode(lcd_mode_t mode);
/**
* @brief Does hardware reset for oled controller.
*
* Does hardware reset for oled controller. The function pulls up rstPin
* for 10 milliseconds, then pulls down rstPin for delayMs milliseconds
* and then finally pulls rstPin up.
*
* @param rstPin reset pin number
* @param delayMs delay in milliseconds to hold rstPin in low state
*/
void ssd1306_resetController(int8_t rstPin, uint8_t delayMs);
/**
* Macro SSD1306_COMPAT_SPI_BLOCK_8BIT_CMDS() generates 2 static functions,
* applicable for many oled controllers with 8-bit commands:
* set_block_compat(), next_page_compat(). These functions are to be used
* when working in ssd1306 compatible mode.
* @param column_cmd command opcode for setting column address according to
* oled controller datasheet
* @param row_cmd command opcode for setting row address according to
* oled controller datasheet
* @note It is assumed that column and row commands accept 2 single byte
* arguments: start and end of region
*/
#define SSD1306_COMPAT_SPI_BLOCK_8BIT_CMDS(column_cmd, row_cmd) \
static uint8_t __s_column; \
static uint8_t __s_page; \
static void set_block_compat(lcduint_t x, lcduint_t y, lcduint_t w) \
{ \
uint8_t rx = w ? (x + w - 1) : (ssd1306_lcd.width - 1); \
__s_column = x; \
__s_page = y; \
ssd1306_intf.start(); \
ssd1306_spiDataMode(0); \
ssd1306_intf.send(column_cmd); \
ssd1306_intf.send(x); \
ssd1306_intf.send(rx < ssd1306_lcd.width ? rx : (ssd1306_lcd.width - 1)); \
ssd1306_intf.send(row_cmd); \
ssd1306_intf.send(y<<3); \
ssd1306_intf.send(((y<<3) + 7) < ssd1306_lcd.height ? ((y<<3) + 7) : (ssd1306_lcd.height - 1)); \
ssd1306_spiDataMode(1); \
} \
static void next_page_compat(void) \
{ \
ssd1306_intf.stop(); \
set_block_compat(__s_column,__s_page+1,0); \
} \
/**
* Macro CONTROLLER_NATIVE_SPI_BLOCK_8BIT_CMDS() generates 2 static functions,
* applicable for many oled controllers with 8-bit commands:
* set_block_native(), next_page_native(). These functions are to be used
* when working in oled controller native mode.
* @param column_cmd command opcode for setting column address according to
* oled controller datasheet
* @param row_cmd command opcode for setting row address according to
* oled controller datasheet
* @note It is assumed that column and row commands accept 2 single byte
* arguments: start and end of region
*/
#define CONTROLLER_NATIVE_SPI_BLOCK_8BIT_CMDS(column_cmd, row_cmd) \
static void set_block_native(lcduint_t x, lcduint_t y, lcduint_t w) \
{ \
uint8_t rx = w ? (x + w - 1) : (ssd1306_lcd.width - 1); \
ssd1306_intf.start(); \
ssd1306_spiDataMode(0); \
ssd1306_intf.send(column_cmd); \
ssd1306_intf.send(x); \
ssd1306_intf.send(rx < ssd1306_lcd.width ? rx : (ssd1306_lcd.width - 1)); \
ssd1306_intf.send(row_cmd); \
ssd1306_intf.send(y); \
ssd1306_intf.send(ssd1306_lcd.height - 1); \
ssd1306_spiDataMode(1); \
} \
static void next_page_native(void) \
{ \
} \
/**
* Macro SSD1306_COMPAT_SEND_PIXELS_RGB8_CMDS() generates 2 static functions,
* applicable for many oled controllers in 8-bit RGB mode:
* send_pixels_compat(), send_pixels_buffer_compat(). These functions are to be used
* when working in ssd1306 compatible mode.
*/
#define SSD1306_COMPAT_SEND_PIXELS_RGB8_CMDS() \
extern uint16_t ssd1306_color; \
static void send_pixels_compat(uint8_t data) \
{ \
for (uint8_t i=8; i>0; i--) \
{ \
if ( data & 0x01 ) \
{ \
ssd1306_intf.send( (uint8_t)ssd1306_color ); \
} \
else \
{ \
ssd1306_intf.send( 0B00000000 ); \
} \
data >>= 1; \
} \
} \
static void send_pixels_buffer_compat(const uint8_t *buffer, uint16_t len) \
{ \
while(len--) \
{ \
send_pixels_compat(*buffer); \
buffer++; \
} \
}
/**
* Macro SSD1306_COMPAT_SEND_PIXELS_RGB16_CMDS() generates 2 static functions,
* applicable for many oled controllers in 16-bit RGB mode:
* send_pixels_compat16(), send_pixels_buffer_compat16(). These functions are to be used
* when working in ssd1306 compatible mode.
*/
#define SSD1306_COMPAT_SEND_PIXELS_RGB16_CMDS() \
extern uint16_t ssd1306_color; \
static void send_pixels_compat16(uint8_t data) \
{ \
for (uint8_t i=8; i>0; i--) \
{ \
if ( data & 0x01 ) \
{ \
ssd1306_intf.send( (uint8_t)(ssd1306_color >> 8 ) ); \
ssd1306_intf.send( (uint8_t)(ssd1306_color & 0xFF) ); \
} \
else \
{ \
ssd1306_intf.send( 0B00000000 ); \
ssd1306_intf.send( 0B00000000 ); \
} \
data >>= 1; \
} \
} \
static void send_pixels_buffer_compat16(const uint8_t *buffer, uint16_t len) \
{ \
while(len--) \
{ \
send_pixels_compat16(*buffer); \
buffer++; \
} \
}
/**
* @}
*/
#ifdef __cplusplus
}
#endif
#endif /* _LCD_COMMON_H_ */