17eb416ef8
* [lvgl_helpers] Cleanup and misc code cleanup Checks for SPI_HOST_MAX symbol before using it. Rename lvgl_driver_init to lvgl_interface_init because it now only initialize the interface bus for display drivers, we still need to remove the indev drivers from here. Use types defined in spi_types.h for spi host (spi_host_device_t) and spi dma channels (spi_dma_chan_t). Also add a couple of symbols to avoid using magic numbers * [lvgl_helpers] Reduce usage of if defined in lvgl_interface_init * [lvgl_helpers] Fix spi dma channel for ESP-IDF versions <= 4.2 * [examples] Update hello_world to call lvgl_interface_init * Add lvgl_get_display_buffer_size helper This helper will allow us to get the calculated display buffer size instead of using a global symbol. * Implement lvgl_get_display_buffer_size This API will be used to get the calculation of display buffer size. * Delete DISP_BUF_SIZE symbols The same functionality is handled by lvgl_get_display_buffer_size * Move SPI max transfer size calculation to helper Use calculate_spi_max_transfer_size to calculate the SPI max transfer size for the SPI master configuration * Remove SPI_BUS_MAX_TRANSFER_SZ definition Same functionality is now handled in calculate_spi_max_transfer_size * Update display buffer size calculation Use lvgl_get_display_buffer_size helper instead of DISP_BUF_SIZE symbol * Update example to LVGL v8 Add comments about changes from: - LVGL v7 to LVGL v8 - Configuration helpers and display drivers * Update lvgl_helpers.c * Update sh1107 driver * Update EVE driver Check for symbols used in previous implementations before trying to use them and add a fallback temporary implementation when not found. The falback implementation isn't tested with hardware. Symbols: - DISP_BUF_SIZE - SPI_TRANSFER_SIZE * Update uc8151d driver * Update jd79653a driver * Update ra8875 driver * Update il3820.h Check for LV_HOR_RES_MAX and LV_VER_RES_MAX before trying to use them * Update lvgl_helpers.c Check for ESP-IDF version before trying to use spi_dma_chan_t type
260 lines
7.1 KiB
C
260 lines
7.1 KiB
C
/**
|
|
* @file sh1107.c
|
|
*
|
|
*/
|
|
|
|
/*********************
|
|
* INCLUDES
|
|
*********************/
|
|
#include "sh1107.h"
|
|
#include "disp_spi.h"
|
|
#include "driver/gpio.h"
|
|
#include "esp_log.h"
|
|
#include "freertos/FreeRTOS.h"
|
|
#include "freertos/task.h"
|
|
|
|
/*********************
|
|
* DEFINES
|
|
*********************/
|
|
|
|
/**********************
|
|
* TYPEDEFS
|
|
**********************/
|
|
|
|
/*The LCD needs a bunch of command/argument values to be initialized. They are stored in this struct. */
|
|
typedef struct {
|
|
uint8_t cmd;
|
|
uint8_t data[16];
|
|
uint8_t databytes; //No of data in data; bit 7 = delay after set; 0xFF = end of cmds.
|
|
} lcd_init_cmd_t;
|
|
|
|
/**********************
|
|
* STATIC PROTOTYPES
|
|
**********************/
|
|
static void sh1107_send_cmd(uint8_t cmd);
|
|
static void sh1107_send_data(void * data, uint16_t length);
|
|
static void sh1107_send_color(void * data, uint16_t length);
|
|
|
|
static lv_coord_t get_display_ver_res(lv_disp_drv_t *disp_drv);
|
|
static lv_coord_t get_display_hor_res(lv_disp_drv_t *disp_drv);
|
|
|
|
/**********************
|
|
* STATIC VARIABLES
|
|
**********************/
|
|
|
|
/**********************
|
|
* MACROS
|
|
**********************/
|
|
|
|
#define BIT_SET(a,b) ((a) |= (1U<<(b)))
|
|
#define BIT_CLEAR(a,b) ((a) &= ~(1U<<(b)))
|
|
|
|
/**********************
|
|
* GLOBAL FUNCTIONS
|
|
**********************/
|
|
|
|
void sh1107_init(void)
|
|
{
|
|
// Use Double Bytes Commands if necessary, but not Command+Data
|
|
// Initialization taken from https://github.com/nopnop2002/esp-idf-m5stick
|
|
lcd_init_cmd_t init_cmds[]={
|
|
{0xAE, {0}, 0}, // Turn display off
|
|
{0xDC, {0}, 0}, // Set display start line
|
|
{0x00, {0}, 0}, // ...value
|
|
{0x81, {0}, 0}, // Set display contrast
|
|
{0x2F, {0}, 0}, // ...value
|
|
{0x20, {0}, 0}, // Set memory mode
|
|
{0xA0, {0}, 0}, // Non-rotated display
|
|
#if defined CONFIG_LV_DISPLAY_ORIENTATION_LANDSCAPE
|
|
{0xC8, {0}, 0}, // flipped vertical
|
|
#elif defined CONFIG_LV_DISPLAY_ORIENTATION_PORTRAIT
|
|
{0xC7, {0}, 0}, // flipped vertical
|
|
#endif
|
|
{0xA8, {0}, 0}, // Set multiplex ratio
|
|
{0x7F, {0}, 0}, // ...value
|
|
{0xD3, {0}, 0}, // Set display offset to zero
|
|
{0x60, {0}, 0}, // ...value
|
|
{0xD5, {0}, 0}, // Set display clock divider
|
|
{0x51, {0}, 0}, // ...value
|
|
{0xD9, {0}, 0}, // Set pre-charge
|
|
{0x22, {0}, 0}, // ...value
|
|
{0xDB, {0}, 0}, // Set com detect
|
|
{0x35, {0}, 0}, // ...value
|
|
{0xB0, {0}, 0}, // Set page address
|
|
{0xDA, {0}, 0}, // Set com pins
|
|
{0x12, {0}, 0}, // ...value
|
|
{0xA4, {0}, 0}, // output ram to display
|
|
#if defined CONFIG_LV_INVERT_COLORS
|
|
{0xA7, {0}, 0}, // inverted display
|
|
#else
|
|
{0xA6, {0}, 0}, // Non-inverted display
|
|
#endif
|
|
{0xAF, {0}, 0}, // Turn display on
|
|
{0, {0}, 0xff},
|
|
};
|
|
|
|
//Initialize non-SPI GPIOs
|
|
gpio_pad_select_gpio(SH1107_DC);
|
|
gpio_set_direction(SH1107_DC, GPIO_MODE_OUTPUT);
|
|
|
|
#if SH1107_USE_RST
|
|
gpio_pad_select_gpio(SH1107_RST);
|
|
gpio_set_direction(SH1107_RST, GPIO_MODE_OUTPUT);
|
|
|
|
//Reset the display
|
|
gpio_set_level(SH1107_RST, 0);
|
|
vTaskDelay(100 / portTICK_RATE_MS);
|
|
gpio_set_level(SH1107_RST, 1);
|
|
vTaskDelay(100 / portTICK_RATE_MS);
|
|
#endif
|
|
|
|
//Send all the commands
|
|
uint16_t cmd = 0;
|
|
while (init_cmds[cmd].databytes!=0xff) {
|
|
sh1107_send_cmd(init_cmds[cmd].cmd);
|
|
sh1107_send_data(init_cmds[cmd].data, init_cmds[cmd].databytes&0x1F);
|
|
if (init_cmds[cmd].databytes & 0x80) {
|
|
vTaskDelay(100 / portTICK_RATE_MS);
|
|
}
|
|
cmd++;
|
|
}
|
|
}
|
|
|
|
void sh1107_set_px_cb(lv_disp_drv_t * disp_drv, uint8_t * buf, lv_coord_t buf_w, lv_coord_t x, lv_coord_t y,
|
|
lv_color_t color, lv_opa_t opa)
|
|
{
|
|
/* buf_w will be ignored, the configured CONFIG_LV_DISPLAY_HEIGHT and _WIDTH,
|
|
and CONFIG_LV_DISPLAY_ORIENTATION_LANDSCAPE and _PORTRAIT will be used. */
|
|
uint16_t byte_index = 0;
|
|
uint8_t bit_index = 0;
|
|
|
|
#if defined CONFIG_LV_DISPLAY_ORIENTATION_LANDSCAPE
|
|
byte_index = y + (( x>>3 ) * get_display_ver_res(disp_drv));
|
|
bit_index = x & 0x7;
|
|
#elif defined CONFIG_LV_DISPLAY_ORIENTATION_PORTRAIT
|
|
byte_index = x + (( y>>3 ) * get_display_hor_res(disp_drv));
|
|
bit_index = y & 0x7;
|
|
#endif
|
|
|
|
if ((color.full == 0) && (LV_OPA_TRANSP != opa)) {
|
|
BIT_SET(buf[byte_index], bit_index);
|
|
} else {
|
|
BIT_CLEAR(buf[byte_index], bit_index);
|
|
}
|
|
}
|
|
|
|
void sh1107_flush(lv_disp_drv_t * drv, const lv_area_t * area, lv_color_t * color_map)
|
|
{
|
|
uint8_t columnLow = area->x1 & 0x0F;
|
|
uint8_t columnHigh = (area->x1 >> 4) & 0x0F;
|
|
uint8_t row1 = 0, row2 = 0;
|
|
uint32_t size = 0;
|
|
void *ptr;
|
|
|
|
#if defined CONFIG_LV_DISPLAY_ORIENTATION_LANDSCAPE
|
|
row1 = area->x1>>3;
|
|
row2 = area->x2>>3;
|
|
#else
|
|
row1 = area->y1>>3;
|
|
row2 = area->y2>>3;
|
|
#endif
|
|
for(int i = row1; i < row2+1; i++){
|
|
sh1107_send_cmd(0x10 | columnHigh); // Set Higher Column Start Address for Page Addressing Mode
|
|
sh1107_send_cmd(0x00 | columnLow); // Set Lower Column Start Address for Page Addressing Mode
|
|
sh1107_send_cmd(0xB0 | i); // Set Page Start Address for Page Addressing Mode
|
|
size = area->y2 - area->y1 + 1;
|
|
#if defined CONFIG_LV_DISPLAY_ORIENTATION_LANDSCAPE
|
|
ptr = color_map + i * get_display_ver_res(drv);
|
|
#else
|
|
ptr = color_map + i * get_display_hor_res(drv);
|
|
#endif
|
|
if(i != row2){
|
|
sh1107_send_data( (void *) ptr, size);
|
|
} else {
|
|
// complete sending data by sh1107_send_color() and thus call lv_flush_ready()
|
|
sh1107_send_color( (void *) ptr, size);
|
|
}
|
|
}
|
|
}
|
|
|
|
void sh1107_rounder(lv_disp_drv_t * disp_drv, lv_area_t *area)
|
|
{
|
|
// workaround: always send complete size display buffer
|
|
area->x1 = 0;
|
|
area->y1 = 0;
|
|
area->x2 = get_display_hor_res(disp_drv) - 1;
|
|
area->y2 = get_display_ver_res(disp_drv) - 1;
|
|
}
|
|
|
|
void sh1107_sleep_in(void)
|
|
{
|
|
sh1107_send_cmd(0xAE);
|
|
}
|
|
|
|
void sh1107_sleep_out(void)
|
|
{
|
|
sh1107_send_cmd(0xAF);
|
|
}
|
|
|
|
/**********************
|
|
* STATIC FUNCTIONS
|
|
**********************/
|
|
|
|
|
|
static void sh1107_send_cmd(uint8_t cmd)
|
|
{
|
|
disp_wait_for_pending_transactions();
|
|
gpio_set_level(SH1107_DC, 0); /*Command mode*/
|
|
disp_spi_send_data(&cmd, 1);
|
|
}
|
|
|
|
static void sh1107_send_data(void * data, uint16_t length)
|
|
{
|
|
disp_wait_for_pending_transactions();
|
|
gpio_set_level(SH1107_DC, 1); /*Data mode*/
|
|
disp_spi_send_data(data, length);
|
|
}
|
|
|
|
static void sh1107_send_color(void * data, uint16_t length)
|
|
{
|
|
disp_wait_for_pending_transactions();
|
|
gpio_set_level(SH1107_DC, 1); /*Data mode*/
|
|
disp_spi_send_colors(data, length);
|
|
}
|
|
|
|
static lv_coord_t get_display_ver_res(lv_disp_drv_t *disp_drv)
|
|
{
|
|
lv_coord_t val = 0;
|
|
|
|
#if LVGL_VERSION_MAJOR < 8
|
|
#if defined CONFIG_LV_DISPLAY_ORIENTATION_LANDSCAPE
|
|
val = LV_VER_RES_MAX;
|
|
#endif
|
|
#else
|
|
/* ToDo Use display rotation API to get vertical size */
|
|
#if defined CONFIG_LV_DISPLAY_ORIENTATION_LANDSCAPE
|
|
val = lv_disp_get_ver_res((lv_disp_t *) disp_drv);
|
|
#endif
|
|
#endif
|
|
|
|
return val;
|
|
}
|
|
|
|
static lv_coord_t get_display_hor_res(lv_disp_drv_t *disp_drv)
|
|
{
|
|
lv_coord_t val = 0;
|
|
|
|
#if LVGL_VERSION_MAJOR < 8
|
|
#if defined CONFIG_LV_DISPLAY_ORIENTATION_PORTRAIT
|
|
val = LV_HOR_RES_MAX;
|
|
#endif
|
|
#else
|
|
/* ToDo Use display rotation API to get horizontal size */
|
|
#if defined CONFIG_LV_DISPLAY_ORIENTATION_PORTRAIT
|
|
val = lv_disp_get_hor_res((lv_disp_t *) disp_drv);
|
|
#endif
|
|
#endif
|
|
|
|
return val;
|
|
}
|