usbd_serial.c 7.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279
  1. /*
  2. * Copyright (c) 2025, sakumisu
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. #include <rtthread.h>
  7. #include <rtdevice.h>
  8. #include "usbd_core.h"
  9. #include "usbd_cdc_acm.h"
  10. #define DEV_FORMAT_CDC_ACM "usb-acm%d"
  11. #ifndef CONFIG_USBDEV_MAX_CDC_ACM_CLASS
  12. #define CONFIG_USBDEV_MAX_CDC_ACM_CLASS (4)
  13. #endif
  14. #ifndef CONFIG_USBDEV_SERIAL_RX_BUFSIZE
  15. #define CONFIG_USBDEV_SERIAL_RX_BUFSIZE (2048)
  16. #endif
  17. struct usbd_serial {
  18. struct rt_device parent;
  19. uint8_t busid;
  20. uint8_t in_ep;
  21. uint8_t out_ep;
  22. struct usbd_interface intf_ctrl;
  23. struct usbd_interface intf_data;
  24. usb_osal_sem_t tx_done;
  25. uint8_t minor;
  26. char name[32];
  27. struct rt_ringbuffer rx_rb;
  28. rt_uint8_t rx_rb_buffer[CONFIG_USBDEV_SERIAL_RX_BUFSIZE];
  29. };
  30. static uint32_t g_devinuse = 0;
  31. static USB_NOCACHE_RAM_SECTION USB_MEM_ALIGNX uint8_t g_usbd_serial_cdc_acm_rx_buf[CONFIG_USBDEV_MAX_CDC_ACM_CLASS][USB_ALIGN_UP(512, CONFIG_USB_ALIGN_SIZE)];
  32. static struct usbd_serial g_usbd_serial_cdc_acm[CONFIG_USBDEV_MAX_CDC_ACM_CLASS];
  33. static struct usbd_serial *usbd_serial_alloc(void)
  34. {
  35. uint8_t devno;
  36. struct usbd_serial *serial;
  37. for (devno = 0; devno < CONFIG_USBDEV_MAX_CDC_ACM_CLASS; devno++) {
  38. if ((g_devinuse & (1U << devno)) == 0) {
  39. g_devinuse |= (1U << devno);
  40. serial = &g_usbd_serial_cdc_acm[devno];
  41. memset(serial, 0, sizeof(struct usbd_serial));
  42. serial->minor = devno;
  43. snprintf(serial->name, CONFIG_USBHOST_DEV_NAMELEN, DEV_FORMAT_CDC_ACM, serial->minor);
  44. return serial;
  45. }
  46. }
  47. return NULL;
  48. }
  49. static void usbd_serial_free(struct usbd_serial *serial)
  50. {
  51. uint8_t devno = serial->minor;
  52. if (devno < 32) {
  53. g_devinuse &= ~(1U << devno);
  54. }
  55. memset(serial, 0, sizeof(struct usbd_serial));
  56. }
  57. static rt_err_t usbd_serial_open(struct rt_device *dev, rt_uint16_t oflag)
  58. {
  59. struct usbd_serial *serial;
  60. RT_ASSERT(dev != RT_NULL);
  61. serial = (struct usbd_serial *)dev;
  62. if (!usb_device_is_configured(serial->busid)) {
  63. USB_LOG_ERR("USB device is not configured\n");
  64. return -RT_EPERM;
  65. }
  66. usbd_ep_start_read(serial->busid, serial->out_ep,
  67. g_usbd_serial_cdc_acm_rx_buf[serial->minor],
  68. usbd_get_ep_mps(serial->busid, serial->out_ep));
  69. return RT_EOK;
  70. }
  71. static rt_ssize_t usbd_serial_read(struct rt_device *dev,
  72. rt_off_t pos,
  73. void *buffer,
  74. rt_size_t size)
  75. {
  76. struct usbd_serial *serial;
  77. RT_ASSERT(dev != RT_NULL);
  78. serial = (struct usbd_serial *)dev;
  79. if (!usb_device_is_configured(serial->busid)) {
  80. return -RT_EPERM;
  81. }
  82. return rt_ringbuffer_get(&serial->rx_rb, (rt_uint8_t *)buffer, size);
  83. }
  84. static rt_ssize_t usbd_serial_write(struct rt_device *dev,
  85. rt_off_t pos,
  86. const void *buffer,
  87. rt_size_t size)
  88. {
  89. struct usbd_serial *serial;
  90. int ret = 0;
  91. rt_uint8_t *align_buf;
  92. RT_ASSERT(dev != RT_NULL);
  93. serial = (struct usbd_serial *)dev;
  94. if (!usb_device_is_configured(serial->busid)) {
  95. return -RT_EPERM;
  96. }
  97. align_buf = (rt_uint8_t *)buffer;
  98. #ifdef CONFIG_USB_DCACHE_ENABLE
  99. if ((uint32_t)buffer & (CONFIG_USB_ALIGN_SIZE - 1)) {
  100. align_buf = rt_malloc_align(size, CONFIG_USB_ALIGN_SIZE);
  101. if (!align_buf) {
  102. USB_LOG_ERR("serial get align buf failed\n");
  103. return 0;
  104. }
  105. usb_memcpy(align_buf, buffer, size);
  106. }
  107. #endif
  108. usb_osal_sem_reset(serial->tx_done);
  109. usbd_ep_start_write(serial->busid, serial->in_ep, align_buf, size);
  110. ret = usb_osal_sem_take(serial->tx_done, 3000);
  111. if (ret < 0) {
  112. USB_LOG_ERR("serial write timeout\n");
  113. ret = -RT_ETIMEOUT;
  114. } else {
  115. ret = size;
  116. }
  117. #ifdef CONFIG_USB_DCACHE_ENABLE
  118. if ((uint32_t)buffer & (CONFIG_USB_ALIGN_SIZE - 1)) {
  119. rt_free_align(align_buf);
  120. }
  121. #endif
  122. return ret;
  123. }
  124. #ifdef RT_USING_DEVICE_OPS
  125. const static struct rt_device_ops usbd_serial_ops = {
  126. NULL,
  127. usbd_serial_open,
  128. NULL,
  129. usbd_serial_read,
  130. usbd_serial_write,
  131. NULL
  132. };
  133. #endif
  134. rt_err_t usbd_serial_register(struct usbd_serial *serial,
  135. void *data)
  136. {
  137. rt_err_t ret;
  138. struct rt_device *device;
  139. RT_ASSERT(serial != RT_NULL);
  140. device = &(serial->parent);
  141. device->type = RT_Device_Class_Char;
  142. device->rx_indicate = RT_NULL;
  143. device->tx_complete = RT_NULL;
  144. #ifdef RT_USING_DEVICE_OPS
  145. device->ops = &usbd_serial_ops;
  146. #else
  147. device->init = NULL;
  148. device->open = usbd_serial_open;
  149. device->close = NULL;
  150. device->read = usbd_serial_read;
  151. device->write = usbd_serial_write;
  152. device->control = NULL;
  153. #endif
  154. device->user_data = data;
  155. /* register a character device */
  156. ret = rt_device_register(device, serial->name, RT_DEVICE_FLAG_RDWR | RT_DEVICE_FLAG_INT_RX | RT_DEVICE_FLAG_REMOVABLE);
  157. #ifdef RT_USING_POSIX_DEVIO
  158. /* set fops */
  159. device->fops = NULL;
  160. #endif
  161. rt_ringbuffer_init(&serial->rx_rb, serial->rx_rb_buffer, sizeof(serial->rx_rb_buffer));
  162. return ret;
  163. }
  164. void usbd_cdc_acm_bulk_out(uint8_t busid, uint8_t ep, uint32_t nbytes)
  165. {
  166. struct usbd_serial *serial;
  167. for (uint8_t devno = 0; devno < CONFIG_USBDEV_MAX_CDC_ACM_CLASS; devno++) {
  168. serial = &g_usbd_serial_cdc_acm[devno];
  169. if (serial->out_ep == ep) {
  170. rt_ringbuffer_put(&serial->rx_rb, g_usbd_serial_cdc_acm_rx_buf[serial->minor], nbytes);
  171. usbd_ep_start_read(serial->busid, serial->out_ep,
  172. g_usbd_serial_cdc_acm_rx_buf[serial->minor],
  173. usbd_get_ep_mps(serial->busid, serial->out_ep));
  174. if (serial->parent.rx_indicate) {
  175. serial->parent.rx_indicate(&serial->parent, nbytes);
  176. }
  177. break;
  178. }
  179. }
  180. }
  181. void usbd_cdc_acm_bulk_in(uint8_t busid, uint8_t ep, uint32_t nbytes)
  182. {
  183. struct usbd_serial *serial;
  184. if ((nbytes % usbd_get_ep_mps(busid, ep)) == 0 && nbytes) {
  185. /* send zlp */
  186. usbd_ep_start_write(busid, ep, NULL, 0);
  187. } else {
  188. for (uint8_t devno = 0; devno < CONFIG_USBDEV_MAX_CDC_ACM_CLASS; devno++) {
  189. serial = &g_usbd_serial_cdc_acm[devno];
  190. if ((serial->in_ep == ep) && serial->tx_done) {
  191. usb_osal_sem_give(serial->tx_done);
  192. break;
  193. }
  194. }
  195. }
  196. }
  197. void usbd_cdc_acm_serial_init(uint8_t busid, uint8_t in_ep, uint8_t out_ep)
  198. {
  199. struct usbd_serial *serial;
  200. struct usbd_endpoint cdc_out_ep = {
  201. .ep_addr = out_ep,
  202. .ep_cb = usbd_cdc_acm_bulk_out
  203. };
  204. struct usbd_endpoint cdc_in_ep = {
  205. .ep_addr = in_ep,
  206. .ep_cb = usbd_cdc_acm_bulk_in
  207. };
  208. serial = usbd_serial_alloc();
  209. if (serial == NULL) {
  210. USB_LOG_ERR("No more serial device available\n");
  211. return;
  212. }
  213. serial->busid = busid;
  214. serial->in_ep = in_ep;
  215. serial->out_ep = out_ep;
  216. serial->tx_done = usb_osal_sem_create(0);
  217. usbd_add_interface(busid, usbd_cdc_acm_init_intf(busid, &serial->intf_ctrl));
  218. usbd_add_interface(busid, usbd_cdc_acm_init_intf(busid, &serial->intf_data));
  219. usbd_add_endpoint(busid, &cdc_out_ep);
  220. usbd_add_endpoint(busid, &cdc_in_ep);
  221. if (usbd_serial_register(serial, NULL) != RT_EOK) {
  222. USB_LOG_ERR("Failed to register serial device\n");
  223. usbd_serial_free(serial);
  224. return;
  225. }
  226. USB_LOG_INFO("USB CDC ACM Serial Device %s initialized\n", serial->name);
  227. }