lcd_ili9341.c 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420
  1. /**************************************************************************//**
  2. *
  3. * @copyright (C) 2020 Nuvoton Technology Corp. All rights reserved.
  4. *
  5. * SPDX-License-Identifier: Apache-2.0
  6. *
  7. * Change Logs:
  8. * Date Author Notes
  9. * 2020-1-16 Wayne First version
  10. *
  11. ******************************************************************************/
  12. #include <rtconfig.h>
  13. #if defined(NU_PKG_USING_ILI9341)
  14. #include <rtdevice.h>
  15. #include <lcd_ili9341.h>
  16. #if defined(NU_PKG_ILI9341_WITH_OFFSCREEN_FRAMEBUFFER)
  17. #define DEF_VRAM_BUFFER_NUMBER 2
  18. #endif
  19. static struct rt_device_graphic_info g_Ili9341Info =
  20. {
  21. .bits_per_pixel = 16,
  22. .pixel_format = RTGRAPHIC_PIXEL_FORMAT_RGB565,
  23. .framebuffer = RT_NULL,
  24. .width = XSIZE_PHYS,
  25. .height = YSIZE_PHYS
  26. };
  27. static void ili9341_delay_ms(rt_uint32_t nms)
  28. {
  29. rt_thread_mdelay(nms);
  30. }
  31. static rt_err_t ili9341_pin_init(void)
  32. {
  33. rt_pin_mode(BOARD_USING_ILI9341_PIN_DC, PIN_MODE_OUTPUT);
  34. rt_pin_mode(BOARD_USING_ILI9341_PIN_RESET, PIN_MODE_OUTPUT);
  35. rt_pin_mode(BOARD_USING_ILI9341_PIN_BACKLIGHT, PIN_MODE_OUTPUT);
  36. SET_RS;
  37. SET_RST;
  38. SET_BACKLIGHT_OFF;
  39. return RT_EOK;
  40. }
  41. static rt_err_t ili9341_lcd_init(rt_device_t dev)
  42. {
  43. /* Hardware reset */
  44. SET_RST;
  45. ili9341_delay_ms(5); // Delay 5ms
  46. CLR_RST;
  47. ili9341_delay_ms(20); // Delay 20ms
  48. SET_RST;
  49. ili9341_delay_ms(40); // Delay 40ms
  50. /* Initial control registers */
  51. ili9341_send_cmd(0xCB);
  52. ili9341_send_cmd_parameter(0x39);
  53. ili9341_send_cmd_parameter(0x2C);
  54. ili9341_send_cmd_parameter(0x00);
  55. ili9341_send_cmd_parameter(0x34);
  56. ili9341_send_cmd_parameter(0x02);
  57. ili9341_send_cmd(0xCF);
  58. ili9341_send_cmd_parameter(0x00);
  59. ili9341_send_cmd_parameter(0xC1);
  60. ili9341_send_cmd_parameter(0x30);
  61. ili9341_send_cmd(0xE8);
  62. ili9341_send_cmd_parameter(0x85);
  63. ili9341_send_cmd_parameter(0x00);
  64. ili9341_send_cmd_parameter(0x78);
  65. ili9341_send_cmd(0xEA);
  66. ili9341_send_cmd_parameter(0x00);
  67. ili9341_send_cmd_parameter(0x00);
  68. ili9341_send_cmd(0xED);
  69. ili9341_send_cmd_parameter(0x64);
  70. ili9341_send_cmd_parameter(0x03);
  71. ili9341_send_cmd_parameter(0x12);
  72. ili9341_send_cmd_parameter(0x81);
  73. ili9341_send_cmd(0xF7);
  74. ili9341_send_cmd_parameter(0x20);
  75. ili9341_send_cmd(0xC0);
  76. ili9341_send_cmd_parameter(0x23);
  77. ili9341_send_cmd(0xC1);
  78. ili9341_send_cmd_parameter(0x10);
  79. ili9341_send_cmd(0xC5);
  80. ili9341_send_cmd_parameter(0x3e);
  81. ili9341_send_cmd_parameter(0x28);
  82. ili9341_send_cmd(0xC7);
  83. ili9341_send_cmd_parameter(0x86);
  84. ili9341_send_cmd(0x36);
  85. if (g_Ili9341Info.width == 240)
  86. ili9341_send_cmd_parameter(0x48); // for 240x320
  87. else
  88. ili9341_send_cmd_parameter(0xE8); // for 320x240
  89. ili9341_send_cmd(0x3A);
  90. ili9341_send_cmd_parameter(0x55);
  91. ili9341_send_cmd(0xB1);
  92. ili9341_send_cmd_parameter(0x00);
  93. ili9341_send_cmd_parameter(0x18);
  94. ili9341_send_cmd(0xB6);
  95. ili9341_send_cmd_parameter(0x08);
  96. ili9341_send_cmd_parameter(0x82);
  97. ili9341_send_cmd_parameter(0x27);
  98. ili9341_send_cmd(0xF2);
  99. ili9341_send_cmd_parameter(0x00);
  100. ili9341_send_cmd(0x26);
  101. ili9341_send_cmd_parameter(0x01);
  102. ili9341_send_cmd(0xE0);
  103. ili9341_send_cmd_parameter(0x0F);
  104. ili9341_send_cmd_parameter(0x31);
  105. ili9341_send_cmd_parameter(0x2B);
  106. ili9341_send_cmd_parameter(0x0C);
  107. ili9341_send_cmd_parameter(0x0E);
  108. ili9341_send_cmd_parameter(0x08);
  109. ili9341_send_cmd_parameter(0x4E);
  110. ili9341_send_cmd_parameter(0xF1);
  111. ili9341_send_cmd_parameter(0x37);
  112. ili9341_send_cmd_parameter(0x07);
  113. ili9341_send_cmd_parameter(0x10);
  114. ili9341_send_cmd_parameter(0x03);
  115. ili9341_send_cmd_parameter(0x0E);
  116. ili9341_send_cmd_parameter(0x09);
  117. ili9341_send_cmd_parameter(0x00);
  118. ili9341_send_cmd(0xE1);
  119. ili9341_send_cmd_parameter(0x00);
  120. ili9341_send_cmd_parameter(0x0E);
  121. ili9341_send_cmd_parameter(0x14);
  122. ili9341_send_cmd_parameter(0x03);
  123. ili9341_send_cmd_parameter(0x11);
  124. ili9341_send_cmd_parameter(0x07);
  125. ili9341_send_cmd_parameter(0x31);
  126. ili9341_send_cmd_parameter(0xC1);
  127. ili9341_send_cmd_parameter(0x48);
  128. ili9341_send_cmd_parameter(0x08);
  129. ili9341_send_cmd_parameter(0x0F);
  130. ili9341_send_cmd_parameter(0x0C);
  131. ili9341_send_cmd_parameter(0x31);
  132. ili9341_send_cmd_parameter(0x36);
  133. ili9341_send_cmd_parameter(0x0F);
  134. ili9341_send_cmd(0x11);
  135. ili9341_delay_ms(120); // Delay 120ms
  136. ili9341_send_cmd(0x29); //Display on
  137. SET_BACKLIGHT_ON;
  138. return RT_EOK;
  139. }
  140. #if defined(NU_PKG_ILI9341_WITH_OFFSCREEN_FRAMEBUFFER)
  141. static void ili9341_fillrect(uint16_t *pixels, struct rt_device_rect_info *pRectInfo)
  142. {
  143. ili9341_set_column(pRectInfo->x, pRectInfo->x + pRectInfo->width);
  144. ili9341_set_page(pRectInfo->y, pRectInfo->y + pRectInfo->height);
  145. ili9341_send_cmd(0x2c);
  146. ili9341_send_pixels(pixels, pRectInfo->height * pRectInfo->width * 2);
  147. }
  148. #endif
  149. static void ili9341_fillscreen(rt_uint16_t color)
  150. {
  151. #if defined(NU_PKG_ILI9341_WITH_OFFSCREEN_FRAMEBUFFER)
  152. struct rt_device_rect_info rectinfo = { 0, 0, XSIZE_PHYS, YSIZE_PHYS };
  153. int pixel_count = XSIZE_PHYS * YSIZE_PHYS;
  154. rt_uint16_t *pu16ShadowBuf = (rt_uint16_t *)g_Ili9341Info.framebuffer;
  155. while (pixel_count--)
  156. {
  157. *pu16ShadowBuf++ = color;
  158. }
  159. ili9341_fillrect((uint16_t *)g_Ili9341Info.framebuffer, &rectinfo);
  160. #else
  161. ili9341_set_column(0, (XSIZE_PHYS - 1));
  162. ili9341_set_page(0, (YSIZE_PHYS - 1));
  163. ili9341_send_cmd(0x2c);
  164. for (int i = 0; i < (XSIZE_PHYS * YSIZE_PHYS); i++)
  165. ili9341_send_pixel_data(color);
  166. #endif
  167. }
  168. static void ili9341_lcd_set_pixel(const char *color, int x, int y)
  169. {
  170. ili9341_set_column(x, x);
  171. ili9341_set_page(y, y);
  172. ili9341_send_cmd(0x2c);
  173. ili9341_send_pixel_data(*(uint16_t *)color);
  174. }
  175. static void ili9341_lcd_draw_hline(const char *pixel, int x1, int x2, int y)
  176. {
  177. ili9341_set_column(x1, x2);
  178. ili9341_set_page(y, y);
  179. ili9341_send_cmd(0x2c);
  180. for (; x1 < x2; x1++)
  181. ili9341_send_pixel_data(*(uint16_t *)pixel);
  182. }
  183. static void ili9341_lcd_draw_vline(const char *pixel, int x, int y1, int y2)
  184. {
  185. ili9341_set_column(x, x);
  186. ili9341_set_page(y1, y2);
  187. ili9341_send_cmd(0x2c);
  188. for (; y1 < y2; y1++)
  189. ili9341_send_pixel_data(*(uint16_t *)pixel);
  190. }
  191. static void ili9341_lcd_blit_line(const char *pixels, int x, int y, rt_size_t size)
  192. {
  193. rt_uint16_t *ptr = (rt_uint16_t *)pixels;
  194. ili9341_set_column(x, x + size);
  195. ili9341_set_page(y, y);
  196. ili9341_send_cmd(0x2c);
  197. while (size--)
  198. ili9341_send_pixel_data(*ptr++);
  199. }
  200. static rt_err_t ili9341_lcd_open(rt_device_t dev, rt_uint16_t oflag)
  201. {
  202. return RT_EOK;
  203. }
  204. static rt_err_t ili9341_lcd_close(rt_device_t dev)
  205. {
  206. return RT_EOK;
  207. }
  208. static rt_err_t ili9341_lcd_control(rt_device_t dev, int cmd, void *args)
  209. {
  210. switch (cmd)
  211. {
  212. case RTGRAPHIC_CTRL_GET_INFO:
  213. {
  214. struct rt_device_graphic_info *info;
  215. info = (struct rt_device_graphic_info *) args;
  216. RT_ASSERT(info != RT_NULL);
  217. rt_memcpy(args, (void *)&g_Ili9341Info, sizeof(struct rt_device_graphic_info));
  218. }
  219. break;
  220. case RTGRAPHIC_CTRL_RECT_UPDATE:
  221. {
  222. #if defined(NU_PKG_ILI9341_WITH_OFFSCREEN_FRAMEBUFFER)
  223. struct rt_device_rect_info *psRectInfo = args;
  224. rt_uint16_t *pixels = (rt_uint16_t *)g_Ili9341Info.framebuffer;
  225. RT_ASSERT(args != RT_NULL);
  226. if (psRectInfo->y >= g_Ili9341Info.height)
  227. {
  228. int buf_id = psRectInfo->y / g_Ili9341Info.height;
  229. pixels += g_Ili9341Info.width * g_Ili9341Info.height * buf_id;
  230. psRectInfo->y %= g_Ili9341Info.height;
  231. }
  232. ili9341_fillrect(pixels, psRectInfo);
  233. #else
  234. /* nothong to be done */
  235. #endif
  236. }
  237. break;
  238. default:
  239. break;
  240. }
  241. return RT_EOK;
  242. }
  243. static struct rt_device lcd_device;
  244. static struct rt_device_graphic_ops ili9341_ops =
  245. {
  246. ili9341_lcd_set_pixel,
  247. ili9341_lcd_get_pixel,
  248. ili9341_lcd_draw_hline,
  249. ili9341_lcd_draw_vline,
  250. ili9341_lcd_blit_line
  251. };
  252. int rt_hw_lcd_ili9341_init(void)
  253. {
  254. ili9341_pin_init();
  255. /* register lcd device */
  256. lcd_device.type = RT_Device_Class_Graphic;
  257. lcd_device.init = ili9341_lcd_init;
  258. lcd_device.open = ili9341_lcd_open;
  259. lcd_device.close = ili9341_lcd_close;
  260. lcd_device.control = ili9341_lcd_control;
  261. lcd_device.read = RT_NULL;
  262. lcd_device.write = RT_NULL;
  263. lcd_device.user_data = &ili9341_ops;
  264. #if defined(NU_PKG_ILI9341_WITH_OFFSCREEN_FRAMEBUFFER)
  265. g_Ili9341Info.framebuffer = rt_malloc_align((DEF_VRAM_BUFFER_NUMBER * g_Ili9341Info.width * g_Ili9341Info.height * (g_Ili9341Info.bits_per_pixel / 8)) + 32, 32);
  266. RT_ASSERT(g_Ili9341Info.framebuffer != RT_NULL);
  267. #endif
  268. /* register graphic device driver */
  269. rt_device_register(&lcd_device, "lcd", RT_DEVICE_FLAG_RDWR | RT_DEVICE_FLAG_STANDALONE);
  270. return 0;
  271. }
  272. #ifdef RT_USING_FINSH
  273. #define LINE_LEN 32
  274. static void lcd_test(int argc, char *argv[])
  275. {
  276. uint16_t pixels[LINE_LEN];
  277. uint16_t color;
  278. int x, y, i;
  279. x = y = 100;
  280. ili9341_lcd_init(NULL);
  281. color = 0x0; //Black, RGB
  282. rt_kprintf("Brush 0x%X on screen.\n", color);
  283. ili9341_fillscreen(color);
  284. ili9341_lcd_get_pixel((char *)&color, x, y);
  285. rt_kprintf("lcd get pixel, pixel: 0x%X, x: %d, y: %d\n", color, x, y);
  286. color = 0xffff; //White, RGB
  287. rt_kprintf("Brush 0x%X on screen.\n", color);
  288. ili9341_fillscreen(color);
  289. ili9341_lcd_get_pixel((char *)&color, x, y);
  290. rt_kprintf("lcd get pixel, pixel: 0x%X, x: %d, y: %d\n", color, x, y);
  291. color = 0x1f; //Blue, RGB
  292. rt_kprintf("Brush 0x%X on screen.\n", color);
  293. ili9341_fillscreen(color);
  294. ili9341_lcd_get_pixel((char *)&color, x, y);
  295. rt_kprintf("lcd get pixel, pixel: 0x%X, x: %d, y: %d\n", color, x, y);
  296. color = 0x07e0; //Green, RGB
  297. rt_kprintf("Brush 0x%X on screen.\n", color);
  298. ili9341_fillscreen(color);
  299. ili9341_lcd_get_pixel((char *)&color, x, y);
  300. rt_kprintf("lcd get pixel, pixel: 0x%X, x: %d, y: %d\n", color, x, y);
  301. color = 0xf800; //Red, RGB
  302. rt_kprintf("Brush 0x%X on screen.\n", color);
  303. ili9341_fillscreen(color);
  304. ili9341_lcd_get_pixel((char *)&color, x, y);
  305. rt_kprintf("lcd get pixel, pixel: 0x%X, x: %d, y: %d\n", color, x, y);
  306. color = 0xffff; //White, RGB
  307. rt_kprintf("lcd draw hline, pixel: 0x%X, x1: %d, x2: %d, y: %d\n", color, x, x + 20, y);
  308. ili9341_lcd_draw_hline((const char *)&color, x, x + 20, y);
  309. color = 0xffff; //White, RGB
  310. rt_kprintf("lcd draw vline, pixel: 0x%X, x: %d, y: %d\n", color, y, y + 20);
  311. ili9341_lcd_draw_vline((const char *)&color, x, y, y + 20);
  312. for (i = 0; i < LINE_LEN; i++)
  313. pixels[i] = 20 + i * 5;
  314. x = y = 50;
  315. rt_kprintf("lcd blit line, start: x: %d, y: %d\n", x, y);
  316. ili9341_lcd_blit_line((const char *)&pixels[0], x, y, LINE_LEN);
  317. x = y = 200;
  318. color = 0x07E0; //Green, RGB
  319. rt_kprintf("lcd set pixel, pixel: 0x%X, x: %d, y: %d\n", color, x, y);
  320. ili9341_lcd_set_pixel((const char *)&color, x, y);
  321. color = 0x0;
  322. ili9341_lcd_get_pixel((char *)&color, x, y);
  323. rt_kprintf("lcd get pixel, pixel: 0x%X, x: %d, y: %d\n", color, x, y);
  324. x = y = 200;
  325. color = 0x1f; //Blue, RGB
  326. rt_kprintf("lcd set pixel, pixel: 0x%X, x: %d, y: %d\n", color, x, y);
  327. ili9341_lcd_set_pixel((const char *)&color, x, y);
  328. color = 0x0;
  329. ili9341_lcd_get_pixel((char *)&color, x, y);
  330. rt_kprintf("lcd get pixel, pixel: 0x%X, x: %d, y: %d\n", color, x, y);
  331. x = y = 200;
  332. color = 0xf800; //Red, RGB
  333. rt_kprintf("lcd set pixel, pixel: 0x%X, x: %d, y: %d\n", color, x, y);
  334. ili9341_lcd_set_pixel((const char *)&color, x, y);
  335. color = 0x0;
  336. ili9341_lcd_get_pixel((char *)&color, x, y);
  337. rt_kprintf("lcd get pixel, pixel: 0x%X, x: %d, y: %d\n", color, x, y);
  338. }
  339. MSH_CMD_EXPORT(lcd_test, test lcd display);
  340. #endif
  341. #endif /* if defined(NU_PKG_USING_ILI9341) */