test_usbd.c 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332
  1. /*
  2. * Copyright (c) 2022-2024, Xiaohua Semiconductor Co., Ltd.
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. *
  6. * Change Logs:
  7. * Date Author Notes
  8. * 2024-12-30 CDT first version
  9. */
  10. #include <rtthread.h>
  11. #include <rtdevice.h>
  12. #ifdef RT_USB_DEVICE_CDC
  13. /* menuconfig:
  14. 1.Hardware Drivers Config--->On-Chip Peripheral Driver--->Enable USB--->
  15. [*]Use USBFS Core
  16. Select USB Mode(USB Device Mode)
  17. [*]Enable VBUS Sensing for Device
  18. 2.RT-Thread Components--->
  19. Using USB legacy version--->
  20. Using USB device--->Device type--->
  21. [*]Enable to use device as CDC device
  22. */
  23. /*
  24. * 程序清单:这是一个 usb device 设备使用例程
  25. * 例程导出了 cdc_sample 命令到控制终端
  26. * PC上需要使用串口助手以DTR[√]方式打开USB的虚拟串口(比如 SSCOM 有这个功能)
  27. * 命令调用格式:cdc_sample
  28. * 程序功能:首先会打印三次str_write字符串内容,同时虚拟串口可输入发送任意小于255个字符的字符串,
  29. * 发送内容可在Finsh串口显示。
  30. */
  31. #define USBD_DEV_NAME "vcom" /* 名称 */
  32. rt_uint8_t str_read[255];
  33. static rt_err_t cdc_rx_handle(rt_device_t dev, rt_size_t size)
  34. {
  35. /* 读取虚拟串口接收内容 */
  36. rt_device_read(dev, 0, str_read, size);
  37. rt_kprintf("Read message: %s\n", str_read);
  38. return RT_EOK;
  39. }
  40. static int cdc_sample(void)
  41. {
  42. rt_uint8_t i;
  43. rt_err_t ret = RT_EOK;
  44. rt_device_t cdc_dev = RT_NULL; /* usb device设备句柄 */
  45. rt_uint8_t str_write[] = "This is a usb cdc device test!\r\n";
  46. /* 查找USB虚拟串口设备 */
  47. cdc_dev = rt_device_find(USBD_DEV_NAME);
  48. if (cdc_dev == RT_NULL)
  49. {
  50. rt_kprintf("cdc sample run failed! can't find %s device!\n", USBD_DEV_NAME);
  51. return -RT_ERROR;
  52. }
  53. /* 以读写方式打开设备 */
  54. ret = rt_device_open(cdc_dev, RT_DEVICE_FLAG_INT_RX);
  55. if (ret != RT_EOK)
  56. {
  57. rt_kprintf("open %s device failed!\n", USBD_DEV_NAME);
  58. return ret;
  59. }
  60. rt_device_set_rx_indicate(cdc_dev, cdc_rx_handle);
  61. for (i = 1; i < 4; i++)
  62. {
  63. rt_kprintf("Start to send test message 3 timers :%d.\n", i);
  64. if (rt_device_write(cdc_dev, 0, str_write, sizeof(str_write)) != sizeof(str_write))
  65. {
  66. rt_kprintf("send test message failed\n");
  67. return -RT_ERROR;
  68. }
  69. /* 延时1000ms */
  70. rt_thread_mdelay(1000);
  71. }
  72. // for (;;);
  73. return ret;
  74. }
  75. /* 导出到 msh 命令列表中 */
  76. MSH_CMD_EXPORT(cdc_sample, usbd cdc sample);
  77. #endif
  78. #if defined(RT_USB_DEVICE_MSTORAGE)
  79. /* F4A0 only FS can used with spi flash */
  80. #if (defined(HC32F4A0) && defined(BSP_USING_USBFS)) || \
  81. defined(HC32F460) || defined(HC32F472)
  82. /* Enable spibus1, SFUD, usb msc */
  83. /* menuconfig:
  84. 1. Hardware Drivers Config--->On-Chip Peripheral Driver--->Enable SPI BUS--->Enable SPI1 BUS
  85. 2. Hardware Drivers Config--->On-Chip Peripheral Driver--->Enable USB--->
  86. [*]Use USBFS Core
  87. Select USB Mode(USB Device Mode)
  88. [*]Enable VBUS Sensing for Device
  89. 3. RT-Thread Components--->Device Drivers--->Using SPI Bus/Device device drivers
  90. [*]Using Serial Flash Universal Driver
  91. [*]Using auto probe flash JEDEC SFDP parameters
  92. [*]Using defined supported flash chip information table
  93. (50000000)Default spi maximum speed(HZ)
  94. 4. RT-Thread Components--->Using USB legacy version
  95. [*]Using USB device--->
  96. Device type--->...Mass Storage device
  97. (spiflash)msc class disk name
  98. */
  99. #include "drv_gpio.h"
  100. #include "drv_spi.h"
  101. #include "dev_spi_flash_sfud.h"
  102. #define SPI_FLASH_CHIP RT_USB_MSTORAGE_DISK_NAME /* msc class disk name */
  103. #if defined(HC32F4A0)
  104. #define SPI_FLASH_SS_PORT GPIO_PORT_C
  105. #define SPI_FLASH_SS_PIN GPIO_PIN_07
  106. #define SPI_BUS_NAME "spi1"
  107. #define SPI_FLASH_DEVICE_NAME "spi10"
  108. #elif defined(HC32F460)
  109. #define SPI_FLASH_SS_PORT GPIO_PORT_C
  110. #define SPI_FLASH_SS_PIN GPIO_PIN_07
  111. #define SPI_BUS_NAME "spi3"
  112. #define SPI_FLASH_DEVICE_NAME "spi30"
  113. #elif defined(HC32F472)
  114. #define SPI_FLASH_SS_PORT GPIO_PORT_B
  115. #define SPI_FLASH_SS_PIN GPIO_PIN_12
  116. #define SPI_BUS_NAME "spi1"
  117. #define SPI_FLASH_DEVICE_NAME "spi10"
  118. #endif
  119. static void rt_hw_spi_flash_reset(char *spi_dev_name)
  120. {
  121. struct rt_spi_device *spi_dev_w25;
  122. rt_uint8_t w25_en_reset = 0x66;
  123. rt_uint8_t w25_reset_dev = 0x99;
  124. spi_dev_w25 = (struct rt_spi_device *)rt_device_find(spi_dev_name);
  125. if (!spi_dev_w25)
  126. {
  127. rt_kprintf("Can't find %s device!\n", spi_dev_name);
  128. }
  129. else
  130. {
  131. rt_spi_send(spi_dev_w25, &w25_en_reset, 1U);
  132. rt_spi_send(spi_dev_w25, &w25_reset_dev, 1U);
  133. DDL_DelayMS(1U);
  134. rt_kprintf("Reset ext flash!\n");
  135. }
  136. }
  137. static int rt_hw_spi_flash_with_sfud_init(void)
  138. {
  139. #if defined(HC32F4A0) || defined(HC32F460)
  140. rt_hw_spi_device_attach(SPI_BUS_NAME, SPI_FLASH_DEVICE_NAME, GET_PIN(C, 7));
  141. #elif defined(HC32F472)
  142. rt_hw_spi_device_attach(SPI_BUS_NAME, SPI_FLASH_DEVICE_NAME, GET_PIN(B, 12));
  143. #endif
  144. if (RT_NULL == rt_sfud_flash_probe(SPI_FLASH_CHIP, SPI_FLASH_DEVICE_NAME))
  145. {
  146. rt_hw_spi_flash_reset(SPI_FLASH_DEVICE_NAME);
  147. if (RT_NULL == rt_sfud_flash_probe(SPI_FLASH_CHIP, SPI_FLASH_DEVICE_NAME))
  148. {
  149. return -RT_ERROR;
  150. }
  151. }
  152. return RT_EOK;
  153. }
  154. INIT_COMPONENT_EXPORT(rt_hw_spi_flash_with_sfud_init);
  155. #endif
  156. #endif
  157. #ifdef RT_USB_DEVICE_HID
  158. #include "drv_gpio.h"
  159. /* menuconfig:
  160. 1. Hardware Drivers Config--->On-Chip Peripheral Driver--->Enable USB--->
  161. [*]Use USBFS Core
  162. Select USB Mode(USB Device Mode)
  163. [*]Enable VBUS Sensing for Device
  164. 2. RT-Thread Components--->
  165. Using USB legacy version--->
  166. Using USB device--->Device type--->
  167. [*]Enable to use device as HID device
  168. */
  169. /*
  170. * 程序清单:这是一个 usb hid device 设备使用例程
  171. * 例程导出了 hid_sample 命令到控制终端
  172. * 命令调用格式:hid_sample
  173. * 程序功能:首先会打印str_write[0]字符串内容,然后按下按键WKUP,hid设备将发送按键keyn,n是按下次数,可
  174. * 通过bus hound查看数据。
  175. * 发送内容可在Finsh串口显示。
  176. */
  177. #define USBD_DEV_NAME "hidd" /* 名称 */
  178. #if defined(HC32F4A0)
  179. #define KEY_PIN_NUM GET_PIN(A,0) /* PA0 */
  180. #elif defined(HC32F460)
  181. #define KEY_PIN_NUM GET_PIN(B,1) /* PB1 */
  182. #elif defined(HC32F472)
  183. #define KEY_PIN_NUM GET_PIN(B,5) /* PB5 */
  184. #endif
  185. static int hid_sample(void)
  186. {
  187. rt_err_t ret = RT_EOK;
  188. rt_device_t hid_dev = RT_NULL; /* usb device设备句柄 */
  189. char str_write[2][5] = {"test", "Key0"};
  190. /* 查找设备 */
  191. hid_dev = rt_device_find(USBD_DEV_NAME);
  192. if (hid_dev == RT_NULL)
  193. {
  194. rt_kprintf("hid sample run failed! can't find %s device!\n", USBD_DEV_NAME);
  195. return -RT_ERROR;
  196. }
  197. /* 以收中断方式打开设备 */
  198. ret = rt_device_open(hid_dev, RT_DEVICE_OFLAG_RDWR);
  199. if (ret != RT_EOK)
  200. {
  201. rt_kprintf("open %s device failed!\n", USBD_DEV_NAME);
  202. return ret;
  203. }
  204. rt_kprintf("This is a usb hid device test!\r\n");
  205. rt_device_write(hid_dev, 0, str_write[0], sizeof(str_write[0]));
  206. rt_pin_mode(KEY_PIN_NUM, PIN_MODE_INPUT);
  207. for (;;)
  208. {
  209. rt_thread_mdelay(200);
  210. if (PIN_LOW == rt_pin_read(KEY_PIN_NUM))
  211. {
  212. rt_device_write(hid_dev, 0, str_write[1], sizeof(str_write[1]));
  213. str_write[1][3] += 0x01;
  214. }
  215. }
  216. //return ret;
  217. }
  218. /* 导出到 msh 命令列表中 */
  219. MSH_CMD_EXPORT(hid_sample, usbd hid sample);
  220. #endif
  221. #ifdef RT_USB_DEVICE_WINUSB
  222. /* menuconfig:
  223. 1. Hardware Drivers Config--->On-Chip Peripheral Driver--->Enable USB--->
  224. [*]Use USBFS Core
  225. Select USB Mode(USB Device Mode)
  226. [*]Enable VBUS Sensing for Device
  227. 2. RT-Thread Components--->
  228. Using USB legacy version--->
  229. Using USB device--->Device type--->
  230. [*]Enable to use device as winusb device
  231. */
  232. /*
  233. * 程序清单:这是一个 usb WINUSB device 设备使用例程
  234. * RTT 的WINUSB Windows无法免驱,需要使用zadig安装winusb驱动程序(如设备管理器-通用串行总线设备-RTT Win USB已识别则不需要安装)。
  235. * 例程导出了 winusb_sample 命令到控制终端
  236. * 命令调用格式:winusb_sample
  237. * 软件:llcom.exe
  238. * 程序功能:MSH命令发送winusb_sample,运行测试程序。
  239. * 打开llcom.exe软件,选择小工具-WinUSB设备-选择对应RTT Win USB设备-打开-勾选Hex发送-发送数据。
  240. * 通过llcom.exe可发送bulk数据(100字符以内)到设备,设备收到后会回发给主机(llcom.exe),同时通过MSH终端显示收到的HEX数据。
  241. * 注意:1、llcom.exe中的GUID与驱动程序中设定保持一致(通过设备管理器选择RTT Win USB设备的属性来查看);
  242. * 2、win_usb_read()函数中的UIO_REQUEST_READ_FULL改为UIO_REQUEST_READ_BEST,实现数据即读即取;
  243. * 否则需要接满传入的size数量,才会回调接收函数。
  244. *
  245. */
  246. #define WINUSB_DEV_NAME "winusb" /* 名称 */
  247. uint8_t str_read[100];
  248. static rt_err_t winusb_rx_handle(rt_device_t dev, rt_size_t size)
  249. {
  250. uint8_t i;
  251. /* 读取定时器当前值 */
  252. rt_kprintf("Rx:");
  253. for (i = 0; i < size; i++)
  254. {
  255. rt_kprintf("%x", str_read[i]);
  256. }
  257. rt_kprintf("\r\n");
  258. rt_device_write(dev, 0, str_read, size);
  259. /* prepare read config */
  260. rt_device_read(dev, 0, str_read, sizeof(str_read));
  261. return RT_EOK;
  262. }
  263. static int winusb_sample(void)
  264. {
  265. rt_err_t ret = RT_EOK;
  266. rt_device_t winusb_dev = RT_NULL; /* usb device设备句柄 */
  267. /* 查找设备 */
  268. winusb_dev = rt_device_find(WINUSB_DEV_NAME);
  269. if (winusb_dev == RT_NULL)
  270. {
  271. rt_kprintf("winusb sample run failed! can't find %s device!\n", WINUSB_DEV_NAME);
  272. return -RT_ERROR;
  273. }
  274. /* 以读写方式打开设备 */
  275. ret = rt_device_open(winusb_dev, RT_DEVICE_OFLAG_RDWR);
  276. if (ret != RT_EOK)
  277. {
  278. rt_kprintf("open %s device failed!\n", WINUSB_DEV_NAME);
  279. return ret;
  280. }
  281. rt_kprintf("Found and open success %s device!\n", WINUSB_DEV_NAME);
  282. ret = rt_device_set_rx_indicate(winusb_dev, winusb_rx_handle);
  283. if (ret == RT_EOK)
  284. {
  285. /* prepare read config,set once,read once, */
  286. rt_device_read(winusb_dev, 0, str_read, sizeof(str_read));
  287. }
  288. return ret;
  289. }
  290. /* 导出到 msh 命令列表中 */
  291. MSH_CMD_EXPORT(winusb_sample, usbd winusb sample);
  292. #endif