msc_client_async_seq.c 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242
  1. /*
  2. * SPDX-FileCopyrightText: 2015-2021 Espressif Systems (Shanghai) CO LTD
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. #include <stdint.h>
  7. #include <string.h>
  8. #include <assert.h>
  9. #include "freertos/FreeRTOS.h"
  10. #include "freertos/task.h"
  11. #include "freertos/semphr.h"
  12. #include "esp_err.h"
  13. #include "esp_log.h"
  14. #include "test_usb_mock_classes.h"
  15. #include "msc_client.h"
  16. #include "usb/usb_host.h"
  17. #include "unity.h"
  18. #include "test_utils.h"
  19. /*
  20. Implementation of an MSC client used for USB Host Tests
  21. - Implemented using sequential call patterns, meaning:
  22. - The entire client is contained within a single task
  23. - All API calls and callbacks are run sequentially
  24. - No critical sections required since everything is sequential
  25. - The MSC client will:
  26. - Register itself as a client
  27. - Receive USB_HOST_CLIENT_EVENT_NEW_DEV event message, and open the device
  28. - Allocate IN and OUT transfer objects for MSC SCSI transfers
  29. - Iterate through multiple MSC SCSI block reads
  30. - Free transfer objects
  31. - Close device
  32. - Deregister MSC client
  33. */
  34. #define MAX(x,y) (((x) >= (y)) ? (x) : (y))
  35. #define MSC_CLIENT_MAX_EVENT_MSGS 5
  36. const char *MSC_CLIENT_TAG = "MSC Client";
  37. typedef enum {
  38. TEST_STAGE_WAIT_CONN,
  39. TEST_STAGE_DEV_OPEN,
  40. TEST_STAGE_MSC_RESET,
  41. TEST_STAGE_MSC_CBW,
  42. TEST_STAGE_MSC_DATA,
  43. TEST_STAGE_MSC_CSW,
  44. TEST_STAGE_DEV_CLOSE,
  45. } test_stage_t;
  46. typedef struct {
  47. msc_client_test_param_t test_param;
  48. test_stage_t cur_stage;
  49. test_stage_t next_stage;
  50. uint8_t dev_addr_to_open;
  51. usb_host_client_handle_t client_hdl;
  52. usb_device_handle_t dev_hdl;
  53. int num_sectors_read;
  54. } msc_client_obj_t;
  55. static void msc_transfer_cb(usb_transfer_t *transfer)
  56. {
  57. msc_client_obj_t *msc_obj = (msc_client_obj_t *)transfer->context;
  58. switch (msc_obj->cur_stage) {
  59. case TEST_STAGE_MSC_RESET: {
  60. //Check MSC SCSI interface reset
  61. TEST_ASSERT_EQUAL(USB_TRANSFER_STATUS_COMPLETED, transfer->status);
  62. TEST_ASSERT_EQUAL(transfer->num_bytes, transfer->actual_num_bytes);
  63. msc_obj->next_stage = TEST_STAGE_MSC_CBW;
  64. break;
  65. }
  66. case TEST_STAGE_MSC_CBW: {
  67. //Check MSC SCSI CBW transfer
  68. TEST_ASSERT_EQUAL(USB_TRANSFER_STATUS_COMPLETED, transfer->status);
  69. TEST_ASSERT_EQUAL(sizeof(mock_msc_bulk_cbw_t), transfer->actual_num_bytes);
  70. msc_obj->next_stage = TEST_STAGE_MSC_DATA;
  71. break;
  72. }
  73. case TEST_STAGE_MSC_DATA: {
  74. //Check MSC SCSI data IN transfer
  75. TEST_ASSERT_EQUAL(USB_TRANSFER_STATUS_COMPLETED, transfer->status);
  76. TEST_ASSERT_EQUAL(MOCK_MSC_SCSI_SECTOR_SIZE * msc_obj->test_param.num_sectors_per_xfer, transfer->actual_num_bytes);
  77. msc_obj->next_stage = TEST_STAGE_MSC_CSW;
  78. break;
  79. }
  80. case TEST_STAGE_MSC_CSW: {
  81. //Check MSC SCSI CSW transfer
  82. TEST_ASSERT_EQUAL(USB_TRANSFER_STATUS_COMPLETED, transfer->status);
  83. TEST_ASSERT_EQUAL(true, mock_msc_scsi_check_csw((mock_msc_bulk_csw_t *)transfer->data_buffer, msc_obj->test_param.msc_scsi_xfer_tag));
  84. msc_obj->num_sectors_read += msc_obj->test_param.num_sectors_per_xfer;
  85. if (msc_obj->num_sectors_read < msc_obj->test_param.num_sectors_to_read) {
  86. msc_obj->next_stage = TEST_STAGE_MSC_CBW;
  87. } else {
  88. msc_obj->next_stage = TEST_STAGE_DEV_CLOSE;
  89. }
  90. break;
  91. }
  92. default: {
  93. abort();
  94. break;
  95. }
  96. }
  97. }
  98. static void msc_client_event_cb(const usb_host_client_event_msg_t *event_msg, void *arg)
  99. {
  100. msc_client_obj_t *msc_obj = (msc_client_obj_t *)arg;
  101. switch (event_msg->event) {
  102. case USB_HOST_CLIENT_EVENT_NEW_DEV:
  103. TEST_ASSERT_EQUAL(TEST_STAGE_WAIT_CONN, msc_obj->cur_stage);
  104. msc_obj->next_stage = TEST_STAGE_DEV_OPEN;
  105. msc_obj->dev_addr_to_open = event_msg->new_dev.address;
  106. break;
  107. default:
  108. abort(); //Should never occur in this test
  109. break;
  110. }
  111. }
  112. void msc_client_async_seq_task(void *arg)
  113. {
  114. msc_client_obj_t msc_obj;
  115. memcpy(&msc_obj.test_param, arg, sizeof(msc_client_test_param_t));
  116. msc_obj.cur_stage = TEST_STAGE_WAIT_CONN;
  117. msc_obj.next_stage = TEST_STAGE_WAIT_CONN;
  118. msc_obj.client_hdl = NULL;
  119. msc_obj.dev_addr_to_open = 0;
  120. msc_obj.dev_hdl = NULL;
  121. msc_obj.num_sectors_read = 0;
  122. //Register client
  123. usb_host_client_config_t client_config = {
  124. .client_event_callback = msc_client_event_cb,
  125. .callback_arg = (void *)&msc_obj,
  126. .max_num_event_msg = MSC_CLIENT_MAX_EVENT_MSGS,
  127. };
  128. TEST_ASSERT_EQUAL(ESP_OK, usb_host_client_register(&client_config, &msc_obj.client_hdl));
  129. //Allocate transfers
  130. usb_transfer_t *xfer_out = NULL; //Must be large enough to contain CBW and MSC reset control transfer
  131. usb_transfer_t *xfer_in = NULL; //Must be large enough to contain CSW and Data
  132. size_t out_worst_case_size = MAX(sizeof(mock_msc_bulk_cbw_t), sizeof(usb_setup_packet_t));
  133. size_t in_worst_case_size = usb_round_up_to_mps(MAX(MOCK_MSC_SCSI_SECTOR_SIZE * msc_obj.test_param.num_sectors_per_xfer, sizeof(mock_msc_bulk_csw_t)), MOCK_MSC_SCSI_BULK_EP_MPS);
  134. TEST_ASSERT_EQUAL(ESP_OK, usb_host_transfer_alloc(out_worst_case_size, 0, &xfer_out));
  135. TEST_ASSERT_EQUAL(ESP_OK, usb_host_transfer_alloc(in_worst_case_size, 0, &xfer_in));
  136. xfer_out->callback = msc_transfer_cb;
  137. xfer_in->callback = msc_transfer_cb;
  138. xfer_out->context = (void *)&msc_obj;
  139. xfer_in->context = (void *)&msc_obj;
  140. //Wait to be started by main thread
  141. ulTaskNotifyTake(pdTRUE, portMAX_DELAY);
  142. ESP_LOGD(MSC_CLIENT_TAG, "Starting");
  143. bool exit_loop = false;
  144. bool skip_event_handling = false;
  145. while (!exit_loop) {
  146. if (!skip_event_handling) {
  147. TEST_ASSERT_EQUAL(ESP_OK, usb_host_client_handle_events(msc_obj.client_hdl, portMAX_DELAY));
  148. }
  149. skip_event_handling = false;
  150. if (msc_obj.cur_stage == msc_obj.next_stage) {
  151. continue;
  152. }
  153. msc_obj.cur_stage = msc_obj.next_stage;
  154. switch (msc_obj.cur_stage) {
  155. case TEST_STAGE_DEV_OPEN: {
  156. ESP_LOGD(MSC_CLIENT_TAG, "Open");
  157. //Open the device
  158. TEST_ASSERT_EQUAL(ESP_OK, usb_host_device_open(msc_obj.client_hdl, msc_obj.dev_addr_to_open, &msc_obj.dev_hdl));
  159. //Target our transfers to the device
  160. xfer_out->device_handle = msc_obj.dev_hdl;
  161. xfer_in->device_handle = msc_obj.dev_hdl;
  162. //Check the VID/PID of the opened device
  163. const usb_device_desc_t *device_desc;
  164. TEST_ASSERT_EQUAL(ESP_OK, usb_host_get_device_descriptor(msc_obj.dev_hdl, &device_desc));
  165. TEST_ASSERT_EQUAL(msc_obj.test_param.idVendor, device_desc->idVendor);
  166. TEST_ASSERT_EQUAL(msc_obj.test_param.idProduct, device_desc->idProduct);
  167. //Claim the MSC interface
  168. 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));
  169. msc_obj.next_stage = TEST_STAGE_MSC_RESET;
  170. skip_event_handling = true; //Need to execute TEST_STAGE_MSC_RESET
  171. break;
  172. }
  173. case TEST_STAGE_MSC_RESET: {
  174. ESP_LOGD(MSC_CLIENT_TAG, "MSC Reset");
  175. //Send an MSC SCSI interface reset
  176. MOCK_MSC_SCSI_REQ_INIT_RESET((usb_setup_packet_t *)xfer_out->data_buffer, MOCK_MSC_SCSI_INTF_NUMBER);
  177. xfer_out->num_bytes = sizeof(usb_setup_packet_t);
  178. xfer_out->bEndpointAddress = 0;
  179. TEST_ASSERT_EQUAL(ESP_OK, usb_host_transfer_submit_control(msc_obj.client_hdl, xfer_out));
  180. //Next stage set from transfer callback
  181. break;
  182. }
  183. case TEST_STAGE_MSC_CBW: {
  184. ESP_LOGD(MSC_CLIENT_TAG, "CBW");
  185. mock_msc_scsi_init_cbw((mock_msc_bulk_cbw_t *)xfer_out->data_buffer, true, msc_obj.next_stage, msc_obj.test_param.num_sectors_per_xfer, msc_obj.test_param.msc_scsi_xfer_tag);
  186. xfer_out->num_bytes = sizeof(mock_msc_bulk_cbw_t);
  187. xfer_out->bEndpointAddress = MOCK_MSC_SCSI_BULK_OUT_EP_ADDR;
  188. TEST_ASSERT_EQUAL(ESP_OK, usb_host_transfer_submit(xfer_out));
  189. //Next stage set from transfer callback
  190. break;
  191. }
  192. case TEST_STAGE_MSC_DATA: {
  193. ESP_LOGD(MSC_CLIENT_TAG, "Data");
  194. xfer_in->num_bytes = usb_round_up_to_mps(MOCK_MSC_SCSI_SECTOR_SIZE * msc_obj.test_param.num_sectors_per_xfer, MOCK_MSC_SCSI_BULK_EP_MPS);
  195. xfer_in->bEndpointAddress = MOCK_MSC_SCSI_BULK_IN_EP_ADDR;
  196. TEST_ASSERT_EQUAL(ESP_OK, usb_host_transfer_submit(xfer_in));
  197. //Next stage set from transfer callback
  198. break;
  199. }
  200. case TEST_STAGE_MSC_CSW: {
  201. ESP_LOGD(MSC_CLIENT_TAG, "CSW");
  202. xfer_in->num_bytes = usb_round_up_to_mps(sizeof(mock_msc_bulk_csw_t), MOCK_MSC_SCSI_BULK_EP_MPS);
  203. xfer_in->bEndpointAddress = MOCK_MSC_SCSI_BULK_IN_EP_ADDR;
  204. TEST_ASSERT_EQUAL(ESP_OK, usb_host_transfer_submit(xfer_in));
  205. //Next stage set from transfer callback
  206. break;
  207. }
  208. case TEST_STAGE_DEV_CLOSE: {
  209. ESP_LOGD(MSC_CLIENT_TAG, "Close");
  210. TEST_ASSERT_EQUAL(ESP_OK, usb_host_interface_release(msc_obj.client_hdl, msc_obj.dev_hdl, MOCK_MSC_SCSI_INTF_NUMBER));
  211. TEST_ASSERT_EQUAL(ESP_OK, usb_host_device_close(msc_obj.client_hdl, msc_obj.dev_hdl));
  212. exit_loop = true;
  213. break;
  214. }
  215. default:
  216. abort();
  217. break;
  218. }
  219. }
  220. //Free transfers and deregister the client
  221. TEST_ASSERT_EQUAL(ESP_OK, usb_host_transfer_free(xfer_out));
  222. TEST_ASSERT_EQUAL(ESP_OK, usb_host_transfer_free(xfer_in));
  223. TEST_ASSERT_EQUAL(ESP_OK, usb_host_client_deregister(msc_obj.client_hdl));
  224. ESP_LOGD(MSC_CLIENT_TAG, "Done");
  225. vTaskDelete(NULL);
  226. }