msc_client_async_dconn.c 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262
  1. /*
  2. * SPDX-FileCopyrightText: 2015-2022 Espressif Systems (Shanghai) CO LTD
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. #include <stdint.h>
  7. #include <string.h>
  8. #include <stdlib.h>
  9. #include <sys/param.h>
  10. #include "freertos/FreeRTOS.h"
  11. #include "freertos/task.h"
  12. #include "esp_err.h"
  13. #include "esp_log.h"
  14. #include "test_usb_mock_classes.h"
  15. #include "test_usb_common.h"
  16. #include "msc_client.h"
  17. #include "usb/usb_host.h"
  18. #include "unity.h"
  19. #include "test_utils.h"
  20. /*
  21. Implementation of an asynchronous MSC client used for USB Host disconnection test.
  22. - The MSC client will:
  23. - Register itself as a client
  24. - Receive USB_HOST_CLIENT_EVENT_NEW_DEV event message, and open the device
  25. - Allocate IN and OUT transfer objects for MSC SCSI transfers
  26. - Trigger a single MSC SCSI transfer
  27. - Split the data stage into multiple transfers (so that the endpoint multiple queued up transfers)
  28. - Cause a disconnection mid-way through the data stage
  29. - All of the transfers should be automatically deqeueud
  30. - Then a USB_HOST_CLIENT_EVENT_DEV_GONE event should occur afterwards
  31. - Free transfer objects
  32. - Close device
  33. - Deregister MSC client
  34. */
  35. #define TEST_DCONN_ITERATIONS 3
  36. typedef enum {
  37. TEST_STAGE_WAIT_CONN,
  38. TEST_STAGE_DEV_OPEN,
  39. TEST_STAGE_MSC_RESET,
  40. TEST_STAGE_MSC_CBW,
  41. TEST_STAGE_MSC_DATA_DCONN,
  42. TEST_STAGE_DEV_CLOSE,
  43. } test_stage_t;
  44. typedef struct {
  45. msc_client_test_param_t test_param;
  46. test_stage_t cur_stage;
  47. test_stage_t next_stage;
  48. uint8_t dev_addr_to_open;
  49. usb_host_client_handle_t client_hdl;
  50. usb_device_handle_t dev_hdl;
  51. int num_data_transfers;
  52. int event_count;
  53. } msc_client_obj_t;
  54. static void msc_reset_cbw_transfer_cb(usb_transfer_t *transfer)
  55. {
  56. msc_client_obj_t *msc_obj = (msc_client_obj_t *)transfer->context;
  57. //We expect the reset and CBW transfers to complete with no issues
  58. TEST_ASSERT_EQUAL(USB_TRANSFER_STATUS_COMPLETED, transfer->status);
  59. TEST_ASSERT_EQUAL(transfer->num_bytes, transfer->actual_num_bytes);
  60. switch (msc_obj->cur_stage) {
  61. case TEST_STAGE_MSC_RESET:
  62. msc_obj->next_stage = TEST_STAGE_MSC_CBW;
  63. break;
  64. case TEST_STAGE_MSC_CBW:
  65. msc_obj->next_stage = TEST_STAGE_MSC_DATA_DCONN;
  66. break;
  67. default:
  68. abort();
  69. break;
  70. }
  71. }
  72. static void msc_data_transfer_cb(usb_transfer_t *transfer)
  73. {
  74. //The data stage should have either completed, or failed due to the disconnection.
  75. TEST_ASSERT(transfer->status == USB_TRANSFER_STATUS_COMPLETED || transfer->status == USB_TRANSFER_STATUS_NO_DEVICE);
  76. if (transfer->status == USB_TRANSFER_STATUS_COMPLETED) {
  77. TEST_ASSERT_EQUAL(transfer->num_bytes, transfer->actual_num_bytes);
  78. } else {
  79. TEST_ASSERT_EQUAL(0, transfer->actual_num_bytes);
  80. }
  81. msc_client_obj_t *msc_obj = (msc_client_obj_t *)transfer->context;
  82. msc_obj->event_count++;
  83. //If all transfers dequeued and device gone event occurred. Go to next stage
  84. if (msc_obj->event_count >= msc_obj->num_data_transfers + 1) {
  85. msc_obj->next_stage = TEST_STAGE_DEV_CLOSE;
  86. }
  87. }
  88. static void msc_client_event_cb(const usb_host_client_event_msg_t *event_msg, void *arg)
  89. {
  90. msc_client_obj_t *msc_obj = (msc_client_obj_t *)arg;
  91. switch (event_msg->event) {
  92. case USB_HOST_CLIENT_EVENT_NEW_DEV:
  93. TEST_ASSERT_EQUAL(TEST_STAGE_WAIT_CONN, msc_obj->cur_stage);
  94. msc_obj->next_stage = TEST_STAGE_DEV_OPEN;
  95. msc_obj->dev_addr_to_open = event_msg->new_dev.address;
  96. break;
  97. case USB_HOST_CLIENT_EVENT_DEV_GONE:
  98. msc_obj->event_count++;
  99. //If all transfers dequeued and device gone event occurred. Go to next stage
  100. if (msc_obj->event_count >= msc_obj->num_data_transfers + 1) {
  101. msc_obj->next_stage = TEST_STAGE_DEV_CLOSE;
  102. }
  103. break;
  104. default:
  105. abort(); //Should never occur in this test
  106. break;
  107. }
  108. }
  109. void msc_client_async_dconn_task(void *arg)
  110. {
  111. msc_client_obj_t msc_obj;
  112. memcpy(&msc_obj.test_param, arg, sizeof(msc_client_test_param_t));
  113. msc_obj.cur_stage = TEST_STAGE_WAIT_CONN;
  114. msc_obj.next_stage = TEST_STAGE_WAIT_CONN;
  115. msc_obj.dev_addr_to_open = 0;
  116. msc_obj.client_hdl = NULL;
  117. msc_obj.dev_hdl = NULL;
  118. msc_obj.num_data_transfers = msc_obj.test_param.num_sectors_per_xfer / MOCK_MSC_SCSI_SECTOR_SIZE;
  119. msc_obj.event_count = 0;
  120. //Register client
  121. usb_host_client_config_t client_config = {
  122. .is_synchronous = false,
  123. .max_num_event_msg = MSC_ASYNC_CLIENT_MAX_EVENT_MSGS,
  124. .async = {
  125. .client_event_callback = msc_client_event_cb,
  126. .callback_arg = (void *)&msc_obj,
  127. },
  128. };
  129. TEST_ASSERT_EQUAL(ESP_OK, usb_host_client_register(&client_config, &msc_obj.client_hdl));
  130. //Allocate transfers
  131. usb_transfer_t *xfer_out; //Must be large enough to contain CBW and MSC reset control transfer
  132. usb_transfer_t *xfer_in[msc_obj.num_data_transfers]; //We manually split the data stage into multiple transfers
  133. size_t xfer_out_size = MAX(sizeof(mock_msc_bulk_cbw_t), sizeof(usb_setup_packet_t));
  134. size_t xfer_in_size = MOCK_MSC_SCSI_SECTOR_SIZE;
  135. TEST_ASSERT_EQUAL(ESP_OK, usb_host_transfer_alloc(xfer_out_size, 0, &xfer_out));
  136. xfer_out->context = (void *)&msc_obj;
  137. for (int i = 0; i < msc_obj.num_data_transfers; i++) {
  138. TEST_ASSERT_EQUAL(ESP_OK, usb_host_transfer_alloc(xfer_in_size, 0, &xfer_in[i]));
  139. xfer_in[i]->context = (void *)&msc_obj;
  140. }
  141. //Wait to be started by main thread
  142. ulTaskNotifyTake(pdTRUE, portMAX_DELAY);
  143. ESP_LOGD(MSC_CLIENT_TAG, "Starting");
  144. bool exit_loop = false;
  145. bool skip_event_handling = false;
  146. int dconn_iter = 0;
  147. while (!exit_loop) {
  148. if (!skip_event_handling) {
  149. TEST_ASSERT_EQUAL(ESP_OK, usb_host_client_handle_events(msc_obj.client_hdl, portMAX_DELAY));
  150. }
  151. skip_event_handling = false;
  152. if (msc_obj.cur_stage == msc_obj.next_stage) {
  153. continue;
  154. }
  155. msc_obj.cur_stage = msc_obj.next_stage;
  156. switch (msc_obj.cur_stage) {
  157. case TEST_STAGE_WAIT_CONN: {
  158. //Nothing to do while waiting for connection
  159. break;
  160. }
  161. case TEST_STAGE_DEV_OPEN: {
  162. ESP_LOGD(MSC_CLIENT_TAG, "Open");
  163. //Open the device
  164. TEST_ASSERT_EQUAL(ESP_OK, usb_host_device_open(msc_obj.client_hdl, msc_obj.dev_addr_to_open, &msc_obj.dev_hdl));
  165. //Target our transfers to the device
  166. xfer_out->device_handle = msc_obj.dev_hdl;
  167. xfer_out->callback = msc_reset_cbw_transfer_cb;
  168. for (int i = 0; i < msc_obj.num_data_transfers; i++) {
  169. xfer_in[i]->device_handle = msc_obj.dev_hdl;
  170. xfer_in[i]->callback = msc_data_transfer_cb;
  171. }
  172. //Check the VID/PID of the opened device
  173. const usb_device_desc_t *device_desc;
  174. TEST_ASSERT_EQUAL(ESP_OK, usb_host_get_device_descriptor(msc_obj.dev_hdl, &device_desc));
  175. TEST_ASSERT_EQUAL(msc_obj.test_param.idVendor, device_desc->idVendor);
  176. TEST_ASSERT_EQUAL(msc_obj.test_param.idProduct, device_desc->idProduct);
  177. //Claim the MSC interface
  178. TEST_ASSERT_EQUAL(ESP_OK, usb_host_interface_claim(msc_obj.client_hdl, msc_obj.dev_hdl, MOCK_MSC_SCSI_INTF_NUMBER, MOCK_MSC_SCSI_INTF_ALT_SETTING));
  179. msc_obj.next_stage = TEST_STAGE_MSC_RESET;
  180. skip_event_handling = true; //Need to execute TEST_STAGE_MSC_RESET
  181. break;
  182. }
  183. case TEST_STAGE_MSC_RESET: {
  184. ESP_LOGD(MSC_CLIENT_TAG, "MSC Reset");
  185. //Send an MSC SCSI interface reset
  186. MOCK_MSC_SCSI_REQ_INIT_RESET((usb_setup_packet_t *)xfer_out->data_buffer, MOCK_MSC_SCSI_INTF_NUMBER);
  187. xfer_out->num_bytes = sizeof(usb_setup_packet_t);
  188. xfer_out->bEndpointAddress = 0;
  189. TEST_ASSERT_EQUAL(ESP_OK, usb_host_transfer_submit_control(msc_obj.client_hdl, xfer_out));
  190. //Next stage set from transfer callback
  191. break;
  192. }
  193. case TEST_STAGE_MSC_CBW: {
  194. ESP_LOGD(MSC_CLIENT_TAG, "CBW");
  195. mock_msc_scsi_init_cbw((mock_msc_bulk_cbw_t *)xfer_out->data_buffer, true, 0, msc_obj.test_param.num_sectors_per_xfer, msc_obj.test_param.msc_scsi_xfer_tag);
  196. xfer_out->num_bytes = sizeof(mock_msc_bulk_cbw_t);
  197. xfer_out->bEndpointAddress = MOCK_MSC_SCSI_BULK_OUT_EP_ADDR;
  198. TEST_ASSERT_EQUAL(ESP_OK, usb_host_transfer_submit(xfer_out));
  199. //Next stage set from transfer callback
  200. break;
  201. }
  202. case TEST_STAGE_MSC_DATA_DCONN: {
  203. ESP_LOGD(MSC_CLIENT_TAG, "Data and disconnect");
  204. //Setup the Data IN transfers
  205. for (int i = 0; i < msc_obj.num_data_transfers; i++) {
  206. xfer_in[i]->num_bytes = usb_round_up_to_mps(MOCK_MSC_SCSI_SECTOR_SIZE, MOCK_MSC_SCSI_BULK_EP_MPS);
  207. xfer_in[i]->bEndpointAddress = MOCK_MSC_SCSI_BULK_IN_EP_ADDR;
  208. }
  209. //Submit those transfers
  210. for (int i = 0; i < msc_obj.num_data_transfers; i++) {
  211. TEST_ASSERT_EQUAL(ESP_OK, usb_host_transfer_submit(xfer_in[i]));
  212. }
  213. //Trigger a disconnect
  214. test_usb_set_phy_state(false, 0);
  215. //Next stage set from transfer callback
  216. break;
  217. }
  218. case TEST_STAGE_DEV_CLOSE: {
  219. ESP_LOGD(MSC_CLIENT_TAG, "Close");
  220. TEST_ASSERT_EQUAL(ESP_OK, usb_host_interface_release(msc_obj.client_hdl, msc_obj.dev_hdl, MOCK_MSC_SCSI_INTF_NUMBER));
  221. TEST_ASSERT_EQUAL(ESP_OK, usb_host_device_close(msc_obj.client_hdl, msc_obj.dev_hdl));
  222. dconn_iter++;
  223. if (dconn_iter < TEST_DCONN_ITERATIONS) {
  224. //Start the next test iteration by going back to TEST_STAGE_WAIT_CONN and reenabling connections
  225. msc_obj.next_stage = TEST_STAGE_WAIT_CONN;
  226. skip_event_handling = true; //Need to execute TEST_STAGE_WAIT_CONN
  227. test_usb_set_phy_state(true, 0);
  228. } else {
  229. exit_loop = true;
  230. }
  231. break;
  232. }
  233. default:
  234. abort();
  235. break;
  236. }
  237. }
  238. //Free transfers
  239. TEST_ASSERT_EQUAL(ESP_OK, usb_host_transfer_free(xfer_out));
  240. for (int i = 0; i < msc_obj.num_data_transfers; i++) {
  241. TEST_ASSERT_EQUAL(ESP_OK, usb_host_transfer_free(xfer_in[i]));
  242. }
  243. //Deregister the client
  244. TEST_ASSERT_EQUAL(ESP_OK, usb_host_client_deregister(msc_obj.client_hdl));
  245. ESP_LOGD(MSC_CLIENT_TAG, "Done");
  246. vTaskDelete(NULL);
  247. }