uart_usb.c 6.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279
  1. /** @file
  2. * @brief USB UART driver
  3. *
  4. * A UART driver which use USB CDC ACM protocol implementation.
  5. */
  6. /*
  7. * Copyright (c) 2021 Actions Semiconductor Co., Ltd
  8. *
  9. * SPDX-License-Identifier: Apache-2.0
  10. */
  11. #include <kernel.h>
  12. #include <device.h>
  13. #include <string.h>
  14. #include <drivers/uart.h>
  15. #include <drivers/console/uart_usb.h>
  16. #include <drivers/usb/usb_phy.h>
  17. #include <usb/class/usb_cdc.h>
  18. #include <sys/ring_buffer.h>
  19. #ifdef CONFIG_NVRAM_CONFIG
  20. #include <drivers/nvram_config.h>
  21. #endif
  22. #define CFG_USB_UART_MODE "USB_UART_MODE"
  23. #define UART_USB_SEND_TIMEOUT_US (1000)
  24. #define UART_USB_PRINTK_BUFFER (128)
  25. extern void __stdout_hook_install(int (*hook)(int));
  26. extern void *__stdout_get_hook(void);
  27. extern void __printk_hook_install(int (*fn)(int));
  28. extern void *__printk_get_hook(void);
  29. static uint8_t __act_s2_notsave uart_usb_buffer[UART_USB_PRINTK_BUFFER];
  30. typedef struct {
  31. const struct device *usb_dev;
  32. int (*stdout_bak)(int);
  33. int (*printk_bak)(int);
  34. uint8_t *buffer; /* buffer for printk */
  35. uint16_t cursor; /* buffer cursor */
  36. uint8_t state : 1; /* 1: initialized 0: uninitialized */
  37. uint8_t tx_done : 1; /* 1: tx done 0: tx not done */
  38. uint8_t enable : 1; /* enable flag in nvram */
  39. } uart_usb_dev_t, *p_uart_dev_t;
  40. /* Get the USB UART device */
  41. static p_uart_dev_t uart_usb_get_dev(void)
  42. {
  43. static uart_usb_dev_t uart_usb_dev = {0};
  44. return &uart_usb_dev;
  45. }
  46. /* Updata the completion for USB UART TX interface */
  47. void uart_usb_update_tx_done(void)
  48. {
  49. p_uart_dev_t p_uart_dev = uart_usb_get_dev();
  50. p_uart_dev->tx_done = 1;
  51. }
  52. /* Send out data by USB UART interface */
  53. int uart_usb_send(const uint8_t *data, int len)
  54. {
  55. p_uart_dev_t p_uart_dev = uart_usb_get_dev();
  56. uint32_t start, duration;
  57. int ret;
  58. if (!p_uart_dev->state)
  59. return -ESRCH;
  60. uart_irq_tx_enable(p_uart_dev->usb_dev);
  61. p_uart_dev->tx_done = 0;
  62. ret = uart_fifo_fill(p_uart_dev->usb_dev, data, len);
  63. if (ret < 0)
  64. return ret;
  65. /* wait tx done */
  66. start = k_cycle_get_32();
  67. while (!p_uart_dev->tx_done) {
  68. duration = k_cycle_get_32() - start;
  69. if (SYS_CLOCK_HW_CYCLES_TO_NS_AVG(duration, 1000)
  70. > UART_USB_SEND_TIMEOUT_US) {
  71. uart_irq_tx_disable(p_uart_dev->usb_dev);
  72. return -ETIMEDOUT;
  73. }
  74. }
  75. uart_irq_tx_disable(p_uart_dev->usb_dev);
  76. return 0;
  77. }
  78. /* recevice data from USB UART interface */
  79. int uart_usb_receive(uint8_t *data, int max_len)
  80. {
  81. p_uart_dev_t p_uart_dev = uart_usb_get_dev();
  82. int rlen;
  83. memset(data, 0, max_len);
  84. rlen = uart_fifo_read(p_uart_dev->usb_dev, data, max_len);
  85. if (rlen < 0) {
  86. uart_irq_rx_disable(p_uart_dev->usb_dev);
  87. return -EIO;
  88. }
  89. /* exceptional that data received too much and drain all data */
  90. if (rlen >= max_len) {
  91. while (uart_fifo_read(p_uart_dev->usb_dev, data, max_len) > 0);
  92. rlen = 0;
  93. }
  94. return rlen;
  95. }
  96. /* Check whether USB UART is enabled or not */
  97. bool uart_usb_is_enabled(void)
  98. {
  99. p_uart_dev_t p_uart_dev = uart_usb_get_dev();
  100. return p_uart_dev->enable;
  101. }
  102. /* USB UART early init that controlled by nvram infomation */
  103. static int uart_usb_early_init(const struct device *dev)
  104. {
  105. char temp[16];
  106. int ret = 0;
  107. p_uart_dev_t p_uart_dev = uart_usb_get_dev();
  108. ARG_UNUSED(dev);
  109. memset(temp, 0, sizeof(temp));
  110. #ifdef CONFIG_NVRAM_CONFIG
  111. ret = nvram_config_get(CFG_USB_UART_MODE, temp, 16);
  112. #endif
  113. if (ret > 0) {
  114. if (!strcmp(temp, "true"))
  115. p_uart_dev->enable = 1;
  116. } else {
  117. /* by default to enable USB UART function */
  118. p_uart_dev->enable = 1;
  119. }
  120. printk("usb uart mode:%d\n", p_uart_dev->enable);
  121. return 0;
  122. }
  123. /* flash all printk buffer out */
  124. static void uart_usb_flush_out(void)
  125. {
  126. p_uart_dev_t p_uart_dev = uart_usb_get_dev();
  127. uart_usb_send(p_uart_dev->buffer, p_uart_dev->cursor);
  128. p_uart_dev->cursor = 0;
  129. }
  130. /* put the character into printk buffer or flush out data */
  131. static int uart_usb_char_out(int c)
  132. {
  133. p_uart_dev_t p_uart_dev = uart_usb_get_dev();
  134. /* If the cursor of buffer is up to the max boundary, flush out all data.
  135. * Note that buffer reserves 2 characters for '\n\r'.
  136. */
  137. if (p_uart_dev->cursor >= (UART_USB_PRINTK_BUFFER - 3)) {
  138. if ('\n' == c) {
  139. p_uart_dev->buffer[UART_USB_PRINTK_BUFFER - 3] = c;
  140. p_uart_dev->buffer[UART_USB_PRINTK_BUFFER - 2] = '\r';
  141. p_uart_dev->cursor = UART_USB_PRINTK_BUFFER - 1;
  142. } else {
  143. p_uart_dev->buffer[UART_USB_PRINTK_BUFFER - 3] = c;
  144. p_uart_dev->buffer[UART_USB_PRINTK_BUFFER - 2] = '\n';
  145. p_uart_dev->buffer[UART_USB_PRINTK_BUFFER - 1] = '\r';
  146. p_uart_dev->cursor = UART_USB_PRINTK_BUFFER;
  147. }
  148. uart_usb_flush_out();
  149. }
  150. /* check the endline indicator('\n') and flush line data */
  151. if ('\n' == c) {
  152. p_uart_dev->buffer[p_uart_dev->cursor++] = c;
  153. p_uart_dev->buffer[p_uart_dev->cursor++] = '\r';
  154. uart_usb_flush_out();
  155. } else {
  156. p_uart_dev->buffer[p_uart_dev->cursor++] = c;
  157. }
  158. return 0;
  159. }
  160. /* USB UART buffer init */
  161. static void uart_usb_buf_init(void)
  162. {
  163. p_uart_dev_t p_uart_dev = uart_usb_get_dev();
  164. p_uart_dev->buffer = uart_usb_buffer;
  165. p_uart_dev->cursor = 0;
  166. }
  167. /* USB UART initialization */
  168. int uart_usb_init(void)
  169. {
  170. p_uart_dev_t p_uart_dev = uart_usb_get_dev();
  171. /* check if has been initialized before */
  172. if (!p_uart_dev->enable)
  173. return -EPERM;
  174. if (!p_uart_dev->state) {
  175. p_uart_dev->usb_dev = device_get_binding(CONFIG_CDC_ACM_PORT_NAME);
  176. if (!p_uart_dev->usb_dev) {
  177. printk("failed to bind usb dev:%s\n", CONFIG_CDC_ACM_PORT_NAME);
  178. return -ENODEV;
  179. }
  180. /* USB CDC ACM class low level init */
  181. if (usb_cdc_acm_init((struct device *)p_uart_dev->usb_dev)) {
  182. printk("failed to init CDC ACM device\n");
  183. return -EFAULT;
  184. }
  185. /* buffer initialzation */
  186. uart_usb_buf_init();
  187. p_uart_dev->state = 1;
  188. p_uart_dev->tx_done = 0;
  189. /* backup stdout/printk hook for uninit stage */
  190. p_uart_dev->printk_bak = __printk_get_hook();
  191. p_uart_dev->stdout_bak = __stdout_get_hook();
  192. __stdout_hook_install(uart_usb_char_out);
  193. __printk_hook_install(uart_usb_char_out);
  194. printk("uart usb init ok\n");
  195. }
  196. return 0;
  197. }
  198. /* Un-init USB UART resources */
  199. void uart_usb_uninit(void)
  200. {
  201. p_uart_dev_t p_uart_dev = uart_usb_get_dev();
  202. if (p_uart_dev->state) {
  203. usb_cdc_acm_exit();
  204. p_uart_dev->state = 0;
  205. p_uart_dev->tx_done = 0;
  206. /* restore the origin stdout hook */
  207. __stdout_hook_install(p_uart_dev->stdout_bak);
  208. __printk_hook_install(p_uart_dev->printk_bak);
  209. printk("uart usb uninit ok\n");
  210. }
  211. }
  212. SYS_INIT(uart_usb_early_init, POST_KERNEL, CONFIG_KERNEL_INIT_PRIORITY_OBJECTS);
  213. #if !defined(CONFIG_ACTIONS_PRINTK_DMA) && !defined(CONFIG_USB_HOTPLUG)
  214. static int uart_usb_later_init(const struct device *dev)
  215. {
  216. int ret;
  217. ARG_UNUSED(dev);
  218. /* usb device resource initialzation */
  219. usb_phy_enter_b_idle();
  220. usb_phy_init();
  221. ret = uart_usb_init();
  222. if (ret)
  223. return ret;
  224. return 0;
  225. }
  226. SYS_INIT(uart_usb_later_init, APPLICATION, CONFIG_KERNEL_INIT_PRIORITY_OBJECTS);
  227. #endif