本文章介紹LVGL繪圖函式庫如何結合本網站先前介紹的Raspberry Pi Pico 利用PIO製作的驅動程式。
- [Raspberry Pi Pico (c-sdk)] Display: Ep 5 :TFT LCD 4-lines Serial(SPI) Driver
- [Raspberry Pi Pico (c-sdk)] Display: Ep 4 : ILI9341 TFT LCD 8-bit parallel
本文章以LVGL 8.4.0為例:
一、LVGL Graphic library所需要的檔案
- src資料夾、
- lvgl.h
- lv_conf.h(由lv_conf_template.h複製)
修改lv_conf.h的內容:- /* clang-format off */
#if 1 /*Set it to "1" to enable content*/ - #define LV_COLOR_DEPTH 16
#define LV_COLOR_16_SWAP 1
CMakeLists.txt內容:二、週期性呼叫 lv_tick_inc(x)
bool lv_inc_timer_cb(repeating_timer_t *rt) { lv_tick_inc(*((uint8_t*)rt->user_data)); return true;}
void pico_lvgl_tick_inc_timer_init(uint8_t tick_inc) { static repeating_timer_t rt; static uint8_t _tick_inc; _tick_inc = tick_inc; add_repeating_timer_ms(tick_inc, lv_inc_timer_cb, (void*) (&_tick_inc), &rt);}
三、呼叫 lv_init()
四、製作draw buffer flush callback function, 將一個block資料寫到TFT Display memory中:void tft_lvgl_draw_bitmap(uint16_t x1, uint16_t y1, uint16_t x2, uint16_t y2, uint8_t *bitmap){ uint32_t total_pixels = (x2-x1+1) * (y2-y1+1)*2;
tft_set_address_window (x1, y1, x2, y2); tft_cmd_dma(TFT_MEMORYWRITE, total_pixels, bitmap); }
void tft_lvgl_disp_flush(lv_disp_drv_t * disp, const lv_area_t * area, lv_color_t * color_p){ tft_lvgl_draw_bitmap( (uint16_t)(area->x1), (uint16_t)(area->y1), (uint16_t)(area->x2), (uint16_t)(area->y2), (uint8_t*)color_p ); lv_disp_flush_ready(disp); /* Indicate you are ready with the flushing*/}其中tft_lvgl_draw_bitmap(...)為前篇文章介紹PIO TFT Display Driver的function。
五、以製作一組音樂播放器為展示範例:
六、程式碼:
- 更改 pico_tft.h:
/*=== according to TFT===*/
#define TFT_WIDTH 320
#define TFT_HEIGHT 480
#define PICO_TFT_SERIAL 1
#define PICO_TFT_PARALLEL 0
#define PICO_TFT_DMA 1
#define TFT_ST7735 0
#define TFT_ST7796 1
#define TFT_ILI9342 0
#define TFT_ILI9341 0
/*=== according to TFT===*/
以符合硬體
- 更改 xpt2046.h:
//== Set the following values according to the hardware ==
#define XPT2046_IRQ_GPIO 11
#define XPT2046_MOSI 15
#define XPT2046_MISO 12
#define XPT2046_CS 13
#define XPT2046_CLK 14
#define XPT2046_SPI spi1
#define XPT2046_MIN_RAW_X 1350
#define XPT2046_MAX_RAW_X 31000
#define XPT2046_MIN_RAW_Y 2050
#define XPT2046_MAX_RAW_Y 31500
//== Set the following values according to the hardware ==
以符合硬體
- lvgl
CmakeList.txt
option(LV_LVGL_H_INCLUDE_SIMPLE "Use #include \"lvgl.h\" instead of #include \"../../lvgl.h\"" ON) # Option to define LV_CONF_INCLUDE_SIMPLE, default: ON option(LV_CONF_INCLUDE_SIMPLE "Use #include \"lv_conf.h\" instead of #include \"../../lv_conf.h\"" ON) add_library(lvgl INTERFACE) file(GLOB_RECURSE SOURCES src/*.c src/*.S) target_sources(lvgl INTERFACE ${SOURCES} ) target_include_directories(lvgl INTERFACE ${CMAKE_CURRENT_LIST_DIR} ${CMAKE_CURRENT_LIST_DIR}/.. )
- pico_tft
CMakeLists.txt
add_library(pico_tft INTERFACE) pico_generate_pio_header(pico_tft ${CMAKE_CURRENT_LIST_DIR}/pico_tft.pio) target_sources(pico_tft INTERFACE ${CMAKE_CURRENT_LIST_DIR}/pico_tft.c ${CMAKE_CURRENT_LIST_DIR}/pico_tft_lvgl.c ) target_include_directories(pico_tft INTERFACE ${CMAKE_CURRENT_LIST_DIR} ${CMAKE_CURRENT_LIST_DIR}/.. ) target_link_libraries(pico_tft INTERFACE hardware_pio hardware_dma )
pico_tft.h
#ifndef _TFT_H_ #define _TFT_H_ /*=== according to TFT===*/ #define TFT_WIDTH 320 #define TFT_HEIGHT 480 #define PICO_TFT_SERIAL 1 #define PICO_TFT_PARALLEL 0 #define PICO_TFT_DMA 1 #define TFT_ST7735 0 #define TFT_ST7796 1 #define TFT_ILI9342 0 #define TFT_ILI9341 0 /*=== according to TFT===*/ enum { TFT_ORIENTATION_PORTRAIT=0, TFT_ORIENTATION_LANDSCAPE, TFT_ORIENTATION_PORTRAIT_MIRROR, TFT_ORIENTATION_LANDSCAPE_MIRROR, }; #include "pico/stdlib.h" #include "hardware/pio.h" #include "pico_tft_lvgl.h" void tft_init(PIO pio, uint sm, uint din_base, uint csx_dcx_sck_side_base_pin); void tft_init_config(); void tft_draw_pixel(uint16_t x, uint16_t y, uint16_t color); void tft_set_address_window(uint16_t x1, uint16_t y1, uint16_t x2, uint16_t y2); void tft_cmd(uint32_t cmd, uint32_t count, uint8_t *param); void tft_cmd_dma(uint32_t cmd, uint32_t count, uint8_t *param); uint16_t tft_color_565RGB(uint8_t R, uint8_t G, uint8_t B); //void tft_lv_draw_bitmap(uint16_t x, uint16_t y, uint16_t width, uint16_t height, uint8_t *bitmap); void tft_pio_cmd_init(PIO pio, uint sm, uint out_base, uint set_base, uint32_t freq); uint16_t tft_get_width(); uint16_t tft_get_height(); uint8_t tft_get_orientation(); void tft_set_orientation(uint8_t orientation); void tft_fill_rect(uint16_t x, uint16_t y, uint16_t width, uint16_t height, uint16_t color); #endif
pico_tft.c
#include "stdio.h" #include "stdlib.h" #include "pico/stdlib.h" #include "hardware/clocks.h" #include "string.h" #include "registers.h" #include "pico_tft.pio.h" #include "pico_tft.h" #include "hardware/dma.h" #define MAX_BYTE_TRANS (TFT_WIDTH*TFT_HEIGHT*2) // MADCTL register: MY,MX,MV,ML,BGR,MH,x,x #if TFT_ST7735 static uint8_t TFT_MADCTL_PORTRAIT = 0b11001000; static uint8_t TFT_MADCTL_LANDSCAPE = 0b10101000; static uint8_t TFT_MADCTL_PORTRAIT_MIRROR = 0b00001000; static uint8_t TFT_MADCTL_LANDSCAPE_MIRROR = 0b01101000; #endif #if TFT_ST7796 static uint8_t TFT_MADCTL_PORTRAIT = 0b01001000; static uint8_t TFT_MADCTL_LANDSCAPE = 0b00101000; static uint8_t TFT_MADCTL_PORTRAIT_MIRROR = 0b10001000; static uint8_t TFT_MADCTL_LANDSCAPE_MIRROR = 0b11101000; #endif #if TFT_ILI9341 // MADCTL register: MY,MX,MV,ML,BGR,MH,x,x static uint8_t TFT_MADCTL_PORTRAIT = 0b00000000; static uint8_t TFT_MADCTL_LANDSCAPE = 0b01100000; static uint8_t TFT_MADCTL_PORTRAIT_MIRROR = 0b11000000; static uint8_t TFT_MADCTL_LANDSCAPE_MIRROR = 0b10100000; #endif #if TFT_ILI9342 // MADCTL register: MY,MX,MV,ML,BGR,MH,x,x static uint8_t TFT_MADCTL_PORTRAIT = 0b00000000; static uint8_t TFT_MADCTL_LANDSCAPE = 0b01100000; static uint8_t TFT_MADCTL_PORTRAIT_MIRROR = 0b11000000; static uint8_t TFT_MADCTL_LANDSCAPE_MIRROR = 0b10100000; #endif static uint16_t tft_width; static uint16_t tft_height; static uint8_t tft_orientation; PIO tft_pio = pio1; uint tft_sm=0; uint in_out_base_pin=4; uint set_base_pin=12; uint sideset_base_pin=16; uint s_in_out_base_pin=19; int tft_dma_channel; void tft_cmd(uint32_t cmd, uint32_t count, uint8_t *param) { #if PICO_TFT_SERIAL pio_sm_put_blocking(tft_pio, tft_sm, cmd << 24); #endif #if PICO_TFT_PARALLEL pio_sm_put_blocking(tft_pio, tft_sm, cmd); #endif pio_sm_put_blocking(tft_pio, tft_sm, count); for (int i = 0; i < count; i++) { #if PICO_TFT_SERIAL pio_sm_put_blocking(tft_pio, tft_sm, param[i]<<24); #endif #if PICO_TFT_PARALLEL pio_sm_put_blocking(tft_pio, tft_sm, param[i]); #endif } } //#ifdef PICO_TFT_DMA void tft_cmd_dma(uint32_t cmd, uint32_t count, uint8_t *param) { #if PICO_TFT_SERIAL tft_cmd(cmd, count, param); return; #endif #if PICO_TFT_PARALLEL #if PICO_TFT_DMA pio_sm_put_blocking(tft_pio, tft_sm, cmd); pio_sm_put_blocking(tft_pio, tft_sm, count); dma_channel_set_trans_count(tft_dma_channel, count >> DMA_SIZE_8, false); dma_channel_set_read_addr(tft_dma_channel, param, false); dma_channel_start(tft_dma_channel); dma_channel_wait_for_finish_blocking(tft_dma_channel); #else tft_cmd(cmd, count, param); #endif #endif } //#endif void tft_pio_cmd_init(PIO pio, uint sm, uint in_out_base, uint set_sideset, uint32_t freq) { uint offset=0; pio_sm_config c; #if PICO_TFT_PARALLEL offset = pio_add_program(pio, &tft_pio_parallel_program); c = tft_pio_parallel_program_get_default_config(offset); for (int i=0; i < 8; i++) pio_gpio_init(pio, in_out_base+i); for (int i=0; i < 4; i++) pio_gpio_init(pio, set_sideset+i); pio_sm_set_consecutive_pindirs(pio, sm, in_out_base, 8, true); pio_sm_set_consecutive_pindirs(pio, sm, set_sideset, 4, true); sm_config_set_in_pins(&c, in_out_base); sm_config_set_out_pins(&c, in_out_base, 8); sm_config_set_set_pins(&c, set_sideset, 4); sm_config_set_out_shift(&c, true, false, 8); sm_config_set_in_shift(&c, false, false, 8); #endif #if PICO_TFT_SERIAL offset = pio_add_program(pio, &tft_pio_serial_program); c = tft_pio_serial_program_get_default_config(offset); pio_gpio_init(pio, in_out_base); for (int i=0; i < 3; i++) pio_gpio_init(pio, set_sideset+i); pio_sm_set_consecutive_pindirs(pio, sm, in_out_base, 1, true); pio_sm_set_consecutive_pindirs(pio, sm, set_sideset, 3, true); sm_config_set_in_pins(&c, in_out_base); sm_config_set_out_pins(&c, in_out_base, 1); sm_config_set_sideset_pins(&c, set_sideset); sm_config_set_out_shift(&c, false, false, 8); sm_config_set_in_shift(&c, true, false, 8); #endif //sm_config_set_fifo_join(&c, PIO_FIFO_JOIN_TX); float div = (float)clock_get_hz(clk_sys)/freq; sm_config_set_clkdiv(&c, div); //sm_config_set_clkdiv(&c, 1.25); #if PICO_TFT_DMA /* DMA */ tft_dma_channel = dma_claim_unused_channel(true); dma_channel_config dc = dma_channel_get_default_config(tft_dma_channel); channel_config_set_write_increment(&dc, false); channel_config_set_read_increment(&dc, true); channel_config_set_dreq(&dc, pio_get_dreq(pio, sm, true)); channel_config_set_transfer_data_size(&dc, DMA_SIZE_8); //DMA_SIZE_8,16,32 uint32_t pio_base, tx_offset; pio_base = (pio == pio0) ? PIO0_BASE: PIO1_BASE; tx_offset = PIO_TXF0_OFFSET + sm*4; dma_channel_configure(tft_dma_channel, &dc, (void*) (pio_base + tx_offset), NULL, MAX_BYTE_TRANS>> DMA_SIZE_8, false); //DMA_SIZE_8 or 16 or 32 /* DMA */ #endif pio_sm_init(pio, sm, offset, &c); pio_sm_set_enabled(pio, sm, true); } /* tft draw functions*/ uint16_t tft_color_565RGB(uint8_t R, uint8_t G, uint8_t B) { uint16_t c; c = (((uint16_t)R)>>3)<<11 | (((uint16_t)G)>>2) << 5 | ((uint16_t)B)>>3; return c; } void tft_memory_write_window(uint16_t x1, uint16_t y1, uint16_t x2, uint16_t y2) { uint8_t addr[4]; addr[0]=(uint8_t)(x1 >> 8); addr[1]= (uint8_t)(x1 & 0xff); addr[2]= (uint8_t)(x2 >> 8); addr[3]= (uint8_t)(x2 & 0xff); tft_cmd(TFT_COLADDRSET, 4, addr); addr[0]=(uint8_t)(y1 >> 8); addr[1]= (uint8_t)(y1 & 0xff); addr[2]= (uint8_t)(y2 >> 8); addr[3]= (uint8_t)(y2 & 0xff); tft_cmd(TFT_PAGEADDRSET, 4, addr ); tft_cmd(TFT_MEMORYWRITE, 0, NULL); } void tft_set_address_window(uint16_t x1, uint16_t y1, uint16_t x2, uint16_t y2) { uint8_t addr[4]; addr[0]=(uint8_t)(x1 >> 8); addr[1]= (uint8_t)(x1 & 0xff); addr[2]= (uint8_t)(x2 >> 8); addr[3]= (uint8_t)(x2 & 0xff); tft_cmd(TFT_COLADDRSET, 4, addr); addr[0]=(uint8_t)(y1 >> 8); addr[1]= (uint8_t)(y1 & 0xff); addr[2]= (uint8_t)(y2 >> 8); addr[3]= (uint8_t)(y2 & 0xff); tft_cmd(TFT_PAGEADDRSET, 4, addr ); } /* put color at point*/ void tft_draw_pixel(uint16_t x, uint16_t y, uint16_t color) { if ( x < 0 || x > TFT_WIDTH-1 || y < 0 || y > TFT_HEIGHT-1) { printf("over range,x,y\n"); return; } tft_set_address_window(x,y,x,y); tft_cmd(TFT_MEMORYWRITE, 2, (uint8_t[2]){(uint8_t)(color >> 8), (uint8_t)color}); } uint16_t tft_get_width() { return tft_width; } uint16_t tft_get_height() { return tft_height; } uint8_t tft_get_orientation() { return tft_orientation; } void tft_set_orientation(uint8_t orientation) { tft_orientation = orientation; switch (orientation) { case TFT_ORIENTATION_PORTRAIT: tft_cmd(TFT_MADCTL, 1, (uint8_t[1]){TFT_MADCTL_PORTRAIT}); tft_width = TFT_WIDTH; tft_height = TFT_HEIGHT; break; case TFT_ORIENTATION_PORTRAIT_MIRROR: tft_cmd(TFT_MADCTL, 1, (uint8_t[1]){TFT_MADCTL_PORTRAIT_MIRROR}); tft_width = TFT_WIDTH; tft_height = TFT_HEIGHT; break; case TFT_ORIENTATION_LANDSCAPE: tft_cmd(TFT_MADCTL, 1, (uint8_t[1]){TFT_MADCTL_LANDSCAPE}); tft_width = TFT_HEIGHT; tft_height = TFT_WIDTH; break; case TFT_ORIENTATION_LANDSCAPE_MIRROR: tft_cmd(TFT_MADCTL, 1, (uint8_t[1]){TFT_MADCTL_LANDSCAPE_MIRROR}); tft_width = TFT_HEIGHT; tft_height = TFT_WIDTH; break; } } #if TFT_ST7735 void tft_init_config() { tft_cmd(TFT_SOFTRESET, 0, NULL); sleep_ms(150); tft_cmd(TFT_DISPLAYOFF, 0, NULL); sleep_ms(150); tft_cmd(TFT_PIXELFORMAT, 1, (uint8_t[1]){0x55}); //0x55 tft_cmd(TFT_POWERCONTROL1, 1, (uint8_t[1]){0x05}); // 0x05 :3.3V tft_cmd(TFT_POWERCONTROL2, 1, (uint8_t[1]){0x10}); tft_cmd(TFT_VCOMCONTROL1, 2, (uint8_t[2]){0x3E, 0x28}); tft_cmd(TFT_VCOMCONTROL2, 1, (uint8_t[1]){0x86}); //tft_cmd(TFT_MADCTL, 1, (uint8_t[1]){0x08}); //MY,MX,MV,ML,BRG,MH,0,0(40) tft_set_orientation(TFT_ORIENTATION_PORTRAIT); tft_cmd(TFT_FRAMECONTROL, 2, (uint8_t[2]){0b00, 0x1B}); // Default 70Hz:0x1B tft_cmd(TFT_DISPLAYFUNC, 4, (uint8_t[4]){0x0A, 0x82, 0x27, 0x04}); //0a,a2,27,04 tft_cmd(TFT_GAMMASET, 1, (uint8_t[1]){0x01}); //tft_cmd(tft_PGAMCOR, 15, (uint8_t[15]){ 0x0f, 0x31, 0x2b, 0x0c, 0x0e, 0x08, 0x4e, 0xf1, 0x37, 0x07, 0x10, 0x03, 0x0e, 0x09, 0x00 }); //tft_cmd(tft_NGAMCOR, 15,0xFF, (uint8_t[15]){ 0x00, 0x0e, 0x14, 0x03, 0x11, 0x07, 0x31, 0xc1, 0x48, 0x08, 0x0f, 0x0c, 0x31, 0x36, 0x0f }); tft_cmd(TFT_SLEEPOUT, 0, NULL); sleep_ms(150); tft_cmd(TFT_DISPLAYON, 0, NULL); sleep_ms(500); } #endif #if TFT_ST7796 void tft_init_config() { //tft_cmd(TFT_SOFTRESET, 1, NULL); tft_cmd(TFT_SOFTRESET, 0, NULL); sleep_ms(120); tft_cmd(TFT_SLEEPOUT, 0, NULL); sleep_ms(120); tft_cmd(TFT_COMMANDSET, 1, (uint8_t[1]){0xC3}); //enable part 1 tft_cmd(TFT_COMMANDSET, 1, (uint8_t[1]){0x96}); //enable part 2 //tft_cmd(TFT_MADCTL, 1, (uint8_t[1]){0x88}); //MY,MX,MV,ML,BRG,MH,0,0(24), 0:RGB tft_set_orientation(TFT_ORIENTATION_PORTRAIT); tft_cmd(TFT_PIXELFORMAT, 1, (uint8_t[1]){0x05}); //0x05:RGB565, 0x06 RGB666 tft_cmd(TFT_DSIPLAY_INVER, 1, (uint8_t[1]){0x01}); // 1-dot #if TFT_ILI9342 tft_cmd(TFT_DISPLAYFUNC, 3, (uint8_t[3]){0x0A, 0x82, 0x27}); // ILI9342 #endif #if TFT_ST7796 tft_cmd(TFT_DISPLAYFUNC, 3, (uint8_t[3]){0x80, 0x02, 0x3B}); // ST7796 #endif tft_cmd(TFT_DISP_OUTPUT_CTRL_ADJUST, 8, (uint8_t[8]) {0x40, 0x8A, 0x00, 0x00, 0x29, //Source eqaulizing period time= 22.5 us 0x19, //Timing for "Gate start"=25 (Tclk) 0xA5, //Timing for "Gate End"=37 (Tclk), Gate driver EQ function ON 0x33}); // ST7796 tft_cmd(TFT_POWERCONTROL2, 1, (uint8_t[1]){0x06}); // 0x05 :3.3V tft_cmd(TFT_POWERCONTROL3, 1, (uint8_t[1]){0xA7}); tft_cmd(TFT_VCOMCONTROL1, 1, (uint8_t[1]){0x18}); sleep_ms(120); tft_cmd(TFT_PGAMCOR, 14, (uint8_t[14]){ 0xf0, 0x09, 0x0b, 0x06, 0x04, 0x15, 0x2f, 0x54, 0x42, 0x3c, 0x17, 0x14, 0x18, 0x1b}); tft_cmd(TFT_NGAMCOR, 14, (uint8_t[14]){ 0xe0, 0x09, 0x0b, 0x06, 0x04, 0x03, 0x2b, 0x43, 0x42, 0x3b, 0x16, 0x14, 0x17, 0x1b}); sleep_ms(120); tft_cmd(TFT_COMMANDSET, 1, (uint8_t[1]){0x3C}); // disable part 1 tft_cmd(TFT_COMMANDSET, 1, (uint8_t[1]){0x69}); // disable part 2 tft_cmd(TFT_DISPLAYOFF, 0, NULL); sleep_ms(120); tft_cmd(TFT_DISPLAYON, 0, NULL); sleep_ms(500); } #endif #if TFT_ILI9342 void tft_init_config() { tft_cmd(TFT_SOFTRESET, 0, NULL); sleep_ms(150); tft_cmd(TFT_DISPLAYOFF, 0, NULL); sleep_ms(150); tft_cmd(TFT_PIXELFORMAT, 1, (uint8_t[1]){0x55}); //0x55 tft_cmd(TFT_POWERCONTROL1, 1, (uint8_t[1]){0x05}); // 0x05 :3.3V tft_cmd(TFT_POWERCONTROL2, 1, (uint8_t[1]){0x10}); tft_cmd(TFT_VCOMCONTROL1, 2, (uint8_t[2]){0x3E, 0x28}); tft_cmd(TFT_VCOMCONTROL2, 1, (uint8_t[1]){0x86}); tft_cmd(TFT_MADCTL, 1, (uint8_t[1]){0x60}); //MY,MX,MV,ML,BRG,MH,0,0(40) tft_set_orientation(TFT_ORIENTATION_PORTRAIT); tft_cmd(TFT_FRAMECONTROL, 2, (uint8_t[2]){0x00, 0x1B}); // Default 70Hz tft_cmd(TFT_DISPLAYFUNC, 4, (uint8_t[4]){0x0A, 0x82, 0x27, 0x04}); //0a,a2,27,04 tft_cmd(TFT_GAMMASET, 1, (uint8_t[1]){0x01}); tft_cmd(TFT_PGAMCOR, 15, (uint8_t[15]){ 0x0f, 0x31, 0x2b, 0x0c, 0x0e, 0x08, 0x4e, 0xf1, 0x37, 0x07, 0x10, 0x03, 0x0e, 0x09, 0x00 }); tft_cmd(TFT_NGAMCOR, 15, (uint8_t[15]){ 0x00, 0x0e, 0x14, 0x03, 0x11, 0x07, 0x31, 0xc1, 0x48, 0x08, 0x0f, 0x0c, 0x31, 0x36, 0x0f }); tft_cmd(TFT_SLEEPOUT, 0, NULL); sleep_ms(150); tft_cmd(TFT_DISPLAYON, 0, NULL); sleep_ms(500); } #endif #if TFT_ILI9341 void tft_init_config() { tft_cmd(TFT_SOFTRESET, 0, NULL); sleep_ms(50); tft_cmd(TFT_DISPLAYOFF, 0, NULL); tft_cmd(TFT_POWERCONTROL1, 1, (uint8_t[1]){0x23}); tft_cmd(TFT_POWERCONTROL2, 1, (uint8_t[1]){0x10}); tft_cmd(TFT_VCOMCONTROL1, 2, (uint8_t[2]){0x2B, 0x2B}); tft_cmd(TFT_VCOMCONTROL2, 1, (uint8_t[1]){0xC0}); tft_cmd(TFT_MADCTL, 1, (uint8_t[1]){0x88}); //MY,MX,MV,ML,BRG,MH,0,0(40) tft_set_orientation(TFT_ORIENTATION_PORTRAIT); tft_cmd(TFT_PIXELFORMAT, 1, (uint8_t[1]){0x55}); tft_cmd(TFT_FRAMECONTROL, 2, (uint8_t[2]){0x00, 0x1B}); tft_cmd(TFT_ENTRYMODE, 1, (uint8_t[1]){0x07}); //tft_cmd(TFT_DISPLAYFUNC, 4, (uint8_t[4]){0x0A, 0x82, 0x27, 0x00}); tft_cmd(TFT_SLEEPOUT, 0, NULL); sleep_ms(150); tft_cmd(TFT_DISPLAYON, 0, NULL); sleep_ms(500); } #endif void tft_init(PIO pio, uint sm, uint din_base, uint csx_dcx_sck_side_base_pin) { tft_pio = pio; tft_sm = sm; #if PICO_TFT_PARALLEL in_out_base_pin = din_base; set_base_pin = csx_dcx_sck_side_base_pin; tft_pio_cmd_init(tft_pio, tft_sm, in_out_base_pin, set_base_pin, 70000000/*70000000*/); //pio freq #endif #if PICO_TFT_SERIAL s_in_out_base_pin = din_base; sideset_base_pin = csx_dcx_sck_side_base_pin; tft_pio_cmd_init(tft_pio, tft_sm, s_in_out_base_pin, sideset_base_pin, 90000000/* 62.5M baud rate for SPI*/); //pio freq #endif tft_init_config(); }
pico_tft_lvgl.h
#ifndef __PICO_TFT_LVGL__ #define __PICO_TFT_LVGL__ #include "pico_tft.h" #include "lvgl.h" void tft_lvgl_draw_bitmap(uint16_t x, uint16_t y, uint16_t width, uint16_t height, uint8_t *bitmap); void tft_lvgl_disp_flush(lv_disp_drv_t * disp, const lv_area_t * area, lv_color_t * color_p); uint32_t pico_lvgl_get_tick_cb(); #endif
pico_tft_lvgl.c#include "pico_tft.h" #include "pico_tft_lvgl.h" #include "registers.h" #include "stdio.h" void tft_lvgl_draw_bitmap(uint16_t x1, uint16_t y1, uint16_t x2, uint16_t y2, uint8_t *bitmap) { uint32_t total_pixels = (x2-x1+1) * (y2-y1+1)*2; tft_set_address_window (x1, y1, x2, y2); tft_cmd_dma(TFT_MEMORYWRITE, total_pixels, bitmap); } void tft_lvgl_disp_flush(lv_disp_drv_t * disp, const lv_area_t * area, lv_color_t * color_p) { tft_lvgl_draw_bitmap( (uint16_t)(area->x1), (uint16_t)(area->y1), (uint16_t)(area->x2), (uint16_t)(area->y2), (uint8_t*)color_p ); lv_disp_flush_ready(disp); /* Indicate you are ready with the flushing*/ } /* tell how many milliseconds have elapsed since start up for lvgl 9.x.x */ uint32_t pico_lvgl_get_tick_cb() { return (time_us_32()/1000); }
pico_tft.pio.program tft_pio_parallel .wrap_target start: pull set pins, 0b0011 mov x, osr ;command code, if 0x0, command nop, only data jmp !x param set pins, 0b0001 out pins, 8 [1] set pins, 0b0011 [1] param: set pins, 0b0111 pull mov x, osr ;how many parameters jmp !x, start ;no parameter return start jmp x--, param_data param_data: pull ; write data set pins, 0b0101 out pins, 8 [1] set pins, 0b0111 [1] jmp x--, param_data set pins, 0b1111 jmp start .wrap .program tft_pio_serial ; CSX, D/CX(A0), SCL --> gpio 13, 12, 11 (side-set pins) .side_set 3 .wrap_target start: pull side 0b100 mov x, osr side 0b000 ;command code, if 0x0, command nop, only data jmp !x param side 0b000 set y,7 side 0b000 cmd_bit_loop: out pins, 1 side 0b000 [1] jmp y--, cmd_bit_loop side 0b001 [1] param: pull side 0b010 mov x, osr side 0b010 ;how many parameters jmp !x, start side 0b010 ;no parameter return start jmp x--, param_data side 0b010 param_data: pull side 0b010 ; write data set y,7 side 0b010 data_bit_loop: out pins, 1 side 0b010 [1] jmp y--, data_bit_loop side 0b011 [1] jmp x--, param_data side 0b010 jmp start side 0b100
- registers.h
#define TFT_NOP 0x00 #define TFT_SOFTRESET 0x01 #define TFT_SLEEPIN 0x10 #define TFT_SLEEPOUT 0x11 #define TFT_NORMALDISP 0x13 #define TFT_INVERTOFF 0x20 #define TFT_INVERTON 0x21 #define TFT_GAMMASET 0x26 #define TFT_DISPLAYOFF 0x28 #define TFT_DISPLAYON 0x29 #define TFT_COLADDRSET 0x2A #define TFT_PAGEADDRSET 0x2B #define TFT_MEMORYWRITE 0x2C #define TFT_MEMORYREAD 0x2E #define TFT_PIXELFORMAT 0x3A #define TFT_MEMORYWRITECONT 0x3C #define TFT_MEMORYREADCONT 0x3E #define TFT_FRAMECONTROL 0xB1 #define TFT_DSIPLAY_INVER 0xB4 #define TFT_DISPLAYFUNC 0xB6 #define TFT_ENTRYMODE 0xB7 #define TFT_POWERCONTROL1 0xC0 #define TFT_POWERCONTROL2 0xC1 #define TFT_POWERCONTROL3 0xC2 #define TFT_VCOMCONTROL1 0xC5 #define TFT_VCOMCONTROL2 0xC7 #define TFT_COMMANDSET 0xF0 #define TFT_VSCROLLDEF 0x33 #define TFT_VSCROLLADDR 0x37 #define TFT_MEMCONTROL 0x36 #define TFT_MADCTL 0x36 #define TFT_PGAMCOR 0xE0 #define TFT_NGAMCOR 0xE1 #define TFT_DISP_OUTPUT_CTRL_ADJUST 0xE8 #define TFT_MADCTL_MY 0x80 #define TFT_MADCTL_MX 0x40 #define TFT_MADCTL_MV 0x20 #define TFT_MADCTL_ML 0x10 #define TFT_MADCTL_RGB 0x00 #define TFT_MADCTL_BGR 0x08 #define TFT_MADCTL_MH 0x04
- XPT2046_touch
CMakeLists.txt
add_library(xpt2046_touch INTERFACE) target_sources(xpt2046_touch INTERFACE ${CMAKE_CURRENT_LIST_DIR}/xpt2046.c ) target_include_directories(xpt2046_touch INTERFACE ${CMAKE_CURRENT_LIST_DIR} ${CMAKE_CURRENT_LIST_DIR}/.. ) target_link_libraries(xpt2046_touch INTERFACE hardware_spi hardware_dma )
xpt2046.h
#ifndef _XPT2046_H_ #define _XPT2046_H_ #include "pico/stdlib.h" #include "lvgl.h" //== Set the following values according to the hardware == #define XPT2046_IRQ_GPIO 11 #define XPT2046_MOSI 15 #define XPT2046_MISO 12 #define XPT2046_CS 13 #define XPT2046_CLK 14 #define XPT2046_SPI spi1 #define XPT2046_MIN_RAW_X 1350 #define XPT2046_MAX_RAW_X 31000 #define XPT2046_MIN_RAW_Y 2050 #define XPT2046_MAX_RAW_Y 31500 //== Set the following values according to the hardware == void xpt2046_init(); bool xpt2046_getXY(uint16_t *x, uint16_t *y); bool xpt2046_TouchPressed(); void xpt2046_lvgl_read_cb(struct _lv_indev_drv_t * indev, lv_indev_data_t* data); #endif
xpt2046.c#include "stdio.h" #include "stdlib.h" #include "xpt2046.h" #include "hardware/spi.h" #include "hardware/gpio.h" #include "pico/stdlib.h" #include "pico_tft.h" uint32_t xpt2046_event=0; uint8_t READ_X = 0xD0; uint8_t READ_Y = 0x90; #define XPT_WIDTH 320 #define XPT_HEIGHT 480 bool xpt2046_getXY(uint16_t *x, uint16_t *y) { uint8_t temp[2]; uint16_t raw_x, raw_y; uint16_t est_raw_x, est_raw_y; uint32_t avg_x = 0; uint32_t avg_y = 0; uint8_t nsamples = 0; uint8_t SAMPLES=10; gpio_put(XPT2046_CS, false); if(gpio_get(XPT2046_IRQ_GPIO)) return false; busy_wait_ms(10); //first pass SAMPLES=20; // get first average; for(uint8_t i = 0; i < SAMPLES; i++, nsamples++) { if(gpio_get(XPT2046_IRQ_GPIO)) { break; } spi_write_blocking(XPT2046_SPI, &READ_X, 1); spi_read_blocking(XPT2046_SPI, 0x00, temp, 2); raw_x = ((uint16_t)temp[0]) << 8 | (uint16_t)temp[1]; spi_write_blocking(XPT2046_SPI, &READ_Y, 1); spi_read_blocking(XPT2046_SPI, 0x00, temp, 2); raw_y = ((uint16_t)temp[0]) << 8 | (uint16_t)temp[1]; avg_x += raw_x; avg_y += raw_y; } if(nsamples < SAMPLES) return false; gpio_put(XPT2046_CS, true); raw_x = (avg_x / SAMPLES); raw_y = (avg_y / SAMPLES); if(raw_x < XPT2046_MIN_RAW_X || raw_x > XPT2046_MAX_RAW_X) return false; if(raw_y < XPT2046_MIN_RAW_Y || raw_y > XPT2046_MAX_RAW_Y) return false; uint16_t tx,ty; tx = (raw_x - XPT2046_MIN_RAW_X) * XPT_WIDTH / (XPT2046_MAX_RAW_X - XPT2046_MIN_RAW_X); ty = (raw_y - XPT2046_MIN_RAW_Y) * XPT_HEIGHT / (XPT2046_MAX_RAW_Y - XPT2046_MIN_RAW_Y); // adjust for TFT orientation uint8_t lot = tft_get_orientation(); switch (lot) { case TFT_ORIENTATION_PORTRAIT: *x=tx; *y=TFT_HEIGHT-ty; break; case TFT_ORIENTATION_LANDSCAPE: *x=TFT_HEIGHT-ty; *y=TFT_WIDTH-tx; break; case TFT_ORIENTATION_PORTRAIT_MIRROR: *x=TFT_WIDTH-tx; *y=ty; break; case TFT_ORIENTATION_LANDSCAPE_MIRROR: *x=ty; *y=tx; break; } return true; } bool xpt2046_TouchPressed() { return !gpio_get(XPT2046_IRQ_GPIO); } void xpt2046_init() { gpio_init(XPT2046_IRQ_GPIO); gpio_init(XPT2046_MISO); gpio_init(XPT2046_MOSI); gpio_init(XPT2046_CLK); gpio_init(XPT2046_CS); gpio_set_dir(XPT2046_CS, GPIO_OUT); gpio_set_dir(XPT2046_IRQ_GPIO, GPIO_OUT); gpio_set_function(XPT2046_CLK,GPIO_FUNC_SPI); gpio_set_function(XPT2046_CS,GPIO_FUNC_SIO); gpio_set_function(XPT2046_MOSI,GPIO_FUNC_SPI); gpio_set_function(XPT2046_MISO,GPIO_FUNC_SPI); //spi_init(XPT2046_SPI,250000); spi_init(XPT2046_SPI,3125000); //3125000 } extern lv_obj_t *mouse_cursor; void xpt2046_lvgl_read_cb(struct _lv_indev_drv_t * indev, lv_indev_data_t* data) { uint16_t x,y; if (xpt2046_TouchPressed()) { if (xpt2046_getXY(&x,&y)) { data->point.x = x; data->point.y = y; data->state = LV_INDEV_STATE_PRESSED; } else { data->state = LV_INDEV_STATE_RELEASED; } } else { data->state = LV_INDEV_STATE_RELEASED; } }
- pico_lvgl
CMakeLists.txt
add_library(pico_lvgl INTERFACE) target_sources(pico_lvgl INTERFACE ${CMAKE_CURRENT_LIST_DIR}/pico_lvgl.c ) add_subdirectory(pico_tft) add_subdirectory(xpt2046_touch) add_subdirectory(lvgl) target_include_directories(pico_lvgl INTERFACE ${CMAKE_CURRENT_LIST_DIR} ${CMAKE_CURRENT_LIST_DIR}/pico_tft ${CMAKE_CURRENT_LIST_DIR}/xpt2046_touch ${CMAKE_CURRENT_LIST_DIR}/lvgl ) target_link_libraries(pico_lvgl INTERFACE hardware_pio hardware_dma pico_tft xpt2046_touch lvgl )
pico_lvgl.c
#include "pico_lvgl.h" bool lv_inc_timer_cb(repeating_timer_t *rt) { lv_tick_inc(*((uint8_t*)rt->user_data)); return true; } /* TFT_SERIAL: @param sdi_gpio: SDI(MOSI) @param csx_dcx_sck_gpio: csx:GPIO_n+2,dcx:GPIO_n+1, sck:GPIO_n TFT_PARALLEL: @param sdi_gpio: LCD_D0(DB0): GPIO_dn, LCD_D1(DB1):GPIO_dn+1,...LCD_D7(DB7):GPIO_dn+7 @param csx_dcx_sck_gpio: CS:GPIO_n+3,RS:GPIO_n+2, WR:GPIO_n+1, RD:GPIO_n */ void pico_lvgl_tft_init(PIO pio, uint sm, uint sdi_gpio, uint csx_dcx_sck_gpio) { tft_init(pio , sm, sdi_gpio, csx_dcx_sck_gpio); } void pico_lvgl_tick_inc_timer_init(uint8_t tick_inc) { static repeating_timer_t rt; static uint8_t _tick_inc; _tick_inc = tick_inc; add_repeating_timer_ms(tick_inc, lv_inc_timer_cb, (void*) (&_tick_inc), &rt); } /* tick_inc: to call the lv_tick_inc(x) function periodically in (x) milliseconds. */ void pico_lvgl_display_init(uint8_t tick_inc) { // init timer if (tick_inc > 10) tick_inc = 10; if (tick_inc < 1) tick_inc = 1; pico_lvgl_tick_inc_timer_init(tick_inc); // lv_init lv_init(); // set draw buffer static lv_disp_draw_buf_t draw_buf; static lv_color_t buf1[TFT_WIDTH * TFT_HEIGHT / 10]; /*Declare a buffer for 1/10 screen size*/ lv_disp_draw_buf_init(&draw_buf, buf1, NULL, TFT_WIDTH * TFT_HEIGHT / 10); /*Initialize the display buffer.*/ //setup display static lv_disp_drv_t disp_drv; /*Descriptor of a display driver*/ lv_disp_drv_init(&disp_drv); /*Basic initialization*/ disp_drv.flush_cb = tft_lvgl_disp_flush; /*Set your driver function*/ disp_drv.draw_buf = &draw_buf; /*Assign the buffer to the display*/ disp_drv.hor_res = tft_get_width(); /*Set the horizontal resolution of the display*/ disp_drv.ver_res = tft_get_height(); /*Set the vertical resolution of the display*/ lv_disp_drv_register(&disp_drv); /*Finally register the driver*/ } void pico_lvgl_xpt2046_init() { // init xpt2046 hardware xpt2046_init(); // setup touch screen input device static lv_indev_drv_t indev_drv; /*Descriptor of a input device driver*/ lv_indev_drv_init(&indev_drv); /*Basic initialization*/ indev_drv.type = LV_INDEV_TYPE_POINTER; /*Touch pad is a pointer-like device*/ indev_drv.read_cb = xpt2046_lvgl_read_cb; /*Set your driver function*/ lv_indev_drv_register(&indev_drv); /*Finally register the driver*/ } lv_indev_t *encoder_indev; lv_group_t *encoder_group; void pico_lvgl_encoder_init(bool pio_mode) { // init rotary encoder hardware if (pio_mode) { lvgl_pio_encoder_init(); } else { gpio_encoder_init(); } // setup rotary encoder input device static lv_indev_drv_t indev_drv; /*Descriptor of a input device driver*/ lv_indev_drv_init(&indev_drv); /*Basic initialization*/ indev_drv.type = LV_INDEV_TYPE_ENCODER; /*Rotary Encoder device*/ if (pio_mode) { indev_drv.read_cb = lvgl_encoder_read_cb; /*Set your driver function*/ } else { indev_drv.read_cb = gpio_encoder_read_cb; } //indev_drv.long_press_time=1000; indev_drv.long_press_repeat_time=2000; encoder_indev = lv_indev_drv_register(&indev_drv); /*Finally register the driver*/ encoder_group = lv_group_create(); //lv_group_set_default(kg); lv_indev_set_group(encoder_indev, encoder_group); }
pico_lvgl.h
#ifndef __PICO_LVGL_H__ #define __PICO_LVGL_H__ #include "stdio.h" #include "pico/stdio.h" #include "hardware/pio.h" #include "pico_tft.h" #include "xpt2046.h" #include "lvgl.h" void pico_lvgl_tft_init(PIO pio, uint sm, uint sdi_gpio, uint csx_dcx_sck_gpio); void pico_lvgl_xpt2046_init(); void pico_lvgl_display_init(uint8_t tick_inc); #endif
- root
CMakeLists.txt
add_subdirectory(pico_lvgl) target_link_libraries(project_name pico_lvgl )
- src資料夾、
- lvgl.h
- lv_conf.h(由lv_conf_template.h複製)
修改lv_conf.h的內容:
- /* clang-format off */
#if 1 /*Set it to "1" to enable content*/ - #define LV_COLOR_DEPTH 16
#define LV_COLOR_16_SWAP 1
CMakeLists.txt內容:
二、週期性呼叫 lv_tick_inc(x)bool lv_inc_timer_cb(repeating_timer_t *rt) {
lv_tick_inc(*((uint8_t*)rt->user_data));
return true;
}
void pico_lvgl_tick_inc_timer_init(uint8_t tick_inc) {
static repeating_timer_t rt;
static uint8_t _tick_inc;
_tick_inc = tick_inc;
add_repeating_timer_ms(tick_inc, lv_inc_timer_cb, (void*) (&_tick_inc), &rt);
}
四、製作draw buffer flush callback function, 將一個block資料寫到TFT Display memory中:
void tft_lvgl_draw_bitmap(uint16_t x1, uint16_t y1, uint16_t x2, uint16_t y2, uint8_t *bitmap)
{
uint32_t total_pixels = (x2-x1+1) * (y2-y1+1)*2;
tft_set_address_window (x1, y1, x2, y2);
tft_cmd_dma(TFT_MEMORYWRITE, total_pixels, bitmap);
}
void tft_lvgl_disp_flush(lv_disp_drv_t * disp, const lv_area_t * area, lv_color_t * color_p)
{
tft_lvgl_draw_bitmap(
(uint16_t)(area->x1),
(uint16_t)(area->y1),
(uint16_t)(area->x2),
(uint16_t)(area->y2), (uint8_t*)color_p
);
lv_disp_flush_ready(disp); /* Indicate you are ready with the flushing*/
}
五、以製作一組音樂播放器為展示範例:
- 更改 pico_tft.h:
/*=== according to TFT===*/
#define TFT_WIDTH 320
#define TFT_HEIGHT 480
#define PICO_TFT_SERIAL 1
#define PICO_TFT_PARALLEL 0
#define PICO_TFT_DMA 1
#define TFT_ST7735 0
#define TFT_ST7796 1
#define TFT_ILI9342 0
#define TFT_ILI9341 0
/*=== according to TFT===*/
以符合硬體
- 更改 xpt2046.h:
//== Set the following values according to the hardware ==
#define XPT2046_IRQ_GPIO 11
#define XPT2046_MOSI 15
#define XPT2046_MISO 12
#define XPT2046_CS 13
#define XPT2046_CLK 14
#define XPT2046_SPI spi1
#define XPT2046_MIN_RAW_X 1350
#define XPT2046_MAX_RAW_X 31000
#define XPT2046_MIN_RAW_Y 2050
#define XPT2046_MAX_RAW_Y 31500
//== Set the following values according to the hardware ==
以符合硬體
- lvgl
CmakeList.txt
pico_tft.h
pico_tft.c
pico_tft_lvgl.h
pico_tft_lvgl.c
pico_tft.pio
option(LV_LVGL_H_INCLUDE_SIMPLE "Use #include \"lvgl.h\" instead of #include \"../../lvgl.h\"" ON) # Option to define LV_CONF_INCLUDE_SIMPLE, default: ON option(LV_CONF_INCLUDE_SIMPLE "Use #include \"lv_conf.h\" instead of #include \"../../lv_conf.h\"" ON) add_library(lvgl INTERFACE) file(GLOB_RECURSE SOURCES src/*.c src/*.S) target_sources(lvgl INTERFACE ${SOURCES} ) target_include_directories(lvgl INTERFACE ${CMAKE_CURRENT_LIST_DIR} ${CMAKE_CURRENT_LIST_DIR}/.. )
- pico_tft
add_library(pico_tft INTERFACE) pico_generate_pio_header(pico_tft ${CMAKE_CURRENT_LIST_DIR}/pico_tft.pio) target_sources(pico_tft INTERFACE ${CMAKE_CURRENT_LIST_DIR}/pico_tft.c ${CMAKE_CURRENT_LIST_DIR}/pico_tft_lvgl.c ) target_include_directories(pico_tft INTERFACE ${CMAKE_CURRENT_LIST_DIR} ${CMAKE_CURRENT_LIST_DIR}/.. ) target_link_libraries(pico_tft INTERFACE hardware_pio hardware_dma )
pico_tft.h
#ifndef _TFT_H_ #define _TFT_H_ /*=== according to TFT===*/ #define TFT_WIDTH 320 #define TFT_HEIGHT 480 #define PICO_TFT_SERIAL 1 #define PICO_TFT_PARALLEL 0 #define PICO_TFT_DMA 1 #define TFT_ST7735 0 #define TFT_ST7796 1 #define TFT_ILI9342 0 #define TFT_ILI9341 0 /*=== according to TFT===*/ enum { TFT_ORIENTATION_PORTRAIT=0, TFT_ORIENTATION_LANDSCAPE, TFT_ORIENTATION_PORTRAIT_MIRROR, TFT_ORIENTATION_LANDSCAPE_MIRROR, }; #include "pico/stdlib.h" #include "hardware/pio.h" #include "pico_tft_lvgl.h" void tft_init(PIO pio, uint sm, uint din_base, uint csx_dcx_sck_side_base_pin); void tft_init_config(); void tft_draw_pixel(uint16_t x, uint16_t y, uint16_t color); void tft_set_address_window(uint16_t x1, uint16_t y1, uint16_t x2, uint16_t y2); void tft_cmd(uint32_t cmd, uint32_t count, uint8_t *param); void tft_cmd_dma(uint32_t cmd, uint32_t count, uint8_t *param); uint16_t tft_color_565RGB(uint8_t R, uint8_t G, uint8_t B); //void tft_lv_draw_bitmap(uint16_t x, uint16_t y, uint16_t width, uint16_t height, uint8_t *bitmap); void tft_pio_cmd_init(PIO pio, uint sm, uint out_base, uint set_base, uint32_t freq); uint16_t tft_get_width(); uint16_t tft_get_height(); uint8_t tft_get_orientation(); void tft_set_orientation(uint8_t orientation); void tft_fill_rect(uint16_t x, uint16_t y, uint16_t width, uint16_t height, uint16_t color); #endif
pico_tft.c
#include "stdio.h" #include "stdlib.h" #include "pico/stdlib.h" #include "hardware/clocks.h" #include "string.h" #include "registers.h" #include "pico_tft.pio.h" #include "pico_tft.h" #include "hardware/dma.h" #define MAX_BYTE_TRANS (TFT_WIDTH*TFT_HEIGHT*2) // MADCTL register: MY,MX,MV,ML,BGR,MH,x,x #if TFT_ST7735 static uint8_t TFT_MADCTL_PORTRAIT = 0b11001000; static uint8_t TFT_MADCTL_LANDSCAPE = 0b10101000; static uint8_t TFT_MADCTL_PORTRAIT_MIRROR = 0b00001000; static uint8_t TFT_MADCTL_LANDSCAPE_MIRROR = 0b01101000; #endif #if TFT_ST7796 static uint8_t TFT_MADCTL_PORTRAIT = 0b01001000; static uint8_t TFT_MADCTL_LANDSCAPE = 0b00101000; static uint8_t TFT_MADCTL_PORTRAIT_MIRROR = 0b10001000; static uint8_t TFT_MADCTL_LANDSCAPE_MIRROR = 0b11101000; #endif #if TFT_ILI9341 // MADCTL register: MY,MX,MV,ML,BGR,MH,x,x static uint8_t TFT_MADCTL_PORTRAIT = 0b00000000; static uint8_t TFT_MADCTL_LANDSCAPE = 0b01100000; static uint8_t TFT_MADCTL_PORTRAIT_MIRROR = 0b11000000; static uint8_t TFT_MADCTL_LANDSCAPE_MIRROR = 0b10100000; #endif #if TFT_ILI9342 // MADCTL register: MY,MX,MV,ML,BGR,MH,x,x static uint8_t TFT_MADCTL_PORTRAIT = 0b00000000; static uint8_t TFT_MADCTL_LANDSCAPE = 0b01100000; static uint8_t TFT_MADCTL_PORTRAIT_MIRROR = 0b11000000; static uint8_t TFT_MADCTL_LANDSCAPE_MIRROR = 0b10100000; #endif static uint16_t tft_width; static uint16_t tft_height; static uint8_t tft_orientation; PIO tft_pio = pio1; uint tft_sm=0; uint in_out_base_pin=4; uint set_base_pin=12; uint sideset_base_pin=16; uint s_in_out_base_pin=19; int tft_dma_channel; void tft_cmd(uint32_t cmd, uint32_t count, uint8_t *param) { #if PICO_TFT_SERIAL pio_sm_put_blocking(tft_pio, tft_sm, cmd << 24); #endif #if PICO_TFT_PARALLEL pio_sm_put_blocking(tft_pio, tft_sm, cmd); #endif pio_sm_put_blocking(tft_pio, tft_sm, count); for (int i = 0; i < count; i++) { #if PICO_TFT_SERIAL pio_sm_put_blocking(tft_pio, tft_sm, param[i]<<24); #endif #if PICO_TFT_PARALLEL pio_sm_put_blocking(tft_pio, tft_sm, param[i]); #endif } } //#ifdef PICO_TFT_DMA void tft_cmd_dma(uint32_t cmd, uint32_t count, uint8_t *param) { #if PICO_TFT_SERIAL tft_cmd(cmd, count, param); return; #endif #if PICO_TFT_PARALLEL #if PICO_TFT_DMA pio_sm_put_blocking(tft_pio, tft_sm, cmd); pio_sm_put_blocking(tft_pio, tft_sm, count); dma_channel_set_trans_count(tft_dma_channel, count >> DMA_SIZE_8, false); dma_channel_set_read_addr(tft_dma_channel, param, false); dma_channel_start(tft_dma_channel); dma_channel_wait_for_finish_blocking(tft_dma_channel); #else tft_cmd(cmd, count, param); #endif #endif } //#endif void tft_pio_cmd_init(PIO pio, uint sm, uint in_out_base, uint set_sideset, uint32_t freq) { uint offset=0; pio_sm_config c; #if PICO_TFT_PARALLEL offset = pio_add_program(pio, &tft_pio_parallel_program); c = tft_pio_parallel_program_get_default_config(offset); for (int i=0; i < 8; i++) pio_gpio_init(pio, in_out_base+i); for (int i=0; i < 4; i++) pio_gpio_init(pio, set_sideset+i); pio_sm_set_consecutive_pindirs(pio, sm, in_out_base, 8, true); pio_sm_set_consecutive_pindirs(pio, sm, set_sideset, 4, true); sm_config_set_in_pins(&c, in_out_base); sm_config_set_out_pins(&c, in_out_base, 8); sm_config_set_set_pins(&c, set_sideset, 4); sm_config_set_out_shift(&c, true, false, 8); sm_config_set_in_shift(&c, false, false, 8); #endif #if PICO_TFT_SERIAL offset = pio_add_program(pio, &tft_pio_serial_program); c = tft_pio_serial_program_get_default_config(offset); pio_gpio_init(pio, in_out_base); for (int i=0; i < 3; i++) pio_gpio_init(pio, set_sideset+i); pio_sm_set_consecutive_pindirs(pio, sm, in_out_base, 1, true); pio_sm_set_consecutive_pindirs(pio, sm, set_sideset, 3, true); sm_config_set_in_pins(&c, in_out_base); sm_config_set_out_pins(&c, in_out_base, 1); sm_config_set_sideset_pins(&c, set_sideset); sm_config_set_out_shift(&c, false, false, 8); sm_config_set_in_shift(&c, true, false, 8); #endif //sm_config_set_fifo_join(&c, PIO_FIFO_JOIN_TX); float div = (float)clock_get_hz(clk_sys)/freq; sm_config_set_clkdiv(&c, div); //sm_config_set_clkdiv(&c, 1.25); #if PICO_TFT_DMA /* DMA */ tft_dma_channel = dma_claim_unused_channel(true); dma_channel_config dc = dma_channel_get_default_config(tft_dma_channel); channel_config_set_write_increment(&dc, false); channel_config_set_read_increment(&dc, true); channel_config_set_dreq(&dc, pio_get_dreq(pio, sm, true)); channel_config_set_transfer_data_size(&dc, DMA_SIZE_8); //DMA_SIZE_8,16,32 uint32_t pio_base, tx_offset; pio_base = (pio == pio0) ? PIO0_BASE: PIO1_BASE; tx_offset = PIO_TXF0_OFFSET + sm*4; dma_channel_configure(tft_dma_channel, &dc, (void*) (pio_base + tx_offset), NULL, MAX_BYTE_TRANS>> DMA_SIZE_8, false); //DMA_SIZE_8 or 16 or 32 /* DMA */ #endif pio_sm_init(pio, sm, offset, &c); pio_sm_set_enabled(pio, sm, true); } /* tft draw functions*/ uint16_t tft_color_565RGB(uint8_t R, uint8_t G, uint8_t B) { uint16_t c; c = (((uint16_t)R)>>3)<<11 | (((uint16_t)G)>>2) << 5 | ((uint16_t)B)>>3; return c; } void tft_memory_write_window(uint16_t x1, uint16_t y1, uint16_t x2, uint16_t y2) { uint8_t addr[4]; addr[0]=(uint8_t)(x1 >> 8); addr[1]= (uint8_t)(x1 & 0xff); addr[2]= (uint8_t)(x2 >> 8); addr[3]= (uint8_t)(x2 & 0xff); tft_cmd(TFT_COLADDRSET, 4, addr); addr[0]=(uint8_t)(y1 >> 8); addr[1]= (uint8_t)(y1 & 0xff); addr[2]= (uint8_t)(y2 >> 8); addr[3]= (uint8_t)(y2 & 0xff); tft_cmd(TFT_PAGEADDRSET, 4, addr ); tft_cmd(TFT_MEMORYWRITE, 0, NULL); } void tft_set_address_window(uint16_t x1, uint16_t y1, uint16_t x2, uint16_t y2) { uint8_t addr[4]; addr[0]=(uint8_t)(x1 >> 8); addr[1]= (uint8_t)(x1 & 0xff); addr[2]= (uint8_t)(x2 >> 8); addr[3]= (uint8_t)(x2 & 0xff); tft_cmd(TFT_COLADDRSET, 4, addr); addr[0]=(uint8_t)(y1 >> 8); addr[1]= (uint8_t)(y1 & 0xff); addr[2]= (uint8_t)(y2 >> 8); addr[3]= (uint8_t)(y2 & 0xff); tft_cmd(TFT_PAGEADDRSET, 4, addr ); } /* put color at point*/ void tft_draw_pixel(uint16_t x, uint16_t y, uint16_t color) { if ( x < 0 || x > TFT_WIDTH-1 || y < 0 || y > TFT_HEIGHT-1) { printf("over range,x,y\n"); return; } tft_set_address_window(x,y,x,y); tft_cmd(TFT_MEMORYWRITE, 2, (uint8_t[2]){(uint8_t)(color >> 8), (uint8_t)color}); } uint16_t tft_get_width() { return tft_width; } uint16_t tft_get_height() { return tft_height; } uint8_t tft_get_orientation() { return tft_orientation; } void tft_set_orientation(uint8_t orientation) { tft_orientation = orientation; switch (orientation) { case TFT_ORIENTATION_PORTRAIT: tft_cmd(TFT_MADCTL, 1, (uint8_t[1]){TFT_MADCTL_PORTRAIT}); tft_width = TFT_WIDTH; tft_height = TFT_HEIGHT; break; case TFT_ORIENTATION_PORTRAIT_MIRROR: tft_cmd(TFT_MADCTL, 1, (uint8_t[1]){TFT_MADCTL_PORTRAIT_MIRROR}); tft_width = TFT_WIDTH; tft_height = TFT_HEIGHT; break; case TFT_ORIENTATION_LANDSCAPE: tft_cmd(TFT_MADCTL, 1, (uint8_t[1]){TFT_MADCTL_LANDSCAPE}); tft_width = TFT_HEIGHT; tft_height = TFT_WIDTH; break; case TFT_ORIENTATION_LANDSCAPE_MIRROR: tft_cmd(TFT_MADCTL, 1, (uint8_t[1]){TFT_MADCTL_LANDSCAPE_MIRROR}); tft_width = TFT_HEIGHT; tft_height = TFT_WIDTH; break; } } #if TFT_ST7735 void tft_init_config() { tft_cmd(TFT_SOFTRESET, 0, NULL); sleep_ms(150); tft_cmd(TFT_DISPLAYOFF, 0, NULL); sleep_ms(150); tft_cmd(TFT_PIXELFORMAT, 1, (uint8_t[1]){0x55}); //0x55 tft_cmd(TFT_POWERCONTROL1, 1, (uint8_t[1]){0x05}); // 0x05 :3.3V tft_cmd(TFT_POWERCONTROL2, 1, (uint8_t[1]){0x10}); tft_cmd(TFT_VCOMCONTROL1, 2, (uint8_t[2]){0x3E, 0x28}); tft_cmd(TFT_VCOMCONTROL2, 1, (uint8_t[1]){0x86}); //tft_cmd(TFT_MADCTL, 1, (uint8_t[1]){0x08}); //MY,MX,MV,ML,BRG,MH,0,0(40) tft_set_orientation(TFT_ORIENTATION_PORTRAIT); tft_cmd(TFT_FRAMECONTROL, 2, (uint8_t[2]){0b00, 0x1B}); // Default 70Hz:0x1B tft_cmd(TFT_DISPLAYFUNC, 4, (uint8_t[4]){0x0A, 0x82, 0x27, 0x04}); //0a,a2,27,04 tft_cmd(TFT_GAMMASET, 1, (uint8_t[1]){0x01}); //tft_cmd(tft_PGAMCOR, 15, (uint8_t[15]){ 0x0f, 0x31, 0x2b, 0x0c, 0x0e, 0x08, 0x4e, 0xf1, 0x37, 0x07, 0x10, 0x03, 0x0e, 0x09, 0x00 }); //tft_cmd(tft_NGAMCOR, 15,0xFF, (uint8_t[15]){ 0x00, 0x0e, 0x14, 0x03, 0x11, 0x07, 0x31, 0xc1, 0x48, 0x08, 0x0f, 0x0c, 0x31, 0x36, 0x0f }); tft_cmd(TFT_SLEEPOUT, 0, NULL); sleep_ms(150); tft_cmd(TFT_DISPLAYON, 0, NULL); sleep_ms(500); } #endif #if TFT_ST7796 void tft_init_config() { //tft_cmd(TFT_SOFTRESET, 1, NULL); tft_cmd(TFT_SOFTRESET, 0, NULL); sleep_ms(120); tft_cmd(TFT_SLEEPOUT, 0, NULL); sleep_ms(120); tft_cmd(TFT_COMMANDSET, 1, (uint8_t[1]){0xC3}); //enable part 1 tft_cmd(TFT_COMMANDSET, 1, (uint8_t[1]){0x96}); //enable part 2 //tft_cmd(TFT_MADCTL, 1, (uint8_t[1]){0x88}); //MY,MX,MV,ML,BRG,MH,0,0(24), 0:RGB tft_set_orientation(TFT_ORIENTATION_PORTRAIT); tft_cmd(TFT_PIXELFORMAT, 1, (uint8_t[1]){0x05}); //0x05:RGB565, 0x06 RGB666 tft_cmd(TFT_DSIPLAY_INVER, 1, (uint8_t[1]){0x01}); // 1-dot #if TFT_ILI9342 tft_cmd(TFT_DISPLAYFUNC, 3, (uint8_t[3]){0x0A, 0x82, 0x27}); // ILI9342 #endif #if TFT_ST7796 tft_cmd(TFT_DISPLAYFUNC, 3, (uint8_t[3]){0x80, 0x02, 0x3B}); // ST7796 #endif tft_cmd(TFT_DISP_OUTPUT_CTRL_ADJUST, 8, (uint8_t[8]) {0x40, 0x8A, 0x00, 0x00, 0x29, //Source eqaulizing period time= 22.5 us 0x19, //Timing for "Gate start"=25 (Tclk) 0xA5, //Timing for "Gate End"=37 (Tclk), Gate driver EQ function ON 0x33}); // ST7796 tft_cmd(TFT_POWERCONTROL2, 1, (uint8_t[1]){0x06}); // 0x05 :3.3V tft_cmd(TFT_POWERCONTROL3, 1, (uint8_t[1]){0xA7}); tft_cmd(TFT_VCOMCONTROL1, 1, (uint8_t[1]){0x18}); sleep_ms(120); tft_cmd(TFT_PGAMCOR, 14, (uint8_t[14]){ 0xf0, 0x09, 0x0b, 0x06, 0x04, 0x15, 0x2f, 0x54, 0x42, 0x3c, 0x17, 0x14, 0x18, 0x1b}); tft_cmd(TFT_NGAMCOR, 14, (uint8_t[14]){ 0xe0, 0x09, 0x0b, 0x06, 0x04, 0x03, 0x2b, 0x43, 0x42, 0x3b, 0x16, 0x14, 0x17, 0x1b}); sleep_ms(120); tft_cmd(TFT_COMMANDSET, 1, (uint8_t[1]){0x3C}); // disable part 1 tft_cmd(TFT_COMMANDSET, 1, (uint8_t[1]){0x69}); // disable part 2 tft_cmd(TFT_DISPLAYOFF, 0, NULL); sleep_ms(120); tft_cmd(TFT_DISPLAYON, 0, NULL); sleep_ms(500); } #endif #if TFT_ILI9342 void tft_init_config() { tft_cmd(TFT_SOFTRESET, 0, NULL); sleep_ms(150); tft_cmd(TFT_DISPLAYOFF, 0, NULL); sleep_ms(150); tft_cmd(TFT_PIXELFORMAT, 1, (uint8_t[1]){0x55}); //0x55 tft_cmd(TFT_POWERCONTROL1, 1, (uint8_t[1]){0x05}); // 0x05 :3.3V tft_cmd(TFT_POWERCONTROL2, 1, (uint8_t[1]){0x10}); tft_cmd(TFT_VCOMCONTROL1, 2, (uint8_t[2]){0x3E, 0x28}); tft_cmd(TFT_VCOMCONTROL2, 1, (uint8_t[1]){0x86}); tft_cmd(TFT_MADCTL, 1, (uint8_t[1]){0x60}); //MY,MX,MV,ML,BRG,MH,0,0(40) tft_set_orientation(TFT_ORIENTATION_PORTRAIT); tft_cmd(TFT_FRAMECONTROL, 2, (uint8_t[2]){0x00, 0x1B}); // Default 70Hz tft_cmd(TFT_DISPLAYFUNC, 4, (uint8_t[4]){0x0A, 0x82, 0x27, 0x04}); //0a,a2,27,04 tft_cmd(TFT_GAMMASET, 1, (uint8_t[1]){0x01}); tft_cmd(TFT_PGAMCOR, 15, (uint8_t[15]){ 0x0f, 0x31, 0x2b, 0x0c, 0x0e, 0x08, 0x4e, 0xf1, 0x37, 0x07, 0x10, 0x03, 0x0e, 0x09, 0x00 }); tft_cmd(TFT_NGAMCOR, 15, (uint8_t[15]){ 0x00, 0x0e, 0x14, 0x03, 0x11, 0x07, 0x31, 0xc1, 0x48, 0x08, 0x0f, 0x0c, 0x31, 0x36, 0x0f }); tft_cmd(TFT_SLEEPOUT, 0, NULL); sleep_ms(150); tft_cmd(TFT_DISPLAYON, 0, NULL); sleep_ms(500); } #endif #if TFT_ILI9341 void tft_init_config() { tft_cmd(TFT_SOFTRESET, 0, NULL); sleep_ms(50); tft_cmd(TFT_DISPLAYOFF, 0, NULL); tft_cmd(TFT_POWERCONTROL1, 1, (uint8_t[1]){0x23}); tft_cmd(TFT_POWERCONTROL2, 1, (uint8_t[1]){0x10}); tft_cmd(TFT_VCOMCONTROL1, 2, (uint8_t[2]){0x2B, 0x2B}); tft_cmd(TFT_VCOMCONTROL2, 1, (uint8_t[1]){0xC0}); tft_cmd(TFT_MADCTL, 1, (uint8_t[1]){0x88}); //MY,MX,MV,ML,BRG,MH,0,0(40) tft_set_orientation(TFT_ORIENTATION_PORTRAIT); tft_cmd(TFT_PIXELFORMAT, 1, (uint8_t[1]){0x55}); tft_cmd(TFT_FRAMECONTROL, 2, (uint8_t[2]){0x00, 0x1B}); tft_cmd(TFT_ENTRYMODE, 1, (uint8_t[1]){0x07}); //tft_cmd(TFT_DISPLAYFUNC, 4, (uint8_t[4]){0x0A, 0x82, 0x27, 0x00}); tft_cmd(TFT_SLEEPOUT, 0, NULL); sleep_ms(150); tft_cmd(TFT_DISPLAYON, 0, NULL); sleep_ms(500); } #endif void tft_init(PIO pio, uint sm, uint din_base, uint csx_dcx_sck_side_base_pin) { tft_pio = pio; tft_sm = sm; #if PICO_TFT_PARALLEL in_out_base_pin = din_base; set_base_pin = csx_dcx_sck_side_base_pin; tft_pio_cmd_init(tft_pio, tft_sm, in_out_base_pin, set_base_pin, 70000000/*70000000*/); //pio freq #endif #if PICO_TFT_SERIAL s_in_out_base_pin = din_base; sideset_base_pin = csx_dcx_sck_side_base_pin; tft_pio_cmd_init(tft_pio, tft_sm, s_in_out_base_pin, sideset_base_pin, 90000000/* 62.5M baud rate for SPI*/); //pio freq #endif tft_init_config(); }
pico_tft_lvgl.h
#ifndef __PICO_TFT_LVGL__ #define __PICO_TFT_LVGL__ #include "pico_tft.h" #include "lvgl.h" void tft_lvgl_draw_bitmap(uint16_t x, uint16_t y, uint16_t width, uint16_t height, uint8_t *bitmap); void tft_lvgl_disp_flush(lv_disp_drv_t * disp, const lv_area_t * area, lv_color_t * color_p); uint32_t pico_lvgl_get_tick_cb(); #endif
pico_tft_lvgl.c
#include "pico_tft.h" #include "pico_tft_lvgl.h" #include "registers.h" #include "stdio.h" void tft_lvgl_draw_bitmap(uint16_t x1, uint16_t y1, uint16_t x2, uint16_t y2, uint8_t *bitmap) { uint32_t total_pixels = (x2-x1+1) * (y2-y1+1)*2; tft_set_address_window (x1, y1, x2, y2); tft_cmd_dma(TFT_MEMORYWRITE, total_pixels, bitmap); } void tft_lvgl_disp_flush(lv_disp_drv_t * disp, const lv_area_t * area, lv_color_t * color_p) { tft_lvgl_draw_bitmap( (uint16_t)(area->x1), (uint16_t)(area->y1), (uint16_t)(area->x2), (uint16_t)(area->y2), (uint8_t*)color_p ); lv_disp_flush_ready(disp); /* Indicate you are ready with the flushing*/ } /* tell how many milliseconds have elapsed since start up for lvgl 9.x.x */ uint32_t pico_lvgl_get_tick_cb() { return (time_us_32()/1000); }
pico_tft.pio
.program tft_pio_parallel .wrap_target start: pull set pins, 0b0011 mov x, osr ;command code, if 0x0, command nop, only data jmp !x param set pins, 0b0001 out pins, 8 [1] set pins, 0b0011 [1] param: set pins, 0b0111 pull mov x, osr ;how many parameters jmp !x, start ;no parameter return start jmp x--, param_data param_data: pull ; write data set pins, 0b0101 out pins, 8 [1] set pins, 0b0111 [1] jmp x--, param_data set pins, 0b1111 jmp start .wrap .program tft_pio_serial ; CSX, D/CX(A0), SCL --> gpio 13, 12, 11 (side-set pins) .side_set 3 .wrap_target start: pull side 0b100 mov x, osr side 0b000 ;command code, if 0x0, command nop, only data jmp !x param side 0b000 set y,7 side 0b000 cmd_bit_loop: out pins, 1 side 0b000 [1] jmp y--, cmd_bit_loop side 0b001 [1] param: pull side 0b010 mov x, osr side 0b010 ;how many parameters jmp !x, start side 0b010 ;no parameter return start jmp x--, param_data side 0b010 param_data: pull side 0b010 ; write data set y,7 side 0b010 data_bit_loop: out pins, 1 side 0b010 [1] jmp y--, data_bit_loop side 0b011 [1] jmp x--, param_data side 0b010 jmp start side 0b100
- registers.h
#define TFT_NOP 0x00 #define TFT_SOFTRESET 0x01 #define TFT_SLEEPIN 0x10 #define TFT_SLEEPOUT 0x11 #define TFT_NORMALDISP 0x13 #define TFT_INVERTOFF 0x20 #define TFT_INVERTON 0x21 #define TFT_GAMMASET 0x26 #define TFT_DISPLAYOFF 0x28 #define TFT_DISPLAYON 0x29 #define TFT_COLADDRSET 0x2A #define TFT_PAGEADDRSET 0x2B #define TFT_MEMORYWRITE 0x2C #define TFT_MEMORYREAD 0x2E #define TFT_PIXELFORMAT 0x3A #define TFT_MEMORYWRITECONT 0x3C #define TFT_MEMORYREADCONT 0x3E #define TFT_FRAMECONTROL 0xB1 #define TFT_DSIPLAY_INVER 0xB4 #define TFT_DISPLAYFUNC 0xB6 #define TFT_ENTRYMODE 0xB7 #define TFT_POWERCONTROL1 0xC0 #define TFT_POWERCONTROL2 0xC1 #define TFT_POWERCONTROL3 0xC2 #define TFT_VCOMCONTROL1 0xC5 #define TFT_VCOMCONTROL2 0xC7 #define TFT_COMMANDSET 0xF0 #define TFT_VSCROLLDEF 0x33 #define TFT_VSCROLLADDR 0x37 #define TFT_MEMCONTROL 0x36 #define TFT_MADCTL 0x36 #define TFT_PGAMCOR 0xE0 #define TFT_NGAMCOR 0xE1 #define TFT_DISP_OUTPUT_CTRL_ADJUST 0xE8 #define TFT_MADCTL_MY 0x80 #define TFT_MADCTL_MX 0x40 #define TFT_MADCTL_MV 0x20 #define TFT_MADCTL_ML 0x10 #define TFT_MADCTL_RGB 0x00 #define TFT_MADCTL_BGR 0x08 #define TFT_MADCTL_MH 0x04
- XPT2046_touch
CMakeLists.txt
xpt2046.h
xpt2046.c
CMakeLists.txt
add_library(xpt2046_touch INTERFACE) target_sources(xpt2046_touch INTERFACE ${CMAKE_CURRENT_LIST_DIR}/xpt2046.c ) target_include_directories(xpt2046_touch INTERFACE ${CMAKE_CURRENT_LIST_DIR} ${CMAKE_CURRENT_LIST_DIR}/.. ) target_link_libraries(xpt2046_touch INTERFACE hardware_spi hardware_dma )
xpt2046.h
#ifndef _XPT2046_H_ #define _XPT2046_H_ #include "pico/stdlib.h" #include "lvgl.h" //== Set the following values according to the hardware == #define XPT2046_IRQ_GPIO 11 #define XPT2046_MOSI 15 #define XPT2046_MISO 12 #define XPT2046_CS 13 #define XPT2046_CLK 14 #define XPT2046_SPI spi1 #define XPT2046_MIN_RAW_X 1350 #define XPT2046_MAX_RAW_X 31000 #define XPT2046_MIN_RAW_Y 2050 #define XPT2046_MAX_RAW_Y 31500 //== Set the following values according to the hardware == void xpt2046_init(); bool xpt2046_getXY(uint16_t *x, uint16_t *y); bool xpt2046_TouchPressed(); void xpt2046_lvgl_read_cb(struct _lv_indev_drv_t * indev, lv_indev_data_t* data); #endif
xpt2046.c
#include "stdio.h" #include "stdlib.h" #include "xpt2046.h" #include "hardware/spi.h" #include "hardware/gpio.h" #include "pico/stdlib.h" #include "pico_tft.h" uint32_t xpt2046_event=0; uint8_t READ_X = 0xD0; uint8_t READ_Y = 0x90; #define XPT_WIDTH 320 #define XPT_HEIGHT 480 bool xpt2046_getXY(uint16_t *x, uint16_t *y) { uint8_t temp[2]; uint16_t raw_x, raw_y; uint16_t est_raw_x, est_raw_y; uint32_t avg_x = 0; uint32_t avg_y = 0; uint8_t nsamples = 0; uint8_t SAMPLES=10; gpio_put(XPT2046_CS, false); if(gpio_get(XPT2046_IRQ_GPIO)) return false; busy_wait_ms(10); //first pass SAMPLES=20; // get first average; for(uint8_t i = 0; i < SAMPLES; i++, nsamples++) { if(gpio_get(XPT2046_IRQ_GPIO)) { break; } spi_write_blocking(XPT2046_SPI, &READ_X, 1); spi_read_blocking(XPT2046_SPI, 0x00, temp, 2); raw_x = ((uint16_t)temp[0]) << 8 | (uint16_t)temp[1]; spi_write_blocking(XPT2046_SPI, &READ_Y, 1); spi_read_blocking(XPT2046_SPI, 0x00, temp, 2); raw_y = ((uint16_t)temp[0]) << 8 | (uint16_t)temp[1]; avg_x += raw_x; avg_y += raw_y; } if(nsamples < SAMPLES) return false; gpio_put(XPT2046_CS, true); raw_x = (avg_x / SAMPLES); raw_y = (avg_y / SAMPLES); if(raw_x < XPT2046_MIN_RAW_X || raw_x > XPT2046_MAX_RAW_X) return false; if(raw_y < XPT2046_MIN_RAW_Y || raw_y > XPT2046_MAX_RAW_Y) return false; uint16_t tx,ty; tx = (raw_x - XPT2046_MIN_RAW_X) * XPT_WIDTH / (XPT2046_MAX_RAW_X - XPT2046_MIN_RAW_X); ty = (raw_y - XPT2046_MIN_RAW_Y) * XPT_HEIGHT / (XPT2046_MAX_RAW_Y - XPT2046_MIN_RAW_Y); // adjust for TFT orientation uint8_t lot = tft_get_orientation(); switch (lot) { case TFT_ORIENTATION_PORTRAIT: *x=tx; *y=TFT_HEIGHT-ty; break; case TFT_ORIENTATION_LANDSCAPE: *x=TFT_HEIGHT-ty; *y=TFT_WIDTH-tx; break; case TFT_ORIENTATION_PORTRAIT_MIRROR: *x=TFT_WIDTH-tx; *y=ty; break; case TFT_ORIENTATION_LANDSCAPE_MIRROR: *x=ty; *y=tx; break; } return true; } bool xpt2046_TouchPressed() { return !gpio_get(XPT2046_IRQ_GPIO); } void xpt2046_init() { gpio_init(XPT2046_IRQ_GPIO); gpio_init(XPT2046_MISO); gpio_init(XPT2046_MOSI); gpio_init(XPT2046_CLK); gpio_init(XPT2046_CS); gpio_set_dir(XPT2046_CS, GPIO_OUT); gpio_set_dir(XPT2046_IRQ_GPIO, GPIO_OUT); gpio_set_function(XPT2046_CLK,GPIO_FUNC_SPI); gpio_set_function(XPT2046_CS,GPIO_FUNC_SIO); gpio_set_function(XPT2046_MOSI,GPIO_FUNC_SPI); gpio_set_function(XPT2046_MISO,GPIO_FUNC_SPI); //spi_init(XPT2046_SPI,250000); spi_init(XPT2046_SPI,3125000); //3125000 } extern lv_obj_t *mouse_cursor; void xpt2046_lvgl_read_cb(struct _lv_indev_drv_t * indev, lv_indev_data_t* data) { uint16_t x,y; if (xpt2046_TouchPressed()) { if (xpt2046_getXY(&x,&y)) { data->point.x = x; data->point.y = y; data->state = LV_INDEV_STATE_PRESSED; } else { data->state = LV_INDEV_STATE_RELEASED; } } else { data->state = LV_INDEV_STATE_RELEASED; } }
- pico_lvgl
add_library(pico_lvgl INTERFACE) target_sources(pico_lvgl INTERFACE ${CMAKE_CURRENT_LIST_DIR}/pico_lvgl.c ) add_subdirectory(pico_tft) add_subdirectory(xpt2046_touch) add_subdirectory(lvgl) target_include_directories(pico_lvgl INTERFACE ${CMAKE_CURRENT_LIST_DIR} ${CMAKE_CURRENT_LIST_DIR}/pico_tft ${CMAKE_CURRENT_LIST_DIR}/xpt2046_touch ${CMAKE_CURRENT_LIST_DIR}/lvgl ) target_link_libraries(pico_lvgl INTERFACE hardware_pio hardware_dma pico_tft xpt2046_touch lvgl )
#include "pico_lvgl.h" bool lv_inc_timer_cb(repeating_timer_t *rt) { lv_tick_inc(*((uint8_t*)rt->user_data)); return true; } /* TFT_SERIAL: @param sdi_gpio: SDI(MOSI) @param csx_dcx_sck_gpio: csx:GPIO_n+2,dcx:GPIO_n+1, sck:GPIO_n TFT_PARALLEL: @param sdi_gpio: LCD_D0(DB0): GPIO_dn, LCD_D1(DB1):GPIO_dn+1,...LCD_D7(DB7):GPIO_dn+7 @param csx_dcx_sck_gpio: CS:GPIO_n+3,RS:GPIO_n+2, WR:GPIO_n+1, RD:GPIO_n */ void pico_lvgl_tft_init(PIO pio, uint sm, uint sdi_gpio, uint csx_dcx_sck_gpio) { tft_init(pio , sm, sdi_gpio, csx_dcx_sck_gpio); } void pico_lvgl_tick_inc_timer_init(uint8_t tick_inc) { static repeating_timer_t rt; static uint8_t _tick_inc; _tick_inc = tick_inc; add_repeating_timer_ms(tick_inc, lv_inc_timer_cb, (void*) (&_tick_inc), &rt); } /* tick_inc: to call the lv_tick_inc(x) function periodically in (x) milliseconds. */ void pico_lvgl_display_init(uint8_t tick_inc) { // init timer if (tick_inc > 10) tick_inc = 10; if (tick_inc < 1) tick_inc = 1; pico_lvgl_tick_inc_timer_init(tick_inc); // lv_init lv_init(); // set draw buffer static lv_disp_draw_buf_t draw_buf; static lv_color_t buf1[TFT_WIDTH * TFT_HEIGHT / 10]; /*Declare a buffer for 1/10 screen size*/ lv_disp_draw_buf_init(&draw_buf, buf1, NULL, TFT_WIDTH * TFT_HEIGHT / 10); /*Initialize the display buffer.*/ //setup display static lv_disp_drv_t disp_drv; /*Descriptor of a display driver*/ lv_disp_drv_init(&disp_drv); /*Basic initialization*/ disp_drv.flush_cb = tft_lvgl_disp_flush; /*Set your driver function*/ disp_drv.draw_buf = &draw_buf; /*Assign the buffer to the display*/ disp_drv.hor_res = tft_get_width(); /*Set the horizontal resolution of the display*/ disp_drv.ver_res = tft_get_height(); /*Set the vertical resolution of the display*/ lv_disp_drv_register(&disp_drv); /*Finally register the driver*/ } void pico_lvgl_xpt2046_init() { // init xpt2046 hardware xpt2046_init(); // setup touch screen input device static lv_indev_drv_t indev_drv; /*Descriptor of a input device driver*/ lv_indev_drv_init(&indev_drv); /*Basic initialization*/ indev_drv.type = LV_INDEV_TYPE_POINTER; /*Touch pad is a pointer-like device*/ indev_drv.read_cb = xpt2046_lvgl_read_cb; /*Set your driver function*/ lv_indev_drv_register(&indev_drv); /*Finally register the driver*/ } lv_indev_t *encoder_indev; lv_group_t *encoder_group; void pico_lvgl_encoder_init(bool pio_mode) { // init rotary encoder hardware if (pio_mode) { lvgl_pio_encoder_init(); } else { gpio_encoder_init(); } // setup rotary encoder input device static lv_indev_drv_t indev_drv; /*Descriptor of a input device driver*/ lv_indev_drv_init(&indev_drv); /*Basic initialization*/ indev_drv.type = LV_INDEV_TYPE_ENCODER; /*Rotary Encoder device*/ if (pio_mode) { indev_drv.read_cb = lvgl_encoder_read_cb; /*Set your driver function*/ } else { indev_drv.read_cb = gpio_encoder_read_cb; } //indev_drv.long_press_time=1000; indev_drv.long_press_repeat_time=2000; encoder_indev = lv_indev_drv_register(&indev_drv); /*Finally register the driver*/ encoder_group = lv_group_create(); //lv_group_set_default(kg); lv_indev_set_group(encoder_indev, encoder_group); }
#ifndef __PICO_LVGL_H__ #define __PICO_LVGL_H__ #include "stdio.h" #include "pico/stdio.h" #include "hardware/pio.h" #include "pico_tft.h" #include "xpt2046.h" #include "lvgl.h" void pico_lvgl_tft_init(PIO pio, uint sm, uint sdi_gpio, uint csx_dcx_sck_gpio); void pico_lvgl_xpt2046_init(); void pico_lvgl_display_init(uint8_t tick_inc); #endif
- root
CMakeLists.txt
add_subdirectory(pico_lvgl) target_link_libraries(project_name pico_lvgl )
沒有留言:
張貼留言