ili9341.c 9.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313
  1. /*
  2. * File : ili9341.c
  3. * This file is part of RT-Thread RTOS
  4. * COPYRIGHT (C) 2020, RT-Thread Development Team
  5. *
  6. * This program is free software; you can redistribute it and/or modify
  7. * it under the terms of the GNU General Public License as published by
  8. * the Free Software Foundation; either version 2 of the License, or
  9. * (at your option) any later version.
  10. *
  11. * This program is distributed in the hope that it will be useful,
  12. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  13. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  14. * GNU General Public License for more details.
  15. *
  16. * You should have received a copy of the GNU General Public License along
  17. * with this program; if not, write to the Free Software Foundation, Inc.,
  18. * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
  19. *
  20. * Change Logs:
  21. * Date Author Notes
  22. * 2020-08-11 RT-Thread the first version
  23. */
  24. #include "rtthread.h"
  25. #include "stm32f4xx_hal.h"
  26. #include "ili9341.h"
  27. /**
  28. * @brief LCD Control pin
  29. */
  30. #define LCD_NCS_PIN GPIO_PIN_2
  31. #define LCD_NCS_GPIO_PORT GPIOC
  32. #define LCD_NCS_GPIO_CLK_ENABLE() __HAL_RCC_GPIOC_CLK_ENABLE()
  33. #define LCD_NCS_GPIO_CLK_DISABLE() __HAL_RCC_GPIOC_CLK_DISABLE()
  34. /**
  35. * @brief LCD Command/data pin
  36. */
  37. #define LCD_WRX_PIN GPIO_PIN_13
  38. #define LCD_WRX_GPIO_PORT GPIOD
  39. #define LCD_WRX_GPIO_CLK_ENABLE() __HAL_RCC_GPIOD_CLK_ENABLE()
  40. #define LCD_WRX_GPIO_CLK_DISABLE() __HAL_RCC_GPIOD_CLK_DISABLE()
  41. #define LCD_RDX_PIN GPIO_PIN_12
  42. #define LCD_RDX_GPIO_PORT GPIOD
  43. #define LCD_RDX_GPIO_CLK_ENABLE() __HAL_RCC_GPIOD_CLK_ENABLE()
  44. #define LCD_RDX_GPIO_CLK_DISABLE() __HAL_RCC_GPIOD_CLK_DISABLE()
  45. /* Maximum Timeout values for flags waiting loops */
  46. #define SPIx_TIMEOUT_MAX ((uint32_t)0x1000)
  47. /* Chip Select macro definition */
  48. #define LCD_CS_LOW() HAL_GPIO_WritePin(LCD_NCS_GPIO_PORT, LCD_NCS_PIN, GPIO_PIN_RESET)
  49. #define LCD_CS_HIGH() HAL_GPIO_WritePin(LCD_NCS_GPIO_PORT, LCD_NCS_PIN, GPIO_PIN_SET)
  50. /* Set WRX High to send data */
  51. #define LCD_WRX_LOW() HAL_GPIO_WritePin(LCD_WRX_GPIO_PORT, LCD_WRX_PIN, GPIO_PIN_RESET)
  52. #define LCD_WRX_HIGH() HAL_GPIO_WritePin(LCD_WRX_GPIO_PORT, LCD_WRX_PIN, GPIO_PIN_SET)
  53. /* Set WRX High to send data */
  54. #define LCD_RDX_LOW() HAL_GPIO_WritePin(LCD_RDX_GPIO_PORT, LCD_RDX_PIN, GPIO_PIN_RESET)
  55. #define LCD_RDX_HIGH() HAL_GPIO_WritePin(LCD_RDX_GPIO_PORT, LCD_RDX_PIN, GPIO_PIN_SET)
  56. static uint8_t Is_LCD_IO_Initialized = 0;
  57. static SPI_HandleTypeDef SpiHandle;
  58. /**
  59. * @brief SPIx Bus initialization
  60. */
  61. static void SPIx_Init(void)
  62. {
  63. if(HAL_SPI_GetState(&SpiHandle) == HAL_SPI_STATE_RESET)
  64. {
  65. /* SPI configuration -----------------------------------------------------*/
  66. SpiHandle.Instance = SPI5;
  67. /* SPI baudrate is set to 5.6 MHz (PCLK2/SPI_BaudRatePrescaler = 90/16 = 5.625 MHz)
  68. */
  69. SpiHandle.Init.BaudRatePrescaler = SPI_BAUDRATEPRESCALER_16;
  70. /* On STM32F429I-Discovery, LCD ID cannot be read then keep a common configuration */
  71. /* for LCD and GYRO (SPI_DIRECTION_2LINES) */
  72. /* Note: To read a register a LCD, SPI_DIRECTION_1LINE should be set */
  73. SpiHandle.Init.Direction = SPI_DIRECTION_2LINES;
  74. SpiHandle.Init.CLKPhase = SPI_PHASE_1EDGE;
  75. SpiHandle.Init.CLKPolarity = SPI_POLARITY_LOW;
  76. SpiHandle.Init.CRCCalculation = SPI_CRCCALCULATION_DISABLED;
  77. SpiHandle.Init.CRCPolynomial = 7;
  78. SpiHandle.Init.DataSize = SPI_DATASIZE_8BIT;
  79. SpiHandle.Init.FirstBit = SPI_FIRSTBIT_MSB;
  80. SpiHandle.Init.NSS = SPI_NSS_SOFT;
  81. SpiHandle.Init.TIMode = SPI_TIMODE_DISABLED;
  82. SpiHandle.Init.Mode = SPI_MODE_MASTER;
  83. HAL_SPI_Init(&SpiHandle);
  84. }
  85. }
  86. /**
  87. * @brief Configures the LCD_SPI interface.
  88. */
  89. static void LCD_GPIO_Init(void)
  90. {
  91. GPIO_InitTypeDef GPIO_InitStructure;
  92. if(Is_LCD_IO_Initialized == 0)
  93. {
  94. Is_LCD_IO_Initialized = 1;
  95. /* Configure NCS in Output Push-Pull mode */
  96. LCD_WRX_GPIO_CLK_ENABLE();
  97. GPIO_InitStructure.Pin = LCD_WRX_PIN;
  98. GPIO_InitStructure.Mode = GPIO_MODE_OUTPUT_PP;
  99. GPIO_InitStructure.Pull = GPIO_NOPULL;
  100. GPIO_InitStructure.Speed = GPIO_SPEED_FAST;
  101. HAL_GPIO_Init(LCD_WRX_GPIO_PORT, &GPIO_InitStructure);
  102. LCD_RDX_GPIO_CLK_ENABLE();
  103. GPIO_InitStructure.Pin = LCD_RDX_PIN;
  104. GPIO_InitStructure.Mode = GPIO_MODE_OUTPUT_PP;
  105. GPIO_InitStructure.Pull = GPIO_NOPULL;
  106. GPIO_InitStructure.Speed = GPIO_SPEED_FAST;
  107. HAL_GPIO_Init(LCD_RDX_GPIO_PORT, &GPIO_InitStructure);
  108. /* Configure the LCD Control pins ----------------------------------------*/
  109. LCD_NCS_GPIO_CLK_ENABLE();
  110. /* Configure NCS in Output Push-Pull mode */
  111. GPIO_InitStructure.Pin = LCD_NCS_PIN;
  112. GPIO_InitStructure.Mode = GPIO_MODE_OUTPUT_PP;
  113. GPIO_InitStructure.Pull = GPIO_NOPULL;
  114. GPIO_InitStructure.Speed = GPIO_SPEED_FAST;
  115. HAL_GPIO_Init(LCD_NCS_GPIO_PORT, &GPIO_InitStructure);
  116. /* Set or Reset the control line */
  117. LCD_CS_LOW();
  118. LCD_CS_HIGH();
  119. SPIx_Init();
  120. }
  121. }
  122. /**
  123. * @brief Writes data to the selected LCD register.
  124. * @param data: data to lcd.
  125. * @retval None
  126. */
  127. static void ili9341_write_data(uint16_t data)
  128. {
  129. /* Set WRX to send data */
  130. LCD_WRX_HIGH();
  131. /* Reset LCD control line(/CS) and Send data */
  132. LCD_CS_LOW();
  133. HAL_SPI_Transmit(&SpiHandle, (uint8_t*) &data, 1, SPIx_TIMEOUT_MAX);
  134. /* Deselect: Chip Select high */
  135. LCD_CS_HIGH();
  136. }
  137. /**
  138. * @brief Writes to the selected LCD register.
  139. * @param reg: address of the selected register.
  140. * @retval None
  141. */
  142. static void ili9341_write_register(uint8_t reg)
  143. {
  144. /* Reset WRX to send command */
  145. LCD_WRX_LOW();
  146. /* Reset LCD control line(/CS) and Send command */
  147. LCD_CS_LOW();
  148. HAL_SPI_Transmit(&SpiHandle, (uint8_t*) &reg, 1, SPIx_TIMEOUT_MAX);
  149. /* Deselect: Chip Select high */
  150. LCD_CS_HIGH();
  151. }
  152. /**
  153. * @brief Power on the LCD.
  154. * @param None
  155. * @retval int
  156. */
  157. int ili9341_hw_init(void)
  158. {
  159. /* Initialize ILI9341 low level bus layer ----------------------------------*/
  160. LCD_GPIO_Init();
  161. /* Configure LCD */
  162. ili9341_write_register(0xCA);
  163. ili9341_write_data(0xC3);
  164. ili9341_write_data(0x08);
  165. ili9341_write_data(0x50);
  166. ili9341_write_register(LCD_POWERB);
  167. ili9341_write_data(0x00);
  168. ili9341_write_data(0xC1);
  169. ili9341_write_data(0x30);
  170. ili9341_write_register(LCD_POWER_SEQ);
  171. ili9341_write_data(0x64);
  172. ili9341_write_data(0x03);
  173. ili9341_write_data(0x12);
  174. ili9341_write_data(0x81);
  175. ili9341_write_register(LCD_DTCA);
  176. ili9341_write_data(0x85);
  177. ili9341_write_data(0x00);
  178. ili9341_write_data(0x78);
  179. ili9341_write_register(LCD_POWERA);
  180. ili9341_write_data(0x39);
  181. ili9341_write_data(0x2C);
  182. ili9341_write_data(0x00);
  183. ili9341_write_data(0x34);
  184. ili9341_write_data(0x02);
  185. ili9341_write_register(LCD_PRC);
  186. ili9341_write_data(0x20);
  187. ili9341_write_register(LCD_DTCB);
  188. ili9341_write_data(0x00);
  189. ili9341_write_data(0x00);
  190. ili9341_write_register(LCD_FRMCTR1);
  191. ili9341_write_data(0x00);
  192. ili9341_write_data(0x1B);
  193. ili9341_write_register(LCD_DFC);
  194. ili9341_write_data(0x0A);
  195. ili9341_write_data(0xA2);
  196. ili9341_write_register(LCD_POWER1);
  197. ili9341_write_data(0x10);
  198. ili9341_write_register(LCD_POWER2);
  199. ili9341_write_data(0x10);
  200. ili9341_write_register(LCD_VCOM1);
  201. ili9341_write_data(0x45);
  202. ili9341_write_data(0x15);
  203. ili9341_write_register(LCD_VCOM2);
  204. ili9341_write_data(0x90);
  205. ili9341_write_register(LCD_MAC);
  206. ili9341_write_data(0xC8);
  207. ili9341_write_register(LCD_3GAMMA_EN);
  208. ili9341_write_data(0x00);
  209. ili9341_write_register(LCD_RGB_INTERFACE);
  210. ili9341_write_data(0xC2);
  211. ili9341_write_register(LCD_DFC);
  212. ili9341_write_data(0x0A);
  213. ili9341_write_data(0xA7);
  214. ili9341_write_data(0x27);
  215. ili9341_write_data(0x04);
  216. /* Colomn address set */
  217. ili9341_write_register(LCD_COLUMN_ADDR);
  218. ili9341_write_data(0x00);
  219. ili9341_write_data(0x00);
  220. ili9341_write_data(0x00);
  221. ili9341_write_data(0xEF);
  222. /* Page address set */
  223. ili9341_write_register(LCD_PAGE_ADDR);
  224. ili9341_write_data(0x00);
  225. ili9341_write_data(0x00);
  226. ili9341_write_data(0x01);
  227. ili9341_write_data(0x3F);
  228. ili9341_write_register(LCD_INTERFACE);
  229. ili9341_write_data(0x01);
  230. ili9341_write_data(0x00);
  231. ili9341_write_data(0x06);
  232. ili9341_write_register(LCD_GRAM);
  233. rt_thread_mdelay(20);
  234. ili9341_write_register(LCD_GAMMA);
  235. ili9341_write_data(0x01);
  236. ili9341_write_register(LCD_PGAMMA);
  237. ili9341_write_data(0x0F);
  238. ili9341_write_data(0x29);
  239. ili9341_write_data(0x24);
  240. ili9341_write_data(0x0C);
  241. ili9341_write_data(0x0E);
  242. ili9341_write_data(0x09);
  243. ili9341_write_data(0x4E);
  244. ili9341_write_data(0x78);
  245. ili9341_write_data(0x3C);
  246. ili9341_write_data(0x09);
  247. ili9341_write_data(0x13);
  248. ili9341_write_data(0x05);
  249. ili9341_write_data(0x17);
  250. ili9341_write_data(0x11);
  251. ili9341_write_data(0x00);
  252. ili9341_write_register(LCD_NGAMMA);
  253. ili9341_write_data(0x00);
  254. ili9341_write_data(0x16);
  255. ili9341_write_data(0x1B);
  256. ili9341_write_data(0x04);
  257. ili9341_write_data(0x11);
  258. ili9341_write_data(0x07);
  259. ili9341_write_data(0x31);
  260. ili9341_write_data(0x33);
  261. ili9341_write_data(0x42);
  262. ili9341_write_data(0x05);
  263. ili9341_write_data(0x0C);
  264. ili9341_write_data(0x0A);
  265. ili9341_write_data(0x28);
  266. ili9341_write_data(0x2F);
  267. ili9341_write_data(0x0F);
  268. ili9341_write_register(LCD_SLEEP_OUT);
  269. rt_thread_mdelay(20);
  270. ili9341_write_register(LCD_DISPLAY_ON);
  271. /* GRAM start writing */
  272. ili9341_write_register(LCD_GRAM);
  273. return 0;
  274. }
  275. INIT_DEVICE_EXPORT(ili9341_hw_init);