2021-05-12 11:26:07 +08:00
/*
2024-01-01 09:00:24 -08:00
* SPDX - FileCopyrightText : 2021 - 2024 Espressif Systems ( Shanghai ) CO LTD
2021-05-12 11:26:07 +08:00
*
* SPDX - License - Identifier : Apache - 2.0
*/
# pragma once
# include <stdbool.h>
# include "esp_err.h"
# include "esp_lcd_types.h"
# include "soc/soc_caps.h"
2021-08-20 11:48:33 +08:00
# include "hal/lcd_types.h"
2023-08-03 12:31:36 +08:00
# include "hal/i2c_types.h"
# include "driver/i2c_types.h"
2021-05-12 11:26:07 +08:00
2023-11-13 17:50:29 +08:00
# define ESP_LCD_I80_BUS_WIDTH_MAX 16 /*!< Maximum width of I80 bus */
2021-05-12 11:26:07 +08:00
# ifdef __cplusplus
extern " C " {
# endif
typedef void * esp_lcd_spi_bus_handle_t ; /*!< Type of LCD SPI bus handle */
2023-08-03 12:31:36 +08:00
typedef uint32_t esp_lcd_i2c_bus_handle_t ; /*!< Type of LCD I2C bus handle */
2021-05-12 11:26:07 +08:00
typedef struct esp_lcd_i80_bus_t * esp_lcd_i80_bus_handle_t ; /*!< Type of LCD intel 8080 bus handle */
2022-11-07 12:21:30 +01:00
/**
* @ brief Type of LCD panel IO event data
*/
typedef struct {
} esp_lcd_panel_io_event_data_t ;
/**
* @ brief Declare the prototype of the function that will be invoked when panel IO finishes transferring color data
*
* @ param [ in ] panel_io LCD panel IO handle , which is created by factory API like ` esp_lcd_new_panel_io_spi ( ) `
* @ param [ in ] edata Panel IO event data , fed by driver
* @ param [ in ] user_ctx User data , passed from ` esp_lcd_panel_io_xxx_config_t `
* @ return Whether a high priority task has been waken up by this function
*/
typedef bool ( * esp_lcd_panel_io_color_trans_done_cb_t ) ( esp_lcd_panel_io_handle_t panel_io , esp_lcd_panel_io_event_data_t * edata , void * user_ctx ) ;
/**
* @ brief Type of LCD panel IO callbacks
*/
typedef struct {
esp_lcd_panel_io_color_trans_done_cb_t on_color_trans_done ; /*!< Callback invoked when color data transfer has finished */
} esp_lcd_panel_io_callbacks_t ;
2022-05-06 09:41:54 +02:00
/**
* @ brief Transmit LCD command and receive corresponding parameters
*
* @ note Commands sent by this function are short , so they are sent using polling transactions .
2022-10-10 16:30:52 +08:00
* The function does not return before the command transfer is completed .
2022-05-06 09:41:54 +02:00
* If any queued transactions sent by ` esp_lcd_panel_io_tx_color ( ) ` are still pending when this function is called ,
* this function will wait until they are finished and the queue is empty before sending the command ( s ) .
*
* @ param [ in ] io LCD panel IO handle , which is created by other factory API like ` esp_lcd_new_panel_io_spi ( ) `
* @ param [ in ] lcd_cmd The specific LCD command , set to - 1 if no command needed
* @ param [ out ] param Buffer for the command data
* @ param [ in ] param_size Size of ` param ` buffer
* @ return
* - ESP_ERR_INVALID_ARG if parameter is invalid
* - ESP_ERR_NOT_SUPPORTED if read is not supported by transport
* - ESP_OK on success
*/
esp_err_t esp_lcd_panel_io_rx_param ( esp_lcd_panel_io_handle_t io , int lcd_cmd , void * param , size_t param_size ) ;
2021-05-12 11:26:07 +08:00
/**
* @ brief Transmit LCD command and corresponding parameters
*
* @ note Commands sent by this function are short , so they are sent using polling transactions .
2022-10-10 16:30:52 +08:00
* The function does not return before the command transfer is completed .
2021-05-12 11:26:07 +08:00
* If any queued transactions sent by ` esp_lcd_panel_io_tx_color ( ) ` are still pending when this function is called ,
* this function will wait until they are finished and the queue is empty before sending the command ( s ) .
*
* @ param [ in ] io LCD panel IO handle , which is created by other factory API like ` esp_lcd_new_panel_io_spi ( ) `
2023-04-03 14:08:59 +08:00
* @ param [ in ] lcd_cmd The specific LCD command , set to - 1 if no command needed
2021-05-12 11:26:07 +08:00
* @ param [ in ] param Buffer that holds the command specific parameters , set to NULL if no parameter is needed for the command
* @ param [ in ] param_size Size of ` param ` in memory , in bytes , set to zero if no parameter is needed for the command
* @ return
* - ESP_ERR_INVALID_ARG if parameter is invalid
* - ESP_OK on success
*/
2021-08-04 20:11:31 +08:00
esp_err_t esp_lcd_panel_io_tx_param ( esp_lcd_panel_io_handle_t io , int lcd_cmd , const void * param , size_t param_size ) ;
2021-05-12 11:26:07 +08:00
/**
* @ brief Transmit LCD RGB data
*
* @ note This function will package the command and RGB data into a transaction , and push into a queue .
* The real transmission is performed in the background ( DMA + interrupt ) .
* The caller should take care of the lifecycle of the ` color ` buffer .
* Recycling of color buffer should be done in the callback ` on_color_trans_done ( ) ` .
*
* @ param [ in ] io LCD panel IO handle , which is created by factory API like ` esp_lcd_new_panel_io_spi ( ) `
2022-10-10 16:30:52 +08:00
* @ param [ in ] lcd_cmd The specific LCD command , set to - 1 if no command needed
2021-05-12 11:26:07 +08:00
* @ param [ in ] color Buffer that holds the RGB color data
* @ param [ in ] color_size Size of ` color ` in memory , in bytes
* @ return
* - ESP_ERR_INVALID_ARG if parameter is invalid
* - ESP_OK on success
*/
2021-08-04 20:11:31 +08:00
esp_err_t esp_lcd_panel_io_tx_color ( esp_lcd_panel_io_handle_t io , int lcd_cmd , const void * color , size_t color_size ) ;
2021-05-12 11:26:07 +08:00
/**
2022-10-10 16:30:52 +08:00
* @ brief Destroy LCD panel IO handle ( deinitialize panel and free all corresponding resource )
2021-05-12 11:26:07 +08:00
*
* @ param [ in ] io LCD panel IO handle , which is created by factory API like ` esp_lcd_new_panel_io_spi ( ) `
* @ return
* - ESP_ERR_INVALID_ARG if parameter is invalid
* - ESP_OK on success
*/
esp_err_t esp_lcd_panel_io_del ( esp_lcd_panel_io_handle_t io ) ;
2021-09-27 11:32:29 +08:00
/**
2022-11-07 12:21:30 +01:00
* @ brief Register LCD panel IO callbacks
2021-09-27 11:32:29 +08:00
*
2022-11-07 12:21:30 +01:00
* @ param [ in ] io LCD panel IO handle , which is created by factory API like ` esp_lcd_new_panel_io_spi ( ) `
* @ param [ in ] cbs structure with all LCD panel IO callbacks
* @ param [ in ] user_ctx User private data , passed directly to callback ' s user_ctx
* @ return
* - ESP_ERR_INVALID_ARG if parameter is invalid
* - ESP_OK on success
2021-09-27 11:32:29 +08:00
*/
2022-11-07 12:21:30 +01:00
esp_err_t esp_lcd_panel_io_register_event_callbacks ( esp_lcd_panel_io_handle_t io , const esp_lcd_panel_io_callbacks_t * cbs , void * user_ctx ) ;
2021-09-27 11:32:29 +08:00
2021-05-12 11:26:07 +08:00
/**
* @ brief Panel IO configuration structure , for SPI interface
*/
typedef struct {
int cs_gpio_num ; /*!< GPIO used for CS line */
2022-10-10 14:57:17 +08:00
int dc_gpio_num ; /*!< GPIO used to select the D/C line, set this to -1 if the D/C line is not used */
2021-05-12 11:26:07 +08:00
int spi_mode ; /*!< Traditional SPI mode (0~3) */
unsigned int pclk_hz ; /*!< Frequency of pixel clock */
size_t trans_queue_depth ; /*!< Size of internal transaction queue */
2021-09-27 11:32:29 +08:00
esp_lcd_panel_io_color_trans_done_cb_t on_color_trans_done ; /*!< Callback invoked when color data transfer has finished */
void * user_ctx ; /*!< User private data, passed directly to on_color_trans_done's user_ctx */
2021-08-04 20:11:31 +08:00
int lcd_cmd_bits ; /*!< Bit-width of LCD command */
int lcd_param_bits ; /*!< Bit-width of LCD parameter */
2021-05-12 11:26:07 +08:00
struct {
2024-01-01 09:00:24 -08:00
unsigned int dc_high_on_cmd : 1 ; /*!< If enabled, DC level = 1 indicates command transfer */
unsigned int dc_low_on_data : 1 ; /*!< If enabled, DC level = 0 indicates color data transfer */
unsigned int dc_low_on_param : 1 ; /*!< If enabled, DC level = 0 indicates parameter transfer */
2021-07-09 16:46:27 +08:00
unsigned int octal_mode : 1 ; /*!< transmit with octal mode (8 data lines), this mode is used to simulate Intel 8080 timing */
2023-08-25 12:43:20 +08:00
unsigned int quad_mode : 1 ; /*!< transmit with quad mode (4 data lines), this mode is useful when transmitting LCD parameters (Only use one line for command) */
2024-01-01 09:00:24 -08:00
unsigned int sio_mode : 1 ; /*!< Read and write through a single data line (MOSI) */
2022-04-18 18:30:47 +08:00
unsigned int lsb_first : 1 ; /*!< transmit LSB bit first */
2022-10-10 14:42:47 +08:00
unsigned int cs_high_active : 1 ; /*!< CS line is high active */
2022-04-18 18:30:47 +08:00
} flags ; /*!< Extra flags to fine-tune the SPI device */
2021-05-12 11:26:07 +08:00
} esp_lcd_panel_io_spi_config_t ;
/**
* @ brief Create LCD panel IO handle , for SPI interface
*
* @ param [ in ] bus SPI bus handle
* @ param [ in ] io_config IO configuration , for SPI interface
* @ param [ out ] ret_io Returned IO handle
* @ return
* - ESP_ERR_INVALID_ARG if parameter is invalid
* - ESP_ERR_NO_MEM if out of memory
* - ESP_OK on success
*/
esp_err_t esp_lcd_new_panel_io_spi ( esp_lcd_spi_bus_handle_t bus , const esp_lcd_panel_io_spi_config_t * io_config , esp_lcd_panel_io_handle_t * ret_io ) ;
2022-05-11 15:52:09 +08:00
/**
* @ brief Panel IO configuration structure , for I2C interface
*
*/
2021-05-12 11:26:07 +08:00
typedef struct {
uint32_t dev_addr ; /*!< I2C device address */
2021-09-27 11:32:29 +08:00
esp_lcd_panel_io_color_trans_done_cb_t on_color_trans_done ; /*!< Callback invoked when color data transfer has finished */
void * user_ctx ; /*!< User private data, passed directly to on_color_trans_done's user_ctx */
2022-10-10 16:30:52 +08:00
size_t control_phase_bytes ; /*!< I2C LCD panel will encode control information (e.g. D/C selection) into control phase, in several bytes */
2021-05-12 11:26:07 +08:00
unsigned int dc_bit_offset ; /*!< Offset of the D/C selection bit in control phase */
2021-08-04 20:11:31 +08:00
int lcd_cmd_bits ; /*!< Bit-width of LCD command */
int lcd_param_bits ; /*!< Bit-width of LCD parameter */
2021-05-12 11:26:07 +08:00
struct {
unsigned int dc_low_on_data : 1 ; /*!< If this flag is enabled, DC line = 0 means transfer data, DC line = 1 means transfer command; vice versa */
2022-05-06 09:41:54 +02:00
unsigned int disable_control_phase : 1 ; /*!< If this flag is enabled, the control phase isn't used */
2022-05-11 15:52:09 +08:00
} flags ; /*!< Extra flags to fine-tune the I2C device */
2023-08-03 12:31:36 +08:00
uint32_t scl_speed_hz ; /*!< I2C LCD SCL frequency (hz) */
2021-05-12 11:26:07 +08:00
} esp_lcd_panel_io_i2c_config_t ;
/**
2023-08-03 12:31:36 +08:00
* @ brief Create LCD panel IO handle , for I2C interface in legacy implementation
*
* @ param [ in ] bus I2C bus handle , ( in uint32_t )
* @ param [ in ] io_config IO configuration , for I2C interface
* @ param [ out ] ret_io Returned IO handle
*
* @ note Please don ' t call this function in your project directly . Please call ` esp_lcd_new_panel_to_i2c ` instead .
*
* @ return
* - ESP_ERR_INVALID_ARG if parameter is invalid
* - ESP_ERR_NO_MEM if out of memory
* - ESP_OK on success
*/
esp_err_t esp_lcd_new_panel_io_i2c_v1 ( uint32_t bus , const esp_lcd_panel_io_i2c_config_t * io_config , esp_lcd_panel_io_handle_t * ret_io ) ;
/**
* @ brief Create LCD panel IO handle , for I2C interface in new implementation
*
* @ param [ in ] bus I2C bus handle , ( in i2c_master_dev_handle_t )
* @ param [ in ] io_config IO configuration , for I2C interface
* @ param [ out ] ret_io Returned IO handle
*
* @ note Please don ' t call this function in your project directly . Please call ` esp_lcd_new_panel_to_i2c ` instead .
*
* @ return
* - ESP_ERR_INVALID_ARG if parameter is invalid
* - ESP_ERR_NO_MEM if out of memory
* - ESP_OK on success
*/
esp_err_t esp_lcd_new_panel_io_i2c_v2 ( i2c_master_bus_handle_t bus , const esp_lcd_panel_io_i2c_config_t * io_config , esp_lcd_panel_io_handle_t * ret_io ) ;
/**
* @ brief Create LCD panel IO handle
2021-05-12 11:26:07 +08:00
*
* @ param [ in ] bus I2C bus handle
* @ param [ in ] io_config IO configuration , for I2C interface
* @ param [ out ] ret_io Returned IO handle
* @ return
* - ESP_ERR_INVALID_ARG if parameter is invalid
* - ESP_ERR_NO_MEM if out of memory
* - ESP_OK on success
*/
2023-08-03 12:31:36 +08:00
# define esp_lcd_new_panel_io_i2c(bus, io_config, ret_io) _Generic((bus), \
i2c_master_bus_handle_t : esp_lcd_new_panel_io_i2c_v2 , \
default : esp_lcd_new_panel_io_i2c_v1 ) ( bus , io_config , ret_io ) \
2021-05-12 11:26:07 +08:00
# if SOC_LCD_I80_SUPPORTED
/**
* @ brief LCD Intel 8080 bus configuration structure
*/
typedef struct {
int dc_gpio_num ; /*!< GPIO used for D/C line */
int wr_gpio_num ; /*!< GPIO used for WR line */
2021-08-20 11:48:33 +08:00
lcd_clock_source_t clk_src ; /*!< Clock source for the I80 LCD peripheral */
2023-11-13 17:50:29 +08:00
int data_gpio_nums [ ESP_LCD_I80_BUS_WIDTH_MAX ] ; /*!< GPIOs used for data lines */
2021-08-04 20:11:31 +08:00
size_t bus_width ; /*!< Number of data lines, 8 or 16 */
2021-05-12 11:26:07 +08:00
size_t max_transfer_bytes ; /*!< Maximum transfer size, this determines the length of internal DMA link */
2021-12-28 15:30:22 +08:00
size_t psram_trans_align ; /*!< DMA transfer alignment for data allocated from PSRAM */
size_t sram_trans_align ; /*!< DMA transfer alignment for data allocated from SRAM */
2021-05-12 11:26:07 +08:00
} esp_lcd_i80_bus_config_t ;
/**
* @ brief Create Intel 8080 bus handle
*
* @ param [ in ] bus_config Bus configuration
* @ param [ out ] ret_bus Returned bus handle
* @ return
* - ESP_ERR_INVALID_ARG if parameter is invalid
* - ESP_ERR_NO_MEM if out of memory
* - ESP_ERR_NOT_FOUND if no free bus is available
* - ESP_OK on success
*/
esp_err_t esp_lcd_new_i80_bus ( const esp_lcd_i80_bus_config_t * bus_config , esp_lcd_i80_bus_handle_t * ret_bus ) ;
/**
2022-10-10 16:30:52 +08:00
* @ brief Destroy Intel 8080 bus handle
2021-05-12 11:26:07 +08:00
*
* @ param [ in ] bus Intel 8080 bus handle , created by ` esp_lcd_new_i80_bus ( ) `
* @ return
* - ESP_ERR_INVALID_ARG if parameter is invalid
* - ESP_ERR_INVALID_STATE if there still be some device attached to the bus
* - ESP_OK on success
*/
esp_err_t esp_lcd_del_i80_bus ( esp_lcd_i80_bus_handle_t bus ) ;
/**
* @ brief Panel IO configuration structure , for intel 8080 interface
*/
typedef struct {
2021-08-24 20:37:58 +08:00
int cs_gpio_num ; /*!< GPIO used for CS line, set to -1 will declaim exclusively use of I80 bus */
2022-08-04 13:08:48 +08:00
uint32_t pclk_hz ; /*!< Frequency of pixel clock */
2021-05-12 11:26:07 +08:00
size_t trans_queue_depth ; /*!< Transaction queue size, larger queue, higher throughput */
2022-10-10 16:30:52 +08:00
esp_lcd_panel_io_color_trans_done_cb_t on_color_trans_done ; /*!< Callback invoked when color data was transferred done */
2021-09-27 11:32:29 +08:00
void * user_ctx ; /*!< User private data, passed directly to on_color_trans_done's user_ctx */
2021-08-04 20:11:31 +08:00
int lcd_cmd_bits ; /*!< Bit-width of LCD command */
int lcd_param_bits ; /*!< Bit-width of LCD parameter */
2021-05-12 11:26:07 +08:00
struct {
unsigned int dc_idle_level : 1 ; /*!< Level of DC line in IDLE phase */
unsigned int dc_cmd_level : 1 ; /*!< Level of DC line in CMD phase */
unsigned int dc_dummy_level : 1 ; /*!< Level of DC line in DUMMY phase */
unsigned int dc_data_level : 1 ; /*!< Level of DC line in DATA phase */
} dc_levels ; /*!< Each i80 device might have its own D/C control logic */
struct {
2021-08-04 20:11:31 +08:00
unsigned int cs_active_high : 1 ; /*!< If set, a high level of CS line will select the device, otherwise, CS line is low level active */
2021-05-12 11:26:07 +08:00
unsigned int reverse_color_bits : 1 ; /*!< Reverse the data bits, D[N:0] -> D[0:N] */
unsigned int swap_color_bytes : 1 ; /*!< Swap adjacent two color bytes */
unsigned int pclk_active_neg : 1 ; /*!< The display will write data lines when there's a falling edge on WR signal (a.k.a the PCLK) */
unsigned int pclk_idle_low : 1 ; /*!< The WR signal (a.k.a the PCLK) stays at low level in IDLE phase */
2022-05-11 15:52:09 +08:00
} flags ; /*!< Panel IO config flags */
2021-05-12 11:26:07 +08:00
} esp_lcd_panel_io_i80_config_t ;
/**
* @ brief Create LCD panel IO , for Intel 8080 interface
*
* @ param [ in ] bus Intel 8080 bus handle , created by ` esp_lcd_new_i80_bus ( ) `
* @ param [ in ] io_config IO configuration , for i80 interface
* @ param [ out ] ret_io Returned panel IO handle
* @ return
* - ESP_ERR_INVALID_ARG if parameter is invalid
* - ESP_ERR_NOT_SUPPORTED if some configuration can ' t be satisfied , e . g . pixel clock out of the range
* - ESP_ERR_NO_MEM if out of memory
* - ESP_OK on success
*/
esp_err_t esp_lcd_new_panel_io_i80 ( esp_lcd_i80_bus_handle_t bus , const esp_lcd_panel_io_i80_config_t * io_config , esp_lcd_panel_io_handle_t * ret_io ) ;
# endif // SOC_LCD_I80_SUPPORTED
# ifdef __cplusplus
}
# endif