2021-12-08 19:46:46 +08:00
/*
* SPDX - FileCopyrightText : 2015 - 2022 Espressif Systems ( Shanghai ) CO LTD
*
* SPDX - License - Identifier : Apache - 2.0
*/
# include <stdint.h>
# include <string.h>
# include <stdlib.h>
# include <sys/param.h>
# include "freertos/FreeRTOS.h"
# include "freertos/task.h"
# include "esp_err.h"
# include "esp_log.h"
2022-12-13 21:21:22 +08:00
# include "test_usb_mock_msc.h"
2021-12-08 19:46:46 +08:00
# include "test_usb_common.h"
# include "msc_client.h"
# include "usb/usb_host.h"
# include "unity.h"
/*
Implementation of an asynchronous MSC client used for USB Host enumeration test .
- The MSC client will :
- Register itself as a client
- Receive USB_HOST_CLIENT_EVENT_NEW_DEV event message , and open the device
- Check the device and configuration descriptor of the device
- Check the device ' s information
- Close device
2021-12-14 18:28:04 +08:00
- Repeat for multiple iterations from waiting connection by forcing a disconnection
2021-12-08 19:46:46 +08:00
- Deregister MSC client
*/
2021-12-14 18:28:04 +08:00
# define TEST_ENUM_ITERATIONS 3
2021-12-08 19:46:46 +08:00
typedef enum {
TEST_STAGE_WAIT_CONN ,
TEST_STAGE_DEV_OPEN ,
TEST_STAGE_CHECK_DEV_DESC ,
TEST_STAGE_CHECK_CONFIG_DESC ,
TEST_STAGE_CHECK_STR_DESC ,
TEST_STAGE_DEV_CLOSE ,
} test_stage_t ;
typedef struct {
test_stage_t cur_stage ;
test_stage_t next_stage ;
uint8_t dev_addr_to_open ;
usb_host_client_handle_t client_hdl ;
usb_device_handle_t dev_hdl ;
} msc_client_obj_t ;
static void msc_client_event_cb ( const usb_host_client_event_msg_t * event_msg , void * arg )
{
msc_client_obj_t * msc_obj = ( msc_client_obj_t * ) arg ;
switch ( event_msg - > event ) {
case USB_HOST_CLIENT_EVENT_NEW_DEV :
TEST_ASSERT_EQUAL ( TEST_STAGE_WAIT_CONN , msc_obj - > cur_stage ) ;
msc_obj - > next_stage = TEST_STAGE_DEV_OPEN ;
msc_obj - > dev_addr_to_open = event_msg - > new_dev . address ;
break ;
default :
abort ( ) ; //Should never occur in this test
break ;
}
}
void msc_client_async_enum_task ( void * arg )
{
msc_client_obj_t msc_obj ;
msc_obj . cur_stage = TEST_STAGE_WAIT_CONN ;
msc_obj . next_stage = TEST_STAGE_WAIT_CONN ;
msc_obj . client_hdl = NULL ;
msc_obj . dev_addr_to_open = 0 ;
msc_obj . dev_hdl = NULL ;
//Register client
usb_host_client_config_t client_config = {
. is_synchronous = false ,
. max_num_event_msg = MSC_ASYNC_CLIENT_MAX_EVENT_MSGS ,
. async = {
. client_event_callback = msc_client_event_cb ,
. callback_arg = ( void * ) & msc_obj ,
} ,
} ;
TEST_ASSERT_EQUAL ( ESP_OK , usb_host_client_register ( & client_config , & msc_obj . client_hdl ) ) ;
//Wait to be started by main thread
ulTaskNotifyTake ( pdTRUE , portMAX_DELAY ) ;
ESP_LOGD ( MSC_CLIENT_TAG , " Starting " ) ;
bool exit_loop = false ;
bool skip_event_handling = false ;
2021-12-14 18:28:04 +08:00
int enum_iter = 0 ;
2021-12-08 19:46:46 +08:00
while ( ! exit_loop ) {
if ( ! skip_event_handling ) {
TEST_ASSERT_EQUAL ( ESP_OK , usb_host_client_handle_events ( msc_obj . client_hdl , portMAX_DELAY ) ) ;
}
skip_event_handling = false ;
if ( msc_obj . cur_stage = = msc_obj . next_stage ) {
continue ;
}
msc_obj . cur_stage = msc_obj . next_stage ;
switch ( msc_obj . cur_stage ) {
2021-12-14 18:28:04 +08:00
case TEST_STAGE_WAIT_CONN : {
//Wait for connection, nothing to do
break ;
}
2021-12-08 19:46:46 +08:00
case TEST_STAGE_DEV_OPEN : {
ESP_LOGD ( MSC_CLIENT_TAG , " Open " ) ;
//Open the device
TEST_ASSERT_EQUAL ( ESP_OK , usb_host_device_open ( msc_obj . client_hdl , msc_obj . dev_addr_to_open , & msc_obj . dev_hdl ) ) ;
msc_obj . next_stage = TEST_STAGE_CHECK_DEV_DESC ;
skip_event_handling = true ; //Need to execute TEST_STAGE_CHECK_DEV_DESC
break ;
}
case TEST_STAGE_CHECK_DEV_DESC : {
//Check the device descriptor
const usb_device_desc_t * device_desc ;
2022-12-13 21:21:22 +08:00
const usb_device_desc_t * device_desc_ref = & mock_msc_scsi_dev_desc ;
2021-12-08 19:46:46 +08:00
TEST_ASSERT_EQUAL ( ESP_OK , usb_host_get_device_descriptor ( msc_obj . dev_hdl , & device_desc ) ) ;
TEST_ASSERT_EQUAL ( device_desc_ref - > bLength , device_desc - > bLength ) ;
2022-12-13 21:21:22 +08:00
TEST_ASSERT_EQUAL_MEMORY_MESSAGE ( device_desc_ref , device_desc , device_desc_ref - > bLength , " Device descriptors do not match. " ) ;
2021-12-08 19:46:46 +08:00
msc_obj . next_stage = TEST_STAGE_CHECK_CONFIG_DESC ;
skip_event_handling = true ; //Need to execute TEST_STAGE_CHECK_CONFIG_DESC
break ;
}
case TEST_STAGE_CHECK_CONFIG_DESC : {
//Check the configuration descriptor
const usb_config_desc_t * config_desc ;
const usb_config_desc_t * config_desc_ref = ( const usb_config_desc_t * ) mock_msc_scsi_config_desc ;
TEST_ASSERT_EQUAL ( ESP_OK , usb_host_get_active_config_descriptor ( msc_obj . dev_hdl , & config_desc ) ) ;
2022-12-13 21:21:22 +08:00
TEST_ASSERT_EQUAL_MESSAGE ( config_desc_ref - > wTotalLength , config_desc - > wTotalLength , " Incorrent length of CFG descriptor " ) ;
TEST_ASSERT_EQUAL_MEMORY_MESSAGE ( config_desc_ref , config_desc , config_desc_ref - > wTotalLength , " Configuration descriptors do not match " ) ;
2021-12-08 19:46:46 +08:00
msc_obj . next_stage = TEST_STAGE_CHECK_STR_DESC ;
skip_event_handling = true ; //Need to execute TEST_STAGE_CHECK_STR_DESC
break ;
}
case TEST_STAGE_CHECK_STR_DESC : {
usb_device_info_t dev_info ;
TEST_ASSERT_EQUAL ( ESP_OK , usb_host_device_info ( msc_obj . dev_hdl , & dev_info ) ) ;
//Check manufacturer string descriptors
const usb_str_desc_t * manu_str_desc_ref = ( const usb_str_desc_t * ) mock_msc_scsi_str_desc_manu ;
const usb_str_desc_t * product_str_desc_ref = ( const usb_str_desc_t * ) mock_msc_scsi_str_desc_prod ;
const usb_str_desc_t * ser_num_str_desc_ref = ( const usb_str_desc_t * ) mock_msc_scsi_str_desc_ser_num ;
TEST_ASSERT_EQUAL ( manu_str_desc_ref - > bLength , dev_info . str_desc_manufacturer - > bLength ) ;
TEST_ASSERT_EQUAL ( product_str_desc_ref - > bLength , dev_info . str_desc_product - > bLength ) ;
TEST_ASSERT_EQUAL ( ser_num_str_desc_ref - > bLength , dev_info . str_desc_serial_num - > bLength ) ;
2022-12-13 21:21:22 +08:00
TEST_ASSERT_EQUAL_MEMORY_MESSAGE ( manu_str_desc_ref , dev_info . str_desc_manufacturer , manu_str_desc_ref - > bLength , " Manufacturer string descriptors do not match. " ) ;
TEST_ASSERT_EQUAL_MEMORY_MESSAGE ( product_str_desc_ref , dev_info . str_desc_product , manu_str_desc_ref - > bLength , " Product string descriptors do not match. " ) ;
//TEST_ASSERT_EQUAL_MEMORY_MESSAGE(ser_num_str_desc_ref, dev_info.str_desc_serial_num , manu_str_desc_ref->bLength, "Serial number string descriptors do not match.");
2021-12-08 19:46:46 +08:00
//Get dev info and compare
msc_obj . next_stage = TEST_STAGE_DEV_CLOSE ;
skip_event_handling = true ; //Need to execute TEST_STAGE_DEV_CLOSE
break ;
}
case TEST_STAGE_DEV_CLOSE : {
ESP_LOGD ( MSC_CLIENT_TAG , " Close " ) ;
TEST_ASSERT_EQUAL ( ESP_OK , usb_host_device_close ( msc_obj . client_hdl , msc_obj . dev_hdl ) ) ;
2021-12-14 18:28:04 +08:00
enum_iter + + ;
if ( enum_iter < TEST_ENUM_ITERATIONS ) {
//Start the next test iteration by disconnecting the device, then going back to TEST_STAGE_WAIT_CONN stage
test_usb_set_phy_state ( false , 0 ) ;
test_usb_set_phy_state ( true , 0 ) ;
msc_obj . next_stage = TEST_STAGE_WAIT_CONN ;
skip_event_handling = true ; //Need to execute TEST_STAGE_WAIT_CONN
} else {
exit_loop = true ;
}
2021-12-08 19:46:46 +08:00
break ;
}
default :
abort ( ) ;
break ;
}
}
//Free transfers and deregister the client
TEST_ASSERT_EQUAL ( ESP_OK , usb_host_client_deregister ( msc_obj . client_hdl ) ) ;
ESP_LOGD ( MSC_CLIENT_TAG , " Done " ) ;
vTaskDelete ( NULL ) ;
}