/**************************************************************************//** * * @copyright (C) 2020 Nuvoton Technology Corp. All rights reserved. * * SPDX-License-Identifier: Apache-2.0 * * Change Logs: * Date Author Notes * 2020-1-16 Wayne First version * ******************************************************************************/ #include #if defined(NU_PKG_USING_ILI9341) #include #include #if defined(NU_PKG_ILI9341_WITH_OFFSCREEN_FRAMEBUFFER) #if !defined(NU_PKG_ILI9341_LINE_BUFFER_NUMBER) #define NU_PKG_ILI9341_LINE_BUFFER_NUMBER YSIZE_PHYS #endif #endif #define ili9341_delay_ms(ms) rt_thread_mdelay(ms) static struct rt_device_graphic_info g_Ili9341Info = { .bits_per_pixel = 16, .pixel_format = RTGRAPHIC_PIXEL_FORMAT_RGB565, .framebuffer = RT_NULL, .width = XSIZE_PHYS, .pitch = XSIZE_PHYS * 2, .height = YSIZE_PHYS }; static rt_err_t ili9341_pin_init(void) { rt_pin_mode(BOARD_USING_ILI9341_PIN_DC, PIN_MODE_OUTPUT); rt_pin_mode(BOARD_USING_ILI9341_PIN_RESET, PIN_MODE_OUTPUT); rt_pin_mode(BOARD_USING_ILI9341_PIN_BACKLIGHT, PIN_MODE_OUTPUT); SET_RS; SET_RST; SET_BACKLIGHT_OFF; return RT_EOK; } static rt_err_t ili9341_lcd_init(rt_device_t dev) { /* Hardware reset */ SET_RST; ili9341_delay_ms(5); // Delay 5ms CLR_RST; ili9341_delay_ms(20); // Delay 20ms SET_RST; ili9341_delay_ms(40); // Delay 40ms /* Initial control registers */ ili9341_send_cmd(0xCB); ili9341_send_cmd_parameter(0x39); ili9341_send_cmd_parameter(0x2C); ili9341_send_cmd_parameter(0x00); ili9341_send_cmd_parameter(0x34); ili9341_send_cmd_parameter(0x02); ili9341_send_cmd(0xCF); ili9341_send_cmd_parameter(0x00); ili9341_send_cmd_parameter(0xC1); ili9341_send_cmd_parameter(0x30); ili9341_send_cmd(0xE8); ili9341_send_cmd_parameter(0x85); ili9341_send_cmd_parameter(0x00); ili9341_send_cmd_parameter(0x78); ili9341_send_cmd(0xEA); ili9341_send_cmd_parameter(0x00); ili9341_send_cmd_parameter(0x00); ili9341_send_cmd(0xED); ili9341_send_cmd_parameter(0x64); ili9341_send_cmd_parameter(0x03); ili9341_send_cmd_parameter(0x12); ili9341_send_cmd_parameter(0x81); ili9341_send_cmd(0xF7); ili9341_send_cmd_parameter(0x20); ili9341_send_cmd(0xC0); ili9341_send_cmd_parameter(0x23); ili9341_send_cmd(0xC1); ili9341_send_cmd_parameter(0x10); ili9341_send_cmd(0xC5); ili9341_send_cmd_parameter(0x3e); ili9341_send_cmd_parameter(0x28); ili9341_send_cmd(0xC7); ili9341_send_cmd_parameter(0x86); ili9341_send_cmd(0x36); if (g_Ili9341Info.width == 240) ili9341_send_cmd_parameter(0x48); // for 240x320 else ili9341_send_cmd_parameter(0xE8); // for 320x240 ili9341_send_cmd(0x3A); ili9341_send_cmd_parameter(0x55); ili9341_send_cmd(0xB1); ili9341_send_cmd_parameter(0x00); ili9341_send_cmd_parameter(0x18); ili9341_send_cmd(0xB6); ili9341_send_cmd_parameter(0x08); ili9341_send_cmd_parameter(0x82); ili9341_send_cmd_parameter(0x27); ili9341_send_cmd(0xF2); ili9341_send_cmd_parameter(0x00); ili9341_send_cmd(0x26); ili9341_send_cmd_parameter(0x01); ili9341_send_cmd(0xE0); ili9341_send_cmd_parameter(0x0F); ili9341_send_cmd_parameter(0x31); ili9341_send_cmd_parameter(0x2B); ili9341_send_cmd_parameter(0x0C); ili9341_send_cmd_parameter(0x0E); ili9341_send_cmd_parameter(0x08); ili9341_send_cmd_parameter(0x4E); ili9341_send_cmd_parameter(0xF1); ili9341_send_cmd_parameter(0x37); ili9341_send_cmd_parameter(0x07); ili9341_send_cmd_parameter(0x10); ili9341_send_cmd_parameter(0x03); ili9341_send_cmd_parameter(0x0E); ili9341_send_cmd_parameter(0x09); ili9341_send_cmd_parameter(0x00); ili9341_send_cmd(0xE1); ili9341_send_cmd_parameter(0x00); ili9341_send_cmd_parameter(0x0E); ili9341_send_cmd_parameter(0x14); ili9341_send_cmd_parameter(0x03); ili9341_send_cmd_parameter(0x11); ili9341_send_cmd_parameter(0x07); ili9341_send_cmd_parameter(0x31); ili9341_send_cmd_parameter(0xC1); ili9341_send_cmd_parameter(0x48); ili9341_send_cmd_parameter(0x08); ili9341_send_cmd_parameter(0x0F); ili9341_send_cmd_parameter(0x0C); ili9341_send_cmd_parameter(0x31); ili9341_send_cmd_parameter(0x36); ili9341_send_cmd_parameter(0x0F); ili9341_send_cmd(0x11); ili9341_delay_ms(120); // Delay 120ms ili9341_send_cmd(0x29); //Display on SET_BACKLIGHT_ON; return RT_EOK; } #if defined(NU_PKG_ILI9341_WITH_OFFSCREEN_FRAMEBUFFER) static void ili9341_fillrect(uint16_t *pixels, struct rt_device_rect_info *pRectInfo) { ili9341_set_column(pRectInfo->x, pRectInfo->x + pRectInfo->width - 1); ili9341_set_page(pRectInfo->y, pRectInfo->y + pRectInfo->height - 1); ili9341_send_cmd(0x2c); ili9341_send_pixels(pixels, pRectInfo->height * pRectInfo->width * 2); } #endif static void ili9341_fillscreen(rt_uint16_t color) { #if defined(NU_PKG_ILI9341_WITH_OFFSCREEN_FRAMEBUFFER) struct rt_device_rect_info rectinfo; int filled_line_num = 0; while (filled_line_num < YSIZE_PHYS) { int pixel_count; rectinfo.x = 0; rectinfo.y = filled_line_num; rectinfo.width = XSIZE_PHYS; rectinfo.height = (NU_PKG_ILI9341_LINE_BUFFER_NUMBER < YSIZE_PHYS) ? NU_PKG_ILI9341_LINE_BUFFER_NUMBER : YSIZE_PHYS; pixel_count = XSIZE_PHYS * NU_PKG_ILI9341_LINE_BUFFER_NUMBER; rt_uint16_t *pu16ShadowBuf = (rt_uint16_t *)g_Ili9341Info.framebuffer; while (pixel_count > 0) { *pu16ShadowBuf++ = color; pixel_count--; } ili9341_fillrect((uint16_t *)g_Ili9341Info.framebuffer, &rectinfo); filled_line_num += NU_PKG_ILI9341_LINE_BUFFER_NUMBER; } #else ili9341_set_column(0, (XSIZE_PHYS - 1)); ili9341_set_page(0, (YSIZE_PHYS - 1)); ili9341_send_cmd(0x2c); for (int i = 0; i < (XSIZE_PHYS * YSIZE_PHYS); i++) ili9341_send_pixel_data(color); #endif } static void ili9341_lcd_set_pixel(const char *color, int x, int y) { ili9341_set_column(x, x); ili9341_set_page(y, y); ili9341_send_cmd(0x2c); ili9341_send_pixel_data(*(uint16_t *)color); } static void ili9341_lcd_draw_hline(const char *pixel, int x1, int x2, int y) { ili9341_set_column(x1, x2); ili9341_set_page(y, y); ili9341_send_cmd(0x2c); for (; x1 < x2; x1++) ili9341_send_pixel_data(*(uint16_t *)pixel); } static void ili9341_lcd_draw_vline(const char *pixel, int x, int y1, int y2) { ili9341_set_column(x, x); ili9341_set_page(y1, y2); ili9341_send_cmd(0x2c); for (; y1 < y2; y1++) ili9341_send_pixel_data(*(uint16_t *)pixel); } static void ili9341_lcd_blit_line(const char *pixels, int x, int y, rt_size_t size) { rt_uint16_t *ptr = (rt_uint16_t *)pixels; ili9341_set_column(x, x + size); ili9341_set_page(y, y); ili9341_send_cmd(0x2c); while (size--) ili9341_send_pixel_data(*ptr++); } static rt_err_t ili9341_lcd_open(rt_device_t dev, rt_uint16_t oflag) { return RT_EOK; } static rt_err_t ili9341_lcd_close(rt_device_t dev) { return RT_EOK; } static rt_err_t ili9341_lcd_control(rt_device_t dev, int cmd, void *args) { switch (cmd) { case RTGRAPHIC_CTRL_GET_INFO: { struct rt_device_graphic_info *info; info = (struct rt_device_graphic_info *) args; RT_ASSERT(info != RT_NULL); rt_memcpy(args, (void *)&g_Ili9341Info, sizeof(struct rt_device_graphic_info)); } break; case RTGRAPHIC_CTRL_RECT_UPDATE: { #if defined(NU_PKG_ILI9341_WITH_OFFSCREEN_FRAMEBUFFER) struct rt_device_rect_info *psRectInfo = (struct rt_device_rect_info *)args; rt_uint16_t *pixels = (rt_uint16_t *)g_Ili9341Info.framebuffer; RT_ASSERT(args); ili9341_fillrect(pixels, psRectInfo); #else /* nothong to be done */ #endif } break; default: return -RT_ERROR; } return RT_EOK; } static struct rt_device lcd_device; static struct rt_device_graphic_ops ili9341_ops = { ili9341_lcd_set_pixel, ili9341_lcd_get_pixel, ili9341_lcd_draw_hline, ili9341_lcd_draw_vline, ili9341_lcd_blit_line }; int rt_hw_lcd_ili9341_init(void) { ili9341_pin_init(); /* register lcd device */ lcd_device.type = RT_Device_Class_Graphic; lcd_device.init = ili9341_lcd_init; lcd_device.open = ili9341_lcd_open; lcd_device.close = ili9341_lcd_close; lcd_device.control = ili9341_lcd_control; lcd_device.read = RT_NULL; lcd_device.write = RT_NULL; lcd_device.user_data = &ili9341_ops; #if defined(NU_PKG_ILI9341_WITH_OFFSCREEN_FRAMEBUFFER) g_Ili9341Info.framebuffer = rt_malloc_align((g_Ili9341Info.pitch * NU_PKG_ILI9341_LINE_BUFFER_NUMBER) + 32, 32); RT_ASSERT(g_Ili9341Info.framebuffer != RT_NULL); g_Ili9341Info.smem_len = g_Ili9341Info.pitch * NU_PKG_ILI9341_LINE_BUFFER_NUMBER; #endif /* register graphic device driver */ rt_device_register(&lcd_device, "lcd", RT_DEVICE_FLAG_RDWR | RT_DEVICE_FLAG_STANDALONE); return 0; } #ifdef RT_USING_FINSH #define LINE_LEN 32 static void lcd_test(int argc, char *argv[]) { uint16_t pixels[LINE_LEN]; uint16_t color; int x, y, i; x = y = 100; ili9341_lcd_init(NULL); color = 0x0; //Black, RGB rt_kprintf("Brush 0x%X on screen.\n", color); ili9341_fillscreen(color); ili9341_lcd_get_pixel((char *)&color, x, y); rt_kprintf("lcd get pixel, pixel: 0x%X, x: %d, y: %d\n", color, x, y); color = 0xffff; //White, RGB rt_kprintf("Brush 0x%X on screen.\n", color); ili9341_fillscreen(color); ili9341_lcd_get_pixel((char *)&color, x, y); rt_kprintf("lcd get pixel, pixel: 0x%X, x: %d, y: %d\n", color, x, y); color = 0x1f; //Blue, RGB rt_kprintf("Brush 0x%X on screen.\n", color); ili9341_fillscreen(color); ili9341_lcd_get_pixel((char *)&color, x, y); rt_kprintf("lcd get pixel, pixel: 0x%X, x: %d, y: %d\n", color, x, y); color = 0x07e0; //Green, RGB rt_kprintf("Brush 0x%X on screen.\n", color); ili9341_fillscreen(color); ili9341_lcd_get_pixel((char *)&color, x, y); rt_kprintf("lcd get pixel, pixel: 0x%X, x: %d, y: %d\n", color, x, y); color = 0xf800; //Red, RGB rt_kprintf("Brush 0x%X on screen.\n", color); ili9341_fillscreen(color); ili9341_lcd_get_pixel((char *)&color, x, y); rt_kprintf("lcd get pixel, pixel: 0x%X, x: %d, y: %d\n", color, x, y); color = 0xffff; //White, RGB rt_kprintf("lcd draw hline, pixel: 0x%X, x1: %d, x2: %d, y: %d\n", color, x, x + 20, y); ili9341_lcd_draw_hline((const char *)&color, x, x + 20, y); color = 0xffff; //White, RGB rt_kprintf("lcd draw vline, pixel: 0x%X, x: %d, y: %d\n", color, y, y + 20); ili9341_lcd_draw_vline((const char *)&color, x, y, y + 20); for (i = 0; i < LINE_LEN; i++) pixels[i] = 20 + i * 5; x = y = 50; rt_kprintf("lcd blit line, start: x: %d, y: %d\n", x, y); ili9341_lcd_blit_line((const char *)&pixels[0], x, y, LINE_LEN); x = y = 200; color = 0x07E0; //Green, RGB rt_kprintf("lcd set pixel, pixel: 0x%X, x: %d, y: %d\n", color, x, y); ili9341_lcd_set_pixel((const char *)&color, x, y); color = 0x0; ili9341_lcd_get_pixel((char *)&color, x, y); rt_kprintf("lcd get pixel, pixel: 0x%X, x: %d, y: %d\n", color, x, y); x = y = 200; color = 0x1f; //Blue, RGB rt_kprintf("lcd set pixel, pixel: 0x%X, x: %d, y: %d\n", color, x, y); ili9341_lcd_set_pixel((const char *)&color, x, y); color = 0x0; ili9341_lcd_get_pixel((char *)&color, x, y); rt_kprintf("lcd get pixel, pixel: 0x%X, x: %d, y: %d\n", color, x, y); x = y = 200; color = 0xf800; //Red, RGB rt_kprintf("lcd set pixel, pixel: 0x%X, x: %d, y: %d\n", color, x, y); ili9341_lcd_set_pixel((const char *)&color, x, y); color = 0x0; ili9341_lcd_get_pixel((char *)&color, x, y); rt_kprintf("lcd get pixel, pixel: 0x%X, x: %d, y: %d\n", color, x, y); } MSH_CMD_EXPORT(lcd_test, test lcd display); #endif #endif /* if defined(NU_PKG_USING_ILI9341) */