drv_lcd.c 7.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253
  1. /*
  2. * Copyright (c) 2021-2024 HPMicro
  3. *
  4. * SPDX-License-Identifier: BSD-3-Clause
  5. *
  6. * Change Logs:
  7. * Date Author Notes
  8. * 2024-02-20 HPMicro First version
  9. */
  10. #include <rtthread.h>
  11. #ifdef BSP_USING_RTT_LCD_DRIVER
  12. #include "board.h"
  13. #include "hpm_l1c_drv.h"
  14. #include "hpm_lcdc_drv.h"
  15. #include "hpm_pdma_drv.h"
  16. #include "hpm_panel.h"
  17. #define LCD_BITS_PER_PIXEL 16
  18. #define LCD_PIXEL_FORMAT RTGRAPHIC_PIXEL_FORMAT_RGB565
  19. #define LCD_LAYER_INDEX (0)
  20. #define LCD_LAYER_DONE_MASK (1U << LCD_LAYER_INDEX)
  21. struct hpm_lcd
  22. {
  23. LCDC_Type *lcd_base;
  24. rt_uint8_t lcd_irq;
  25. rt_uint8_t lcd_irq_priority;
  26. struct rt_semaphore lcd_lock;
  27. char *bus_name;
  28. struct rt_device parent;
  29. struct rt_device_graphic_info lcd_info;
  30. uint32_t lcd_buffer_size;
  31. };
  32. static rt_err_t hpm_lcd_init(struct rt_device *device);
  33. static rt_err_t hpm_lcd_control(struct rt_device *device, int cmd, void *args);
  34. static int hpm_lcdc_init(struct hpm_lcd *lcd, struct rt_device_graphic_info *info);
  35. static uint8_t __attribute__((section(".framebuffer"), aligned(HPM_L1C_CACHELINE_SIZE))) lcdc_framebuffer[PANEL_SIZE_WIDTH * PANEL_SIZE_HEIGHT * LCD_BITS_PER_PIXEL / 8];
  36. #ifdef RT_USING_DEVICE_OPS
  37. const struct rt_device_ops hpm_lcd_ops = {
  38. .init = hpm_lcd_init,
  39. .open = RT_NULL,
  40. .close = RT_NULL,
  41. .read = RT_NULL,
  42. .write = RT_NULL,
  43. .control = hpm_lcd_control,
  44. };
  45. #endif
  46. static struct hpm_lcd hpm_lcds[] =
  47. {
  48. {
  49. .bus_name = "lcd0",
  50. .lcd_buffer_size = (PANEL_SIZE_WIDTH * PANEL_SIZE_HEIGHT * LCD_BITS_PER_PIXEL / 8),
  51. .lcd_base = HPM_LCDC,
  52. .lcd_irq = BOARD_LCD_IRQ,
  53. #if defined(BSP_RTT_LCD_IRQ_PRIORITY)
  54. .lcd_irq_priority = BSP_RTT_LCD_IRQ_PRIORITY,
  55. #else
  56. .lcd_irq_priority = 7,
  57. #endif
  58. .parent.type = RT_Device_Class_Graphic,
  59. #ifdef RT_USING_DEVICE_OPS
  60. .parent.ops = &hpm_lcd_ops,
  61. #else
  62. .parent.init = hpm_lcd_init,
  63. .parent.open = RT_NULL,
  64. .parent.close = RT_NULL,
  65. .parent.read = RT_NULL,
  66. .parent.write = RT_NULL,
  67. .parent.control = hpm_lcd_control,
  68. #endif
  69. },
  70. };
  71. SDK_DECLARE_EXT_ISR_M(BOARD_LCD_IRQ, isr_lcd_d0)
  72. void isr_lcd_d0(void)
  73. {
  74. lcdc_disable_interrupt(hpm_lcds[0].lcd_base, LCDC_INT_EN_VSYNC_MASK);
  75. rt_sem_release(&hpm_lcds[0].lcd_lock);
  76. lcdc_clear_status(hpm_lcds[0].lcd_base, LCDC_ST_VSYNC_MASK);
  77. }
  78. static rt_err_t hpm_lcd_init(struct rt_device *device)
  79. {
  80. /* nothing, right now */
  81. (void *)device;
  82. return RT_EOK;
  83. }
  84. static rt_err_t hpm_lcd_control(struct rt_device *device, int cmd, void *args)
  85. {
  86. uint32_t aligned_start, aligned_end, aligned_size;
  87. struct hpm_lcd *lcd = (struct hpm_lcd *)device->user_data;
  88. hpm_panel_t *panel = hpm_panel_find_device_default();
  89. struct rt_device_graphic_info *info = RT_NULL;
  90. uint32_t buffer;
  91. switch (cmd)
  92. {
  93. case RTGRAPHIC_CTRL_SET_MODE:
  94. info = (struct rt_device_graphic_info *)args;
  95. rt_sem_trytake(&lcd->lcd_lock);
  96. lcdc_disable_interrupt(lcd->lcd_base, LCDC_INT_EN_VSYNC_MASK);
  97. rt_thread_delay(10);
  98. hpm_lcdc_init(lcd, info);
  99. break;
  100. case RTGRAPHIC_CTRL_RECT_UPDATE:
  101. if (args != RT_NULL)
  102. {
  103. buffer = (uint32_t)args;
  104. }
  105. else
  106. {
  107. buffer = (uint32_t)lcd->lcd_info.framebuffer;
  108. }
  109. if (l1c_dc_is_enabled())
  110. {
  111. aligned_start = HPM_L1C_CACHELINE_ALIGN_DOWN(buffer);
  112. aligned_end = HPM_L1C_CACHELINE_ALIGN_UP(buffer + lcd->lcd_buffer_size);
  113. aligned_size = aligned_end - aligned_start;
  114. l1c_dc_writeback(aligned_start, aligned_size);
  115. }
  116. if (lcdc_layer_control_shadow_loaded(lcd->lcd_base, 0))
  117. {
  118. lcdc_layer_set_next_buffer(lcd->lcd_base, 0, (rt_uint32_t)buffer);
  119. }
  120. break;
  121. case RTGRAPHIC_CTRL_WAIT_VSYNC:
  122. rt_sem_trytake(&lcd->lcd_lock);
  123. lcdc_enable_interrupt(lcd->lcd_base, LCDC_INT_EN_VSYNC_MASK);
  124. rt_sem_take(&lcd->lcd_lock, RT_WAITING_FOREVER);
  125. break;
  126. case RTGRAPHIC_CTRL_POWERON:
  127. hpm_panel_set_backlight(panel, true);
  128. break;
  129. case RTGRAPHIC_CTRL_POWEROFF:
  130. hpm_panel_set_backlight(panel, false);
  131. break;
  132. case RTGRAPHIC_CTRL_GET_INFO:
  133. info = (struct rt_device_graphic_info *)args;
  134. RT_ASSERT(info != RT_NULL);
  135. info->pixel_format = lcd->lcd_info.pixel_format;
  136. info->bits_per_pixel = 16;
  137. info->width = lcd->lcd_info.width;
  138. info->height = lcd->lcd_info.height;
  139. info->framebuffer = lcd->lcd_info.framebuffer;
  140. break;
  141. default:
  142. break;
  143. }
  144. }
  145. static int hpm_lcdc_init(struct hpm_lcd *lcd, struct rt_device_graphic_info *info)
  146. {
  147. lcdc_config_t config = {0};
  148. display_pixel_format_t pixel_format;
  149. lcdc_get_default_config(lcd->lcd_base, &config);
  150. board_panel_para_to_lcdc(&config);
  151. if (info->framebuffer == RT_NULL)
  152. {
  153. return -RT_ERROR;
  154. }
  155. rt_memcpy(&lcd->lcd_info, info, sizeof(struct rt_device_graphic_info));
  156. lcd->lcd_info.framebuffer = lcdc_framebuffer;
  157. if (info->pixel_format == RTGRAPHIC_PIXEL_FORMAT_RGB565)
  158. {
  159. pixel_format = display_pixel_format_rgb565;
  160. }
  161. else if (info->pixel_format == RTGRAPHIC_PIXEL_FORMAT_ARGB888)
  162. {
  163. pixel_format = display_pixel_format_rgb565;
  164. }
  165. else {
  166. return -RT_ERROR;
  167. }
  168. lcdc_init(lcd->lcd_base, &config);
  169. memset(lcd->lcd_info.framebuffer, 0, info->width * info->height * info->bits_per_pixel / 8);
  170. lcdc_layer_config_t layer;
  171. lcdc_get_default_layer_config(lcd->lcd_base, &layer, pixel_format, LCD_LAYER_INDEX);
  172. layer.position_x = 0;
  173. layer.position_y = 0;
  174. layer.width = info->width;
  175. layer.height = info->height;
  176. layer.buffer = (rt_uint32_t)lcd->lcd_info.framebuffer;
  177. layer.background.u = 0;
  178. if (status_success != lcdc_config_layer(lcd->lcd_base, LCD_LAYER_INDEX, &layer, true)) {
  179. return -RT_ERROR;
  180. }
  181. lcdc_turn_on_display(lcd->lcd_base);
  182. lcdc_enable_interrupt(lcd->lcd_base, LCDC_INT_EN_VSYNC_MASK);
  183. intc_m_enable_irq_with_priority(lcd->lcd_irq, lcd->lcd_irq_priority);
  184. return 0;
  185. }
  186. int drv_lcd_hw_init(void)
  187. {
  188. rt_err_t result = RT_EOK;
  189. struct rt_device_graphic_info lcd_info;
  190. for (uint32_t i = 0; i < sizeof(hpm_lcds) / sizeof(hpm_lcds[0]); i++)
  191. {
  192. struct hpm_lcd *lcd = &hpm_lcds[i];
  193. struct rt_device *device = &lcd->parent;
  194. lcd->parent.user_data = lcd;
  195. /* init lcd_lock semaphore */
  196. result = rt_sem_init(&hpm_lcds[0].lcd_lock, "lcd_lock", 0, RT_IPC_FLAG_FIFO);
  197. if (result != RT_EOK)
  198. {
  199. result = -RT_ENOMEM;
  200. goto __exit;
  201. }
  202. /* config LCD dev info */
  203. lcd_info.height = PANEL_SIZE_HEIGHT;
  204. lcd_info.width = PANEL_SIZE_WIDTH;
  205. lcd_info.bits_per_pixel = LCD_BITS_PER_PIXEL;
  206. lcd_info.pixel_format = LCD_PIXEL_FORMAT;
  207. lcd_info.framebuffer = lcdc_framebuffer;
  208. /* register lcd device */
  209. rt_device_register(device, hpm_lcds[i].bus_name, RT_DEVICE_FLAG_RDWR);
  210. board_init_lcd();
  211. if (hpm_lcdc_init(&hpm_lcds[i], &lcd_info) != RT_EOK)
  212. {
  213. result = -RT_ERROR;
  214. goto __exit;
  215. }
  216. __exit:
  217. if (result != RT_EOK)
  218. {
  219. rt_sem_delete(&hpm_lcds[i].lcd_lock);
  220. }
  221. return result;
  222. }
  223. }
  224. INIT_BOARD_EXPORT(drv_lcd_hw_init);
  225. #endif /* BSP_USING_RTT_LCD_DRIVER */