drv_uart.c 6.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284
  1. /*
  2. * Copyright (c) 2006-2018, Synwit Technology Co.,Ltd.
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. *
  6. * Change Logs:
  7. * Date Author Notes
  8. * 2018-05-31 ZYH first version
  9. * 2018-12-10 Zohar_Lee format file
  10. * 2020-07-10 lik format file
  11. */
  12. #include "drv_uart.h"
  13. #ifdef RT_USING_SERIAL
  14. #ifdef BSP_USING_UART
  15. //#define DRV_DEBUG
  16. #define LOG_TAG "drv.uart"
  17. #include <drv_log.h>
  18. #if !defined(BSP_USING_UART0) && !defined(BSP_USING_UART1) && !defined(BSP_USING_UART2) && \
  19. !defined(BSP_USING_UART3)
  20. #error "Please define at least one BSP_USING_UARTx"
  21. /* this driver can be disabled at menuconfig -> RT-Thread Components -> Device Drivers */
  22. #endif
  23. enum
  24. {
  25. #ifdef BSP_USING_UART0
  26. UART0_INDEX,
  27. #endif
  28. #ifdef BSP_USING_UART1
  29. UART1_INDEX,
  30. #endif
  31. #ifdef BSP_USING_UART2
  32. UART2_INDEX,
  33. #endif
  34. #ifdef BSP_USING_UART3
  35. UART3_INDEX,
  36. #endif
  37. };
  38. static struct swm_uart_cfg uart_cfg[] =
  39. {
  40. #ifdef BSP_USING_UART0
  41. UART0_CFG,
  42. #endif
  43. #ifdef BSP_USING_UART1
  44. UART1_CFG,
  45. #endif
  46. #ifdef BSP_USING_UART2
  47. UART2_CFG,
  48. #endif
  49. #ifdef BSP_USING_UART3
  50. UART3_CFG,
  51. #endif
  52. };
  53. static struct swm_uart uart_drv[sizeof(uart_cfg) / sizeof(uart_cfg[0])] = {0};
  54. static rt_err_t swm_uart_init(struct rt_serial_device *serial_device, struct serial_configure *configure)
  55. {
  56. struct swm_uart_cfg *cfg;
  57. RT_ASSERT(serial_device != RT_NULL);
  58. RT_ASSERT(configure != RT_NULL);
  59. cfg = serial_device->parent.user_data;
  60. cfg->uart_initstruct.Baudrate = configure->baud_rate;
  61. switch (configure->data_bits)
  62. {
  63. case DATA_BITS_8:
  64. cfg->uart_initstruct.DataBits = UART_DATA_8BIT;
  65. break;
  66. case DATA_BITS_9:
  67. cfg->uart_initstruct.DataBits = UART_DATA_9BIT;
  68. break;
  69. default:
  70. cfg->uart_initstruct.DataBits = UART_DATA_8BIT;
  71. break;
  72. }
  73. switch (configure->stop_bits)
  74. {
  75. case STOP_BITS_1:
  76. cfg->uart_initstruct.StopBits = UART_STOP_1BIT;
  77. break;
  78. case STOP_BITS_2:
  79. cfg->uart_initstruct.StopBits = UART_STOP_2BIT;
  80. break;
  81. default:
  82. cfg->uart_initstruct.StopBits = UART_STOP_1BIT;
  83. break;
  84. }
  85. switch (configure->parity)
  86. {
  87. case PARITY_NONE:
  88. cfg->uart_initstruct.Parity = UART_PARITY_NONE;
  89. break;
  90. case PARITY_ODD:
  91. cfg->uart_initstruct.Parity = UART_PARITY_ODD;
  92. break;
  93. case PARITY_EVEN:
  94. cfg->uart_initstruct.Parity = UART_PARITY_EVEN;
  95. break;
  96. default:
  97. cfg->uart_initstruct.Parity = UART_PARITY_NONE;
  98. break;
  99. }
  100. switch ((uint32_t)cfg->UARTx)
  101. {
  102. case ((uint32_t)UART0):
  103. PORT_Init(PORTA, PIN2, FUNMUX0_UART0_RXD, 1);
  104. PORT_Init(PORTA, PIN3, FUNMUX1_UART0_TXD, 0);
  105. break;
  106. case ((uint32_t)UART1):
  107. PORT_Init(PORTC, PIN2, FUNMUX0_UART1_RXD, 1);
  108. PORT_Init(PORTC, PIN3, FUNMUX1_UART1_TXD, 0);
  109. break;
  110. case ((uint32_t)UART2):
  111. PORT_Init(PORTC, PIN4, FUNMUX0_UART2_RXD, 1);
  112. PORT_Init(PORTC, PIN5, FUNMUX1_UART2_TXD, 0);
  113. break;
  114. case ((uint32_t)UART3):
  115. PORT_Init(PORTC, PIN6, FUNMUX0_UART3_RXD, 1);
  116. PORT_Init(PORTC, PIN7, FUNMUX1_UART3_TXD, 0);
  117. break;
  118. default:
  119. break;
  120. }
  121. UART_Init(cfg->UARTx, &(cfg->uart_initstruct));
  122. UART_Open(cfg->UARTx);
  123. return RT_EOK;
  124. }
  125. static rt_err_t swm_uart_control(struct rt_serial_device *serial_device, int cmd, void *arg)
  126. {
  127. struct swm_uart_cfg *cfg;
  128. RT_ASSERT(serial_device != RT_NULL);
  129. cfg = serial_device->parent.user_data;
  130. switch (cmd)
  131. {
  132. case RT_DEVICE_CTRL_CLR_INT:
  133. /* disable rx irq */
  134. NVIC_DisableIRQ(cfg->irq);
  135. break;
  136. case RT_DEVICE_CTRL_SET_INT:
  137. /* enable rx irq */
  138. NVIC_EnableIRQ(cfg->irq);
  139. break;
  140. }
  141. return RT_EOK;
  142. }
  143. static int swm_uart_putc(struct rt_serial_device *serial_device, char c)
  144. {
  145. struct swm_uart_cfg *cfg;
  146. RT_ASSERT(serial_device != RT_NULL);
  147. cfg = serial_device->parent.user_data;
  148. while (UART_IsTXFIFOFull(cfg->UARTx))
  149. ;
  150. UART_WriteByte(cfg->UARTx, c);
  151. while (UART_IsTXBusy(cfg->UARTx))
  152. ;
  153. return 1;
  154. }
  155. static int swm_uart_getc(struct rt_serial_device *serial_device)
  156. {
  157. int ch;
  158. struct swm_uart_cfg *cfg;
  159. RT_ASSERT(serial_device != RT_NULL);
  160. cfg = serial_device->parent.user_data;
  161. ch = -1;
  162. if (UART_IsRXFIFOEmpty(cfg->UARTx) == 0)
  163. {
  164. UART_ReadByte(cfg->UARTx, (uint32_t *)&ch);
  165. }
  166. return ch;
  167. }
  168. static const struct rt_uart_ops swm_uart_ops =
  169. {
  170. .configure = swm_uart_init,
  171. .control = swm_uart_control,
  172. .putc = swm_uart_putc,
  173. .getc = swm_uart_getc,
  174. .dma_transmit = RT_NULL};
  175. /**
  176. * Uart common interrupt process. This need add to uart ISR.
  177. *
  178. * @param serial serial device
  179. */
  180. static void rt_hw_uart_isr(struct rt_serial_device *serial_device)
  181. {
  182. struct swm_uart_cfg *cfg;
  183. RT_ASSERT(serial_device != RT_NULL);
  184. cfg = serial_device->parent.user_data;
  185. /* UART in mode Receiver -------------------------------------------------*/
  186. if (UART_INTRXThresholdStat(cfg->UARTx) || UART_INTTimeoutStat(cfg->UARTx))
  187. {
  188. rt_hw_serial_isr(serial_device, RT_SERIAL_EVENT_RX_IND);
  189. }
  190. }
  191. #if defined(BSP_USING_UART0)
  192. void UART0_Handler(void)
  193. {
  194. /* enter interrupt */
  195. rt_interrupt_enter();
  196. rt_hw_uart_isr(&(uart_drv[UART0_INDEX].serial_device));
  197. /* leave interrupt */
  198. rt_interrupt_leave();
  199. }
  200. #endif /* BSP_USING_UART0 */
  201. #if defined(BSP_USING_UART1)
  202. void UART1_Handler(void)
  203. {
  204. /* enter interrupt */
  205. rt_interrupt_enter();
  206. rt_hw_uart_isr(&(uart_drv[UART1_INDEX].serial_device));
  207. /* leave interrupt */
  208. rt_interrupt_leave();
  209. }
  210. #endif /* BSP_USING_UART1 */
  211. #if defined(BSP_USING_UART2)
  212. void UART2_Handler(void)
  213. {
  214. /* enter interrupt */
  215. rt_interrupt_enter();
  216. rt_hw_uart_isr(&(uart_drv[UART2_INDEX].serial_device));
  217. /* leave interrupt */
  218. rt_interrupt_leave();
  219. }
  220. #endif /* BSP_USING_UART2 */
  221. #if defined(BSP_USING_UART3)
  222. void UART3_Handler(void)
  223. {
  224. /* enter interrupt */
  225. rt_interrupt_enter();
  226. rt_hw_uart_isr(&(uart_drv[UART3_INDEX].serial_device));
  227. /* leave interrupt */
  228. rt_interrupt_leave();
  229. }
  230. #endif /* BSP_USING_UART3 */
  231. int rt_hw_uart_init(void)
  232. {
  233. struct serial_configure cfg = RT_SERIAL_CONFIG_DEFAULT;
  234. int i = 0;
  235. rt_err_t result = RT_EOK;
  236. for (i = 0; i < sizeof(uart_cfg) / sizeof(uart_cfg[0]); i++)
  237. {
  238. uart_drv[i].cfg = &uart_cfg[i];
  239. uart_drv[i].serial_device.ops = &swm_uart_ops;
  240. uart_drv[i].serial_device.config = cfg;
  241. /* register UART device */
  242. result = rt_hw_serial_register(&uart_drv[i].serial_device, uart_drv[i].cfg->name,
  243. RT_DEVICE_FLAG_RDWR | RT_DEVICE_FLAG_INT_RX, uart_drv[i].cfg);
  244. RT_ASSERT(result == RT_EOK);
  245. }
  246. return result;
  247. }
  248. INIT_BOARD_EXPORT(rt_hw_uart_init);
  249. #endif /* BSP_USING_UART */
  250. #endif /* RT_USING_SERIAL */