tracing_backend_usb.c 3.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173
  1. /*
  2. * Copyright (c) 2019 Intel corporation
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. #include <sys/util.h>
  7. #include <sys/atomic.h>
  8. #include <sys/__assert.h>
  9. #include <sys/byteorder.h>
  10. #include <usb/usb_device.h>
  11. #include <tracing_core.h>
  12. #include <tracing_buffer.h>
  13. #include <tracing_backend.h>
  14. #define USB_TRANSFER_ONGOING 1
  15. #define USB_TRANSFER_FREE 0
  16. #define TRACING_IF_IN_EP_ADDR 0x81
  17. #define TRACING_IF_OUT_EP_ADDR 0x01
  18. struct usb_device_desc {
  19. struct usb_if_descriptor if0;
  20. struct usb_ep_descriptor if0_in_ep;
  21. struct usb_ep_descriptor if0_out_ep;
  22. } __packed;
  23. static volatile int transfer_state;
  24. static enum usb_dc_status_code usb_device_status = USB_DC_UNKNOWN;
  25. USBD_CLASS_DESCR_DEFINE(primary, 0) struct usb_device_desc dev_desc = {
  26. /*
  27. * Interface descriptor 0
  28. */
  29. .if0 = {
  30. .bLength = sizeof(struct usb_if_descriptor),
  31. .bDescriptorType = USB_DESC_INTERFACE,
  32. .bInterfaceNumber = 0,
  33. .bAlternateSetting = 0,
  34. .bNumEndpoints = 2,
  35. .bInterfaceClass = USB_BCC_VENDOR,
  36. .bInterfaceSubClass = 0,
  37. .bInterfaceProtocol = 0,
  38. .iInterface = 0,
  39. },
  40. /*
  41. * Data Endpoint IN
  42. */
  43. .if0_in_ep = {
  44. .bLength = sizeof(struct usb_ep_descriptor),
  45. .bDescriptorType = USB_DESC_ENDPOINT,
  46. .bEndpointAddress = TRACING_IF_IN_EP_ADDR,
  47. .bmAttributes = USB_DC_EP_BULK,
  48. .wMaxPacketSize = sys_cpu_to_le16(CONFIG_TRACING_USB_MPS),
  49. .bInterval = 0x00,
  50. },
  51. /*
  52. * Data Endpoint OUT
  53. */
  54. .if0_out_ep = {
  55. .bLength = sizeof(struct usb_ep_descriptor),
  56. .bDescriptorType = USB_DESC_ENDPOINT,
  57. .bEndpointAddress = TRACING_IF_OUT_EP_ADDR,
  58. .bmAttributes = USB_DC_EP_BULK,
  59. .wMaxPacketSize = sys_cpu_to_le16(CONFIG_TRACING_USB_MPS),
  60. .bInterval = 0x00,
  61. },
  62. };
  63. static void dev_status_cb(struct usb_cfg_data *cfg,
  64. enum usb_dc_status_code status,
  65. const uint8_t *param)
  66. {
  67. ARG_UNUSED(cfg);
  68. ARG_UNUSED(param);
  69. usb_device_status = status;
  70. }
  71. static void tracing_ep_out_cb(uint8_t ep, enum usb_dc_ep_cb_status_code ep_status)
  72. {
  73. uint8_t *cmd = NULL;
  74. uint32_t bytes_to_read, length;
  75. usb_read(ep, NULL, 0, &bytes_to_read);
  76. while (bytes_to_read) {
  77. length = tracing_cmd_buffer_alloc(&cmd);
  78. if (cmd) {
  79. length = MIN(length, bytes_to_read);
  80. usb_read(ep, cmd, length, NULL);
  81. tracing_cmd_handle(cmd, length);
  82. bytes_to_read -= length;
  83. }
  84. }
  85. /*
  86. * send ZLP to sync with host receive thread
  87. */
  88. usb_write(TRACING_IF_IN_EP_ADDR, NULL, 0, NULL);
  89. }
  90. static void tracing_ep_in_cb(uint8_t ep, enum usb_dc_ep_cb_status_code ep_status)
  91. {
  92. ARG_UNUSED(ep);
  93. ARG_UNUSED(ep_status);
  94. transfer_state = USB_TRANSFER_FREE;
  95. }
  96. static struct usb_ep_cfg_data ep_cfg[] = {
  97. {
  98. .ep_cb = tracing_ep_out_cb,
  99. .ep_addr = TRACING_IF_OUT_EP_ADDR,
  100. },
  101. {
  102. .ep_cb = tracing_ep_in_cb,
  103. .ep_addr = TRACING_IF_IN_EP_ADDR,
  104. },
  105. };
  106. USBD_CFG_DATA_DEFINE(primary, tracing_backend_usb)
  107. struct usb_cfg_data tracing_backend_usb_config = {
  108. .usb_device_description = NULL,
  109. .interface_descriptor = &dev_desc.if0,
  110. .cb_usb_status = dev_status_cb,
  111. .interface = {
  112. .class_handler = NULL,
  113. .custom_handler = NULL,
  114. .vendor_handler = NULL,
  115. },
  116. .num_endpoints = ARRAY_SIZE(ep_cfg),
  117. .endpoint = ep_cfg,
  118. };
  119. static void tracing_backend_usb_output(const struct tracing_backend *backend,
  120. uint8_t *data, uint32_t length)
  121. {
  122. int ret = 0;
  123. uint32_t bytes;
  124. while (length > 0) {
  125. transfer_state = USB_TRANSFER_ONGOING;
  126. /*
  127. * make sure every USB tansfer no need ZLP at all
  128. * because we are in lowest priority thread content
  129. * there are no deterministic time between real USB
  130. * packet and ZLP
  131. */
  132. ret = usb_write(TRACING_IF_IN_EP_ADDR, data,
  133. length >= CONFIG_TRACING_USB_MPS ?
  134. CONFIG_TRACING_USB_MPS - 1 : length, &bytes);
  135. if (ret) {
  136. continue;
  137. }
  138. data += bytes;
  139. length -= bytes;
  140. while (transfer_state == USB_TRANSFER_ONGOING) {
  141. }
  142. }
  143. }
  144. const struct tracing_backend_api tracing_backend_usb_api = {
  145. .output = tracing_backend_usb_output
  146. };
  147. TRACING_BACKEND_DEFINE(tracing_backend_usb, tracing_backend_usb_api);