drv_lcd_mipi.c 7.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254
  1. /*
  2. * Copyright (c) 2006-2021, RT-Thread Development Team
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. *
  6. * Change Logs:
  7. * Date Author Notes
  8. * 2019-05-23 WillianChan first version
  9. */
  10. #include <board.h>
  11. #ifdef BSP_USING_LCD_MIPI
  12. #include <lcd_port.h>
  13. #include <string.h>
  14. DSI_HandleTypeDef hdsi;
  15. DSI_VidCfgTypeDef hdsi_video;
  16. LTDC_HandleTypeDef hltdc;
  17. struct stm32_lcd
  18. {
  19. struct rt_device parent;
  20. struct rt_device_graphic_info info;
  21. };
  22. struct stm32_lcd lcd;
  23. extern void stm32_mipi_lcd_init(void);
  24. extern void stm32_mipi_lcd_config(rt_uint32_t pixel_format);
  25. extern void stm32_mipi_display_on(void);
  26. extern void stm32_mipi_display_off(void);
  27. rt_err_t ltdc_init(void)
  28. {
  29. uint32_t lcd_clock = 27429;
  30. uint32_t lanebyte_clock = 62500;
  31. uint32_t HSA = LCD_HSYNC, HFP = LCD_HFP, HBP = LCD_HBP, HACT = LCD_WIDTH;
  32. uint32_t VSA = LCD_VSYNC, VFP = LCD_VFP, VBP = LCD_VBP, VACT = LCD_HEIGHT;
  33. stm32_mipi_lcd_init();
  34. __HAL_RCC_LTDC_CLK_ENABLE();
  35. __HAL_RCC_LTDC_FORCE_RESET();
  36. __HAL_RCC_LTDC_RELEASE_RESET();
  37. __HAL_RCC_DSI_CLK_ENABLE();
  38. __HAL_RCC_DSI_FORCE_RESET();
  39. __HAL_RCC_DSI_RELEASE_RESET();
  40. RCC_PeriphCLKInitTypeDef PeriphClkInitStruct;
  41. PeriphClkInitStruct.PeriphClockSelection = RCC_PERIPHCLK_LTDC;
  42. PeriphClkInitStruct.PLLSAI.PLLSAIN = 384;
  43. PeriphClkInitStruct.PLLSAI.PLLSAIR = 7;
  44. PeriphClkInitStruct.PLLSAIDivR = RCC_PLLSAIDIVR_2;
  45. HAL_RCCEx_PeriphCLKConfig(&PeriphClkInitStruct);
  46. HAL_NVIC_SetPriority(LTDC_IRQn, 3, 0);
  47. HAL_NVIC_SetPriority(DSI_IRQn, 3, 0);
  48. HAL_NVIC_EnableIRQ(LTDC_IRQn);
  49. HAL_NVIC_EnableIRQ(DSI_IRQn);
  50. DSI_PLLInitTypeDef dsi_pll;
  51. hdsi.Instance = DSI;
  52. hdsi.Init.NumberOfLanes = DSI_TWO_DATA_LANES;
  53. hdsi.Init.TXEscapeCkdiv = lanebyte_clock / 15620;
  54. dsi_pll.PLLNDIV = 125;
  55. dsi_pll.PLLIDF = DSI_PLL_IN_DIV2;
  56. dsi_pll.PLLODF = DSI_PLL_OUT_DIV1;
  57. HAL_DSI_DeInit(&hdsi);
  58. HAL_DSI_Init(&hdsi, &dsi_pll);
  59. hdsi_video.VirtualChannelID = 0;
  60. hdsi_video.ColorCoding = DSI_RGB565;
  61. hdsi_video.VSPolarity = DSI_VSYNC_ACTIVE_HIGH;
  62. hdsi_video.HSPolarity = DSI_HSYNC_ACTIVE_HIGH;
  63. hdsi_video.DEPolarity = DSI_DATA_ENABLE_ACTIVE_HIGH;
  64. hdsi_video.Mode = DSI_VID_MODE_BURST;
  65. hdsi_video.NullPacketSize = 0xFFF;
  66. hdsi_video.NumberOfChunks = 0;
  67. hdsi_video.PacketSize = HACT;
  68. hdsi_video.HorizontalSyncActive = (HSA * lanebyte_clock) / lcd_clock;
  69. hdsi_video.HorizontalBackPorch = (HBP * lanebyte_clock) / lcd_clock;
  70. hdsi_video.HorizontalLine = ((HACT + HSA + HBP + HFP) * lanebyte_clock) / lcd_clock;
  71. hdsi_video.VerticalSyncActive = VSA;
  72. hdsi_video.VerticalBackPorch = VBP;
  73. hdsi_video.VerticalFrontPorch = VFP;
  74. hdsi_video.VerticalActive = VACT;
  75. hdsi_video.LPCommandEnable = DSI_LP_COMMAND_ENABLE;
  76. hdsi_video.LPLargestPacketSize = 16;
  77. hdsi_video.LPVACTLargestPacketSize = 0;
  78. hdsi_video.LPHorizontalFrontPorchEnable = DSI_LP_HFP_ENABLE;
  79. hdsi_video.LPHorizontalBackPorchEnable = DSI_LP_HBP_ENABLE;
  80. hdsi_video.LPVerticalActiveEnable = DSI_LP_VACT_ENABLE;
  81. hdsi_video.LPVerticalFrontPorchEnable = DSI_LP_VFP_ENABLE;
  82. hdsi_video.LPVerticalBackPorchEnable = DSI_LP_VBP_ENABLE;
  83. hdsi_video.LPVerticalSyncActiveEnable = DSI_LP_VSYNC_ENABLE;
  84. HAL_DSI_ConfigVideoMode(&hdsi, &hdsi_video);
  85. DSI_PHY_TimerTypeDef dsi_phy;
  86. dsi_phy.ClockLaneHS2LPTime = 35;
  87. dsi_phy.ClockLaneLP2HSTime = 35;
  88. dsi_phy.DataLaneHS2LPTime = 35;
  89. dsi_phy.DataLaneLP2HSTime = 35;
  90. dsi_phy.DataLaneMaxReadTime = 0;
  91. dsi_phy.StopWaitTime = 10;
  92. HAL_DSI_ConfigPhyTimer(&hdsi, &dsi_phy);
  93. hltdc.Instance = LTDC;
  94. hltdc.Init.PCPolarity = LTDC_PCPOLARITY_IPC;
  95. hltdc.Init.HorizontalSync = (HSA - 1);
  96. hltdc.Init.AccumulatedHBP = (HSA + HBP - 1);
  97. hltdc.Init.AccumulatedActiveW = (LCD_WIDTH + HSA + HBP - 1);
  98. hltdc.Init.TotalWidth = (LCD_WIDTH + HSA + HBP + HFP - 1);
  99. hltdc.LayerCfg->ImageWidth = LCD_WIDTH;
  100. hltdc.LayerCfg->ImageHeight = LCD_HEIGHT;
  101. hltdc.Init.Backcolor.Blue = 0x00;
  102. hltdc.Init.Backcolor.Green = 0x00;
  103. hltdc.Init.Backcolor.Red = 0x00;
  104. HAL_LTDCEx_StructInitFromVideoConfig(&hltdc, &(hdsi_video));
  105. HAL_LTDC_Init(&(hltdc));
  106. HAL_DSI_Start(&(hdsi));
  107. stm32_mipi_lcd_config(RTGRAPHIC_PIXEL_FORMAT_RGB565);
  108. return RT_EOK;
  109. }
  110. void ltdc_layer_init(uint16_t index, uint32_t framebuffer)
  111. {
  112. LTDC_LayerCfgTypeDef layer_cfg;
  113. layer_cfg.WindowX0 = 0;
  114. layer_cfg.WindowX1 = LCD_WIDTH;
  115. layer_cfg.WindowY0 = 0;
  116. layer_cfg.WindowY1 = LCD_HEIGHT;
  117. layer_cfg.PixelFormat = LTDC_PIXEL_FORMAT_RGB565;
  118. layer_cfg.BlendingFactor1 = LTDC_BLENDING_FACTOR1_PAxCA;
  119. layer_cfg.BlendingFactor2 = LTDC_BLENDING_FACTOR2_PAxCA;
  120. layer_cfg.Alpha = 255;
  121. layer_cfg.Alpha0 = 0;
  122. layer_cfg.ImageWidth = LCD_WIDTH;
  123. layer_cfg.ImageHeight = LCD_HEIGHT;
  124. layer_cfg.Backcolor.Blue = 0;
  125. layer_cfg.Backcolor.Green = 0;
  126. layer_cfg.Backcolor.Red = 0;
  127. layer_cfg.FBStartAdress = framebuffer;
  128. HAL_LTDC_ConfigLayer(&hltdc, &layer_cfg, index);
  129. }
  130. void LTDC_IRQHandler(void)
  131. {
  132. rt_interrupt_enter();
  133. HAL_LTDC_IRQHandler(&hltdc);
  134. rt_interrupt_leave();
  135. }
  136. static rt_err_t stm32_lcd_init(rt_device_t device)
  137. {
  138. lcd.info.width = LCD_WIDTH;
  139. lcd.info.height = LCD_HEIGHT;
  140. lcd.info.pixel_format = RTGRAPHIC_PIXEL_FORMAT_RGB565;
  141. lcd.info.bits_per_pixel = 16;
  142. lcd.info.framebuffer = (void *)rt_malloc_align(LCD_WIDTH * LCD_HEIGHT * (lcd.info.bits_per_pixel / 8), 32);;
  143. memset(lcd.info.framebuffer, 0, LCD_WIDTH * LCD_HEIGHT * (lcd.info.bits_per_pixel / 8));
  144. ltdc_init();
  145. ltdc_layer_init(0, (uint32_t)lcd.info.framebuffer);
  146. return RT_EOK;
  147. }
  148. static rt_err_t stm32_lcd_control(rt_device_t device, int cmd, void *args)
  149. {
  150. switch(cmd)
  151. {
  152. case RTGRAPHIC_CTRL_RECT_UPDATE:
  153. break;
  154. case RTGRAPHIC_CTRL_POWERON:
  155. stm32_mipi_display_on();
  156. break;
  157. case RTGRAPHIC_CTRL_POWEROFF:
  158. stm32_mipi_display_off();
  159. break;
  160. case RTGRAPHIC_CTRL_GET_INFO:
  161. rt_memcpy(args, &lcd.info, sizeof(lcd.info));
  162. break;
  163. case RTGRAPHIC_CTRL_SET_MODE:
  164. break;
  165. case RTGRAPHIC_CTRL_GET_EXT:
  166. break;
  167. }
  168. return RT_EOK;
  169. }
  170. int rt_hw_lcd_init(void)
  171. {
  172. rt_err_t ret;
  173. rt_memset(&lcd, 0x00, sizeof(lcd));
  174. lcd.parent.type = RT_Device_Class_Graphic;
  175. lcd.parent.init = stm32_lcd_init;
  176. lcd.parent.open = RT_NULL;
  177. lcd.parent.close = RT_NULL;
  178. lcd.parent.read = RT_NULL;
  179. lcd.parent.write = RT_NULL;
  180. lcd.parent.control = stm32_lcd_control;
  181. lcd.parent.user_data = (void *)&lcd.info;
  182. ret = rt_device_register(&lcd.parent, "lcd", RT_DEVICE_FLAG_RDWR);
  183. return ret;
  184. }
  185. INIT_DEVICE_EXPORT(rt_hw_lcd_init);
  186. RT_WEAK void stm32_mipi_lcd_init(void)
  187. {
  188. rt_kprintf("please Implementation function %s\n", __func__);
  189. }
  190. RT_WEAK void stm32_mipi_lcd_config(rt_uint32_t pixel_format)
  191. {
  192. rt_kprintf("please Implementation function %s\n", __func__);
  193. }
  194. RT_WEAK void stm32_mipi_display_on(void)
  195. {
  196. rt_kprintf("please Implementation function %s\n", __func__);
  197. }
  198. RT_WEAK void stm32_mipi_display_off(void)
  199. {
  200. rt_kprintf("please Implementation function %s\n", __func__);
  201. }
  202. #endif /* BSP_USING_LCD_MIPI */