api_device.rst 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476
  1. 设备协议栈
  2. =========================
  3. 设备协议栈主要负责枚举和驱动加载,枚举这边就不说了,驱动加载,也就是接口驱动加载,主要是依靠 `usbd_add_interface` 函数,记录传入的接口驱动保存到链表中,当主机进行类请求时就可以查找链表进行访问了。
  4. 在调用 `usbd_desc_register` 以后需要进行接口注册和端点注册,口诀如下:
  5. - 有多少个接口就调用多少次 `usbd_add_interface`,参数填相关 `xxx_init_intf`, 如果没有支持的,手动创建一个填入
  6. - 有多少个端点就调用多少次 `usbd_add_endpoint`,当中断完成时,会调用到注册的端点回调中。
  7. CORE
  8. -----------------
  9. 端点结构体
  10. """"""""""""""""""""""""""""""""""""
  11. 端点结构体主要用于注册不同端点地址的中断完成回调函数。
  12. .. code-block:: C
  13. struct usbd_endpoint {
  14. uint8_t ep_addr;
  15. usbd_endpoint_callback ep_cb;
  16. };
  17. - **list** 端点的链表节点
  18. - **ep_addr** 端点地址(带方向)
  19. - **ep_cb** 端点完成中断回调函数。
  20. .. note:: 总结一句话:in 回调函数等价于 dma 发送完成中断回调函数;out 回调函数等价于 dma 接收完成中断回调函数
  21. 接口结构体
  22. """"""""""""""""""""""""""""""""""""
  23. 接口结构体主要用于注册不同类设备除了标准设备请求外的其他请求,包括类设备请求、厂商设备请求和自定义设备请求。以及协议栈中的相关通知回调函数。
  24. .. code-block:: C
  25. struct usbd_interface {
  26. usb_slist_t list;
  27. usbd_request_handler class_interface_handler;
  28. usbd_request_handler class_endpoint_handler;
  29. usbd_request_handler vendor_handler;
  30. usbd_notify_handler notify_handler;
  31. const uint8_t *hid_report_descriptor;
  32. uint32_t hid_report_descriptor_len;
  33. uint8_t intf_num;
  34. };
  35. - **list** 接口的链表节点
  36. - **class_interface_handler** class setup 请求回调函数,接收者为接口
  37. - **class_endpoint_handler** class setup 请求回调函数,接收者为端点
  38. - **vendor_handler** vendor setup 请求回调函数
  39. - **notify_handler** 中断标志、协议栈相关状态回调函数
  40. - **hid_report_descriptor** hid 报告描述符
  41. - **hid_report_descriptor_len** hid 报告描述符长度
  42. - **intf_num** 当前接口偏移
  43. - **ep_list** 端点的链表节点
  44. usbd_desc_register
  45. """"""""""""""""""""""""""""""""""""
  46. ``usbd_desc_register`` 用来注册 USB 描述符,描述符种类包括:设备描述符、配置描述符(包含配置描述符、接口描述符、class 类描述符、端点描述符)、字符串描述符、设备限定描述符。
  47. .. code-block:: C
  48. void usbd_desc_register(const uint8_t *desc);
  49. - **desc** 描述符的句柄
  50. usbd_msosv1_desc_register
  51. """"""""""""""""""""""""""""""""""""
  52. ``usbd_msosv1_desc_register`` 用来注册一个 WINUSB 1.0 描述符。
  53. .. code-block:: C
  54. void usbd_msosv1_desc_register(struct usb_msosv1_descriptor *desc);
  55. - **desc** 描述符句柄
  56. usbd_msosv2_desc_register
  57. """"""""""""""""""""""""""""""""""""
  58. ``usbd_msosv2_desc_register`` 用来注册一个 WINUSB 2.0 描述符。
  59. .. code-block:: C
  60. void usbd_msosv2_desc_register(struct usb_msosv2_descriptor *desc);
  61. - **desc** 描述符句柄
  62. usbd_bos_desc_register
  63. """"""""""""""""""""""""""""""""""""
  64. ``usbd_bos_desc_register`` 用来注册一个 BOS 描述符, USB 2.1 版本以上必须注册。
  65. .. code-block:: C
  66. void usbd_bos_desc_register(struct usb_bos_descriptor *desc);
  67. - **desc** 描述符句柄
  68. usbd_add_interface
  69. """"""""""""""""""""""""""""""""""""
  70. ``usbd_add_interface`` 添加一个接口驱动。 **添加顺序必须按照描述符顺序**。
  71. .. code-block:: C
  72. void usbd_add_interface(struct usbd_interface *intf);
  73. - **intf** 接口驱动句柄,通常从不同 class 的 `xxx_init_intf` 函数获取
  74. usbd_add_endpoint
  75. """"""""""""""""""""""""""""""""""""
  76. ``usbd_add_endpoint`` 添加一个端点中断完成回调函数。
  77. .. code-block:: C
  78. void usbd_add_endpoint(struct usbd_endpoint *ep);;
  79. - **ep** 端点句柄
  80. usbd_initialize
  81. """"""""""""""""""""""""""""""""""""
  82. ``usbd_initialize`` 用来初始化 usb device 寄存器配置、usb 时钟、中断等,需要注意,此函数必须在所有列出的 API 最后。 **如果使用 os,必须放在线程中执行**。
  83. .. code-block:: C
  84. int usbd_initialize(void);
  85. usbd_event_handler
  86. """"""""""""""""""""""""""""""""""""
  87. ``usbd_event_handler`` 是协议栈中中断或者协议栈一些状态的回调函数。大部分 IP 仅支持 USBD_EVENT_RESET 和 USBD_EVENT_CONFIGURED
  88. .. code-block:: C
  89. void usbd_event_handler(uint8_t event);
  90. CDC ACM
  91. -----------------
  92. usbd_cdc_acm_init_intf
  93. """"""""""""""""""""""""""""""""""""
  94. ``usbd_cdc_acm_init_intf`` 用来初始化 USB CDC ACM 类接口,并实现该接口相关的函数。
  95. - ``cdc_acm_class_interface_request_handler`` 用来处理 USB CDC ACM 类 Setup 请求。
  96. - ``cdc_notify_handler`` 用来处理 USB CDC 其他中断回调函数。
  97. .. code-block:: C
  98. struct usbd_interface *usbd_cdc_acm_init_intf(struct usbd_interface *intf);
  99. - **return** 接口句柄
  100. usbd_cdc_acm_set_line_coding
  101. """"""""""""""""""""""""""""""""""""
  102. ``usbd_cdc_acm_set_line_coding`` 用来对串口进行配置,如果仅使用 USB 而不用 串口,该接口不用用户实现,使用默认。
  103. .. code-block:: C
  104. void usbd_cdc_acm_set_line_coding(uint8_t intf, struct cdc_line_coding *line_coding);
  105. - **intf** 控制接口号
  106. - **line_coding** 串口配置
  107. usbd_cdc_acm_get_line_coding
  108. """"""""""""""""""""""""""""""""""""
  109. ``usbd_cdc_acm_get_line_coding`` 用来获取串口进行配置,如果仅使用 USB 而不用 串口,该接口不用用户实现,使用默认。
  110. .. code-block:: C
  111. void usbd_cdc_acm_get_line_coding(uint8_t intf, struct cdc_line_coding *line_coding);
  112. - **intf** 控制接口号
  113. - **line_coding** 串口配置
  114. usbd_cdc_acm_set_dtr
  115. """"""""""""""""""""""""""""""""""""
  116. ``usbd_cdc_acm_set_dtr`` 用来控制串口 DTR 。如果仅使用 USB 而不用 串口,该接口不用用户实现,使用默认。
  117. .. code-block:: C
  118. void usbd_cdc_acm_set_dtr(uint8_t intf, bool dtr);
  119. - **intf** 控制接口号
  120. - **dtr** dtr 为1表示拉低电平,为0表示拉高电平
  121. usbd_cdc_acm_set_rts
  122. """"""""""""""""""""""""""""""""""""
  123. ``usbd_cdc_acm_set_rts`` 用来控制串口 RTS 。如果仅使用 USB 而不用 串口,该接口不用用户实现,使用默认。
  124. .. code-block:: C
  125. void usbd_cdc_acm_set_rts(uint8_t intf, bool rts);
  126. - **intf** 控制接口号
  127. - **rts** rts 为1表示拉低电平,为0表示拉高电平
  128. CDC_ACM_DESCRIPTOR_INIT
  129. """"""""""""""""""""""""""""""""""""
  130. ``CDC_ACM_DESCRIPTOR_INIT`` 配置了默认的 cdc acm 需要的描述符以及参数,方便用户使用。总长度为 `CDC_ACM_DESCRIPTOR_LEN` 。
  131. .. code-block:: C
  132. CDC_ACM_DESCRIPTOR_INIT(bFirstInterface, int_ep, out_ep, in_ep, str_idx);
  133. - **bFirstInterface** 表示该 cdc acm 第一个接口所在所有接口的偏移
  134. - **int_ep** 表示中断端点地址(带方向)
  135. - **out_ep** 表示 bulk out 端点地址(带方向)
  136. - **in_ep** 表示 bulk in 端点地址(带方向)
  137. - **str_idx** 控制接口对应的字符串 id
  138. HID
  139. -----------------
  140. usbd_hid_init_intf
  141. """"""""""""""""""""""""""""""""""""
  142. ``usbd_hid_init_intf`` 用来初始化 USB HID 类接口,并实现该接口相关的函数:
  143. - ``hid_class_interface_request_handler`` 用来处理 USB HID 类的 Setup 请求。
  144. - ``hid_notify_handler`` 用来处理 USB HID 其他中断回调函数。
  145. .. code-block:: C
  146. struct usbd_interface *usbd_hid_init_intf(struct usbd_interface *intf, const uint8_t *desc, uint32_t desc_len);
  147. - **desc** 报告描述符
  148. - **desc_len** 报告描述符长度
  149. MSC
  150. -----------------
  151. usbd_msc_init_intf
  152. """"""""""""""""""""""""""""""""""""
  153. ``usbd_msc_init_intf`` 用来初始化 MSC 类接口,并实现该接口相关函数,并且注册端点回调函数。(因为 msc bot 协议是固定的,所以不需要用于实现,因此端点回调函数自然不需要用户实现)。
  154. - ``msc_storage_class_interface_request_handler`` 用于处理 USB MSC Setup 中断请求。
  155. - ``msc_storage_notify_handler`` 用于实现 USB MSC 其他中断回调函数。
  156. - ``mass_storage_bulk_out`` 用于处理 USB MSC 端点 out 中断。
  157. - ``mass_storage_bulk_in`` 用于处理 USB MSC 端点 in 中断。
  158. .. code-block:: C
  159. struct usbd_interface *usbd_msc_init_intf(struct usbd_interface *intf, const uint8_t out_ep, const uint8_t in_ep);
  160. - **out_ep** out 端点地址
  161. - **in_ep** in 端点地址
  162. usbd_msc_get_cap
  163. """"""""""""""""""""""""""""""""""""
  164. ``usbd_msc_get_cap`` 用来获取存储器的 lun、扇区个数和每个扇区大小。用户必须实现该函数。
  165. .. code-block:: C
  166. void usbd_msc_get_cap(uint8_t lun, uint32_t *block_num, uint16_t *block_size);
  167. - **lun** 存储逻辑单元,暂时无用,默认支持一个
  168. - **block_num** 存储扇区个数
  169. - **block_size** 存储扇区大小
  170. usbd_msc_sector_read
  171. """"""""""""""""""""""""""""""""""""
  172. ``usbd_msc_sector_read`` 用来对存储器某个扇区开始的地址进行数据读取。用户必须实现该函数。
  173. .. code-block:: C
  174. int usbd_msc_sector_read(uint32_t sector, uint8_t *buffer, uint32_t length);
  175. - **sector** 扇区偏移
  176. - **buffer** 存储读取的数据的指针
  177. - **length** 读取长度,当前为1个扇区的大小
  178. usbd_msc_sector_write
  179. """"""""""""""""""""""""""""""""""""
  180. ``usbd_msc_sector_write`` 用来对存储器某个扇区开始写入数据。用户必须实现该函数。
  181. .. code-block:: C
  182. int usbd_msc_sector_write(uint32_t sector, uint8_t *buffer, uint32_t length);
  183. - **sector** 扇区偏移
  184. - **buffer** 写入数据指针
  185. - **length** 写入长度,当前为1个扇区的大小
  186. UAC
  187. -----------------
  188. usbd_audio_init_intf
  189. """"""""""""""""""""""""""""""""""""
  190. ``usbd_audio_init_intf`` 用来初始化 USB Audio 类接口,并实现该接口相关的函数:
  191. - ``audio_class_interface_request_handler`` 用于处理 USB Audio Setup 接口接收者中断请求。
  192. - ``audio_class_endpoint_request_handler`` 用于处理 USB Audio Setup 端点接收者中断请求。
  193. - ``audio_notify_handler`` 用于实现 USB Audio 其他中断回调函数。
  194. .. code-block:: C
  195. struct usbd_interface *usbd_audio_init_intf(struct usbd_interface *intf);
  196. - **class** 类的句柄
  197. - **intf** 接口句柄
  198. usbd_audio_open
  199. """"""""""""""""""""""""""""""""""""
  200. ``usbd_audio_open`` 用来开启音频数据传输。
  201. .. code-block:: C
  202. void usbd_audio_open(uint8_t intf);
  203. - **intf** 开启的接口号
  204. usbd_audio_close
  205. """"""""""""""""""""""""""""""""""""
  206. ``usbd_audio_close`` 用来关闭音频数据传输。
  207. .. code-block:: C
  208. void usbd_audio_close(uint8_t intf);
  209. - **intf** 关闭的接口号
  210. usbd_audio_add_entity
  211. """"""""""""""""""""""""""""""""""""
  212. ``usbd_audio_add_entity`` 用来添加 unit 相关控制,例如 feature unit、clock source。
  213. .. code-block:: C
  214. void usbd_audio_add_entity(uint8_t entity_id, uint16_t bDescriptorSubtype);
  215. - **entity_id** 要添加的 unit id
  216. - **bDescriptorSubtype** entity_id 的描述符子类型
  217. usbd_audio_set_mute
  218. """"""""""""""""""""""""""""""""""""
  219. ``usbd_audio_set_mute`` 用来设置静音。
  220. .. code-block:: C
  221. void usbd_audio_set_mute(uint8_t ch, uint8_t enable);
  222. - **ch** 要设置静音的通道
  223. - **enable** 为1 表示静音,0相反
  224. usbd_audio_set_volume
  225. """"""""""""""""""""""""""""""""""""
  226. ``usbd_audio_set_volume`` 用来设置音量。
  227. .. code-block:: C
  228. void usbd_audio_set_volume(uint8_t ch, float dB);
  229. - **ch** 要设置音量的通道
  230. - **dB** 要设置音量的分贝,其中 UAC1.0范围从 -127 ~ +127dB,UAC2.0 从 0 ~ 256dB
  231. usbd_audio_set_sampling_freq
  232. """"""""""""""""""""""""""""""""""""
  233. ``usbd_audio_set_sampling_freq`` 用来设置设备上音频模块的采样率
  234. .. code-block:: C
  235. void usbd_audio_set_sampling_freq(uint8_t ep_ch, uint32_t sampling_freq);
  236. - **ch** 要设置采样率的端点或者通道,UAC1.0为端点,UAC2.0 为通道
  237. - **dB** 要设置的采样率
  238. usbd_audio_get_sampling_freq_table
  239. """"""""""""""""""""""""""""""""""""
  240. ``usbd_audio_get_sampling_freq_table`` 用来获取支持的采样率列表,如果函数没有实现,则使用默认采样率列表。
  241. .. code-block:: C
  242. void usbd_audio_get_sampling_freq_table(uint8_t **sampling_freq_table);
  243. - **sampling_freq_table** 采样率列表地址,格式参考默认采样率列表
  244. usbd_audio_set_pitch
  245. """"""""""""""""""""""""""""""""""""
  246. ``usbd_audio_set_pitch`` 用来设置音频音调,仅 UAC1.0 有这功能。
  247. .. code-block:: C
  248. void usbd_audio_set_pitch(uint8_t ep, bool enable);
  249. - **ep** 要设置音调的端点
  250. - **enable** 开启或关闭音调
  251. UVC
  252. -----------------
  253. usbd_video_init_intf
  254. """"""""""""""""""""""""""""""""""""
  255. ``usbd_video_init_intf`` 用来初始化 USB Video 类接口,并实现该接口相关的函数:
  256. - ``video_class_interface_request_handler`` 用于处理 USB Video Setup 中断请求。
  257. - ``video_notify_handler`` 用于实现 USB Video 其他中断回调函数。
  258. .. code-block:: C
  259. struct usbd_interface *usbd_video_init_intf(struct usbd_interface *intf,
  260. uint32_t dwFrameInterval,
  261. uint32_t dwMaxVideoFrameSize,
  262. uint32_t dwMaxPayloadTransferSize);
  263. - **class** 类的句柄
  264. - **intf** 接口句柄
  265. usbd_video_open
  266. """"""""""""""""""""""""""""""""""""
  267. ``usbd_video_open`` 用来开启视频数据传输。
  268. .. code-block:: C
  269. void usbd_video_open(uint8_t intf);
  270. - **intf** 开启的接口号
  271. usbd_video_close
  272. """"""""""""""""""""""""""""""""""""
  273. ``usbd_video_close`` 用来关闭视频数据传输。
  274. .. code-block:: C
  275. void usbd_video_open(uint8_t intf);
  276. - **intf** 关闭的接口号
  277. usbd_video_mjpeg_payload_fill
  278. """"""""""""""""""""""""""""""""""""
  279. ``usbd_video_mjpeg_payload_fill`` 用来填充 mjpeg 到新的 buffer中,其中会对 mjpeg 数据按帧进行切分,切分大小由 ``dwMaxPayloadTransferSize`` 控制,并添加头部信息,当前头部字节数为 2。头部信息见 ``struct video_mjpeg_payload_header``
  280. .. code-block:: C
  281. uint32_t usbd_video_mjpeg_payload_fill(uint8_t *input, uint32_t input_len, uint8_t *output, uint32_t *out_len);
  282. - **input** mjpeg 格式的数据包,从 FFD8~FFD9结束
  283. - **input_len** mjpeg数据包大小
  284. - **output** 输出缓冲区
  285. - **out_len** 输出实际要发送的长度大小
  286. - **return** 返回 usb 按照 ``dwMaxPayloadTransferSize`` 大小要发多少帧
  287. DFU
  288. -----------------
  289. PRINTER
  290. -----------------
  291. MTP
  292. -----------------