2021-10-27 03:50:21 -04:00
|
|
|
/*
|
2021-11-17 13:07:53 -05:00
|
|
|
* SPDX-FileCopyrightText: 2015-2022 Espressif Systems (Shanghai) CO LTD
|
2021-10-27 03:50:21 -04:00
|
|
|
*
|
|
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include "freertos/FreeRTOS.h"
|
|
|
|
#include "freertos/task.h"
|
|
|
|
#include "esp_err.h"
|
|
|
|
#include "esp_intr_alloc.h"
|
|
|
|
#include "test_usb_common.h"
|
2022-12-13 08:21:22 -05:00
|
|
|
#include "test_usb_mock_msc.h"
|
2021-10-27 03:50:21 -04:00
|
|
|
#include "msc_client.h"
|
|
|
|
#include "ctrl_client.h"
|
|
|
|
#include "usb/usb_host.h"
|
|
|
|
#include "unity.h"
|
|
|
|
|
|
|
|
// --------------------------------------------------- Test Cases ------------------------------------------------------
|
|
|
|
|
2021-12-14 05:28:04 -05:00
|
|
|
/*
|
|
|
|
Test USB Host Library Sudden Disconnection Handling (no clients)
|
|
|
|
Purpose:
|
|
|
|
- Test that sudden disconnections are handled properly when there are no clients
|
|
|
|
- Test that devices can reconnect after a sudden disconnection has been handled by the USB Host Library
|
2021-10-27 03:50:21 -04:00
|
|
|
|
2021-12-14 05:28:04 -05:00
|
|
|
Procedure:
|
|
|
|
- Install USB Host Library
|
|
|
|
- Wait for connection (and enumeration) to occur
|
|
|
|
- Force a disconnection, then wait for disconnection to be handled (USB_HOST_LIB_EVENT_FLAGS_ALL_FREE)
|
|
|
|
- Allow connections again, and repeat test for multiple iterations
|
|
|
|
*/
|
|
|
|
|
|
|
|
#define TEST_DCONN_NO_CLIENT_ITERATIONS 3
|
2021-10-27 03:50:21 -04:00
|
|
|
|
|
|
|
TEST_CASE("Test USB Host sudden disconnection (no client)", "[usb_host][ignore]")
|
|
|
|
{
|
2021-11-17 13:07:53 -05:00
|
|
|
test_usb_init_phy(); //Initialize the internal USB PHY and USB Controller for testing
|
2021-10-27 03:50:21 -04:00
|
|
|
//Install USB Host Library
|
|
|
|
usb_host_config_t host_config = {
|
2021-11-17 13:07:53 -05:00
|
|
|
.skip_phy_setup = true, //test_usb_init_phy() will already have setup the internal USB PHY for us
|
2021-10-27 03:50:21 -04:00
|
|
|
.intr_flags = ESP_INTR_FLAG_LEVEL1,
|
|
|
|
};
|
|
|
|
ESP_ERROR_CHECK(usb_host_install(&host_config));
|
|
|
|
printf("Installed\n");
|
|
|
|
|
2021-12-14 05:28:04 -05:00
|
|
|
bool connected = false;
|
|
|
|
int dconn_iter = 0;
|
2021-10-27 03:50:21 -04:00
|
|
|
while (1) {
|
|
|
|
//Start handling system events
|
|
|
|
uint32_t event_flags;
|
|
|
|
usb_host_lib_handle_events(portMAX_DELAY, &event_flags);
|
2021-12-14 05:28:04 -05:00
|
|
|
if (!connected) {
|
|
|
|
usb_host_lib_info_t lib_info;
|
|
|
|
TEST_ASSERT_EQUAL(ESP_OK, usb_host_lib_info(&lib_info));
|
|
|
|
if (lib_info.num_devices == 1) {
|
|
|
|
//We've just connected. Trigger a disconnect
|
|
|
|
connected = true;
|
|
|
|
printf("Forcing Sudden Disconnect\n");
|
|
|
|
test_usb_set_phy_state(false, 0);
|
|
|
|
}
|
|
|
|
}
|
2021-10-27 03:50:21 -04:00
|
|
|
if (event_flags & USB_HOST_LIB_EVENT_FLAGS_ALL_FREE) {
|
2021-12-14 05:28:04 -05:00
|
|
|
//The device has disconnected and it's disconnection has been handled
|
|
|
|
printf("Dconn iter %d done\n", dconn_iter);
|
|
|
|
if (++dconn_iter < TEST_DCONN_NO_CLIENT_ITERATIONS) {
|
|
|
|
//Start next iteration
|
|
|
|
connected = false;
|
|
|
|
test_usb_set_phy_state(true, 0);
|
|
|
|
} else {
|
|
|
|
break;
|
|
|
|
}
|
2021-10-27 03:50:21 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//Clean up USB Host
|
|
|
|
ESP_ERROR_CHECK(usb_host_uninstall());
|
2021-11-17 13:07:53 -05:00
|
|
|
test_usb_deinit_phy(); //Deinitialize the internal USB PHY after testing
|
2021-10-27 03:50:21 -04:00
|
|
|
}
|
|
|
|
|
2021-12-14 05:28:04 -05:00
|
|
|
/*
|
|
|
|
Test USB Host Library Sudden Disconnection Handling (with client)
|
|
|
|
Purpose:
|
|
|
|
- Test that sudden disconnections are handled properly when there are registered clients
|
|
|
|
- Test that devices can reconnect after a sudden disconnection has been handled by the USB Host Library
|
|
|
|
|
|
|
|
Procedure:
|
|
|
|
- Install USB Host Library
|
|
|
|
- Create a task to run an MSC client
|
|
|
|
- Start the MSC disconnect client task. It will open the device then force a disconnect for multiple iterations
|
|
|
|
- Wait for USB_HOST_LIB_EVENT_FLAGS_NO_CLIENTS and USB_HOST_LIB_EVENT_FLAGS_ALL_FREE before uninstalling
|
|
|
|
*/
|
|
|
|
|
2021-10-27 03:50:21 -04:00
|
|
|
#define TEST_FORCE_DCONN_NUM_TRANSFERS 3
|
|
|
|
#define TEST_MSC_SCSI_TAG 0xDEADBEEF
|
|
|
|
|
|
|
|
TEST_CASE("Test USB Host sudden disconnection (single client)", "[usb_host][ignore]")
|
|
|
|
{
|
2021-11-17 13:07:53 -05:00
|
|
|
test_usb_init_phy(); //Initialize the internal USB PHY and USB Controller for testing
|
2021-10-27 03:50:21 -04:00
|
|
|
//Install USB Host
|
|
|
|
usb_host_config_t host_config = {
|
2021-11-17 13:07:53 -05:00
|
|
|
.skip_phy_setup = true, //test_usb_init_phy() will already have setup the internal USB PHY for us
|
2021-10-27 03:50:21 -04:00
|
|
|
.intr_flags = ESP_INTR_FLAG_LEVEL1,
|
|
|
|
};
|
|
|
|
ESP_ERROR_CHECK(usb_host_install(&host_config));
|
|
|
|
printf("Installed\n");
|
|
|
|
|
|
|
|
//Create task to run client that communicates with MSC SCSI interface
|
|
|
|
msc_client_test_param_t params = {
|
|
|
|
.num_sectors_to_read = 1, //Unused by disconnect MSC client
|
|
|
|
.num_sectors_per_xfer = TEST_FORCE_DCONN_NUM_TRANSFERS * MOCK_MSC_SCSI_SECTOR_SIZE,
|
|
|
|
.msc_scsi_xfer_tag = TEST_MSC_SCSI_TAG,
|
|
|
|
.idVendor = MOCK_MSC_SCSI_DEV_ID_VENDOR,
|
|
|
|
.idProduct = MOCK_MSC_SCSI_DEV_ID_PRODUCT,
|
|
|
|
};
|
|
|
|
TaskHandle_t task_hdl;
|
|
|
|
xTaskCreatePinnedToCore(msc_client_async_dconn_task, "async", 4096, (void *)¶ms, 2, &task_hdl, 0);
|
|
|
|
//Start the task
|
|
|
|
xTaskNotifyGive(task_hdl);
|
|
|
|
|
|
|
|
bool all_clients_gone = false;
|
|
|
|
bool all_dev_free = false;
|
|
|
|
while (!all_clients_gone || !all_dev_free) {
|
|
|
|
//Start handling system events
|
|
|
|
uint32_t event_flags;
|
|
|
|
usb_host_lib_handle_events(portMAX_DELAY, &event_flags);
|
|
|
|
if (event_flags & USB_HOST_LIB_EVENT_FLAGS_NO_CLIENTS) {
|
|
|
|
printf("No more clients\n");
|
|
|
|
all_clients_gone = true;
|
|
|
|
}
|
|
|
|
if (event_flags & USB_HOST_LIB_EVENT_FLAGS_ALL_FREE) {
|
|
|
|
printf("All device's freed\n");
|
|
|
|
all_dev_free = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//Short delay to allow task to be cleaned up
|
|
|
|
vTaskDelay(10);
|
|
|
|
//Clean up USB Host
|
|
|
|
ESP_ERROR_CHECK(usb_host_uninstall());
|
2021-11-17 13:07:53 -05:00
|
|
|
test_usb_deinit_phy(); //Deinitialize the internal USB PHY after testing
|
2021-10-27 03:50:21 -04:00
|
|
|
}
|
2021-12-08 06:46:46 -05:00
|
|
|
|
2021-12-14 05:28:04 -05:00
|
|
|
/*
|
|
|
|
Test USB Host Library Enumeration
|
|
|
|
Purpose:
|
|
|
|
- Test that the USB Host Library enumerates device correctly
|
|
|
|
|
|
|
|
Procedure:
|
|
|
|
- Install USB Host Library
|
|
|
|
- Create a task to run an MSC client
|
|
|
|
- Start the MSC enumeration client task. It will:
|
|
|
|
- Wait for device connection
|
|
|
|
- Open the device
|
|
|
|
- Check details of the device's enumeration
|
|
|
|
- Disconnect the device, and repeat the steps above for multiple iterations.
|
|
|
|
- Wait for USB_HOST_LIB_EVENT_FLAGS_NO_CLIENTS and USB_HOST_LIB_EVENT_FLAGS_ALL_FREE before uninstalling
|
|
|
|
*/
|
|
|
|
|
|
|
|
#define TEST_ENUM_ITERATIONS 3
|
|
|
|
|
2021-12-08 06:46:46 -05:00
|
|
|
TEST_CASE("Test USB Host enumeration", "[usb_host][ignore]")
|
|
|
|
{
|
2022-12-13 08:21:22 -05:00
|
|
|
mock_msc_scsi_init_reference_descriptors();
|
2021-12-08 06:46:46 -05:00
|
|
|
test_usb_init_phy(); //Initialize the internal USB PHY and USB Controller for testing
|
|
|
|
//Install USB Host
|
|
|
|
usb_host_config_t host_config = {
|
|
|
|
.skip_phy_setup = true, //test_usb_init_phy() will already have setup the internal USB PHY for us
|
|
|
|
.intr_flags = ESP_INTR_FLAG_LEVEL1,
|
|
|
|
};
|
|
|
|
ESP_ERROR_CHECK(usb_host_install(&host_config));
|
|
|
|
printf("Installed\n");
|
|
|
|
|
|
|
|
//Create task to run client that checks the enumeration of the device
|
|
|
|
TaskHandle_t task_hdl;
|
2021-12-14 05:28:04 -05:00
|
|
|
xTaskCreatePinnedToCore(msc_client_async_enum_task, "async", 6144, NULL, 2, &task_hdl, 0);
|
2021-12-08 06:46:46 -05:00
|
|
|
//Start the task
|
|
|
|
xTaskNotifyGive(task_hdl);
|
|
|
|
|
2021-12-14 05:28:04 -05:00
|
|
|
bool all_clients_gone = false;
|
|
|
|
bool all_dev_free = false;
|
|
|
|
while (!all_clients_gone || !all_dev_free) {
|
2021-12-08 06:46:46 -05:00
|
|
|
//Start handling system events
|
|
|
|
uint32_t event_flags;
|
|
|
|
usb_host_lib_handle_events(portMAX_DELAY, &event_flags);
|
|
|
|
if (event_flags & USB_HOST_LIB_EVENT_FLAGS_NO_CLIENTS) {
|
|
|
|
printf("No more clients\n");
|
|
|
|
TEST_ASSERT_EQUAL(ESP_ERR_NOT_FINISHED, usb_host_device_free_all());
|
2021-12-14 05:28:04 -05:00
|
|
|
all_clients_gone = true;
|
2021-12-08 06:46:46 -05:00
|
|
|
}
|
2021-12-14 05:28:04 -05:00
|
|
|
if (all_clients_gone && event_flags & USB_HOST_LIB_EVENT_FLAGS_ALL_FREE) {
|
|
|
|
all_dev_free = true;
|
2021-12-08 06:46:46 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//Short delay to allow task to be cleaned up
|
|
|
|
vTaskDelay(10);
|
|
|
|
//Clean up USB Host
|
|
|
|
ESP_ERROR_CHECK(usb_host_uninstall());
|
|
|
|
test_usb_deinit_phy(); //Deinitialize the internal USB PHY after testing
|
|
|
|
}
|