mirror of
				https://source.denx.de/u-boot/u-boot.git
				synced 2025-11-04 10:21:25 +01:00 
			
		
		
		
	The USB subsystem has a few counters that need to be reset since they are stored in static variables rather than driver-model data. An example is usb_max_devs. Ultimately we should move this data into the USB uclass. For now, make sure that USB is reset after each test, so that the counters go back to zero. Note: this is not a perfect solution: It a USB test fails it will exit immediately and leave USB un-reset. The impact here is that it may cause subsequence test failures in the same run. Signed-off-by: Simon Glass <sjg@chromium.org>
		
			
				
	
	
		
			303 lines
		
	
	
		
			7.8 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			303 lines
		
	
	
		
			7.8 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/*
 | 
						|
 * Copyright (C) 2015 Google, Inc
 | 
						|
 *
 | 
						|
 * SPDX-License-Identifier:	GPL-2.0+
 | 
						|
 */
 | 
						|
 | 
						|
#include <common.h>
 | 
						|
#include <console.h>
 | 
						|
#include <dm.h>
 | 
						|
#include <usb.h>
 | 
						|
#include <asm/io.h>
 | 
						|
#include <asm/state.h>
 | 
						|
#include <asm/test.h>
 | 
						|
#include <dm/device-internal.h>
 | 
						|
#include <dm/test.h>
 | 
						|
#include <dm/uclass-internal.h>
 | 
						|
#include <test/ut.h>
 | 
						|
 | 
						|
DECLARE_GLOBAL_DATA_PTR;
 | 
						|
 | 
						|
/* Test that sandbox USB works correctly */
 | 
						|
static int dm_test_usb_base(struct unit_test_state *uts)
 | 
						|
{
 | 
						|
	struct udevice *bus;
 | 
						|
 | 
						|
	ut_asserteq(-ENODEV, uclass_get_device_by_seq(UCLASS_USB, 0, &bus));
 | 
						|
	ut_assertok(uclass_get_device(UCLASS_USB, 0, &bus));
 | 
						|
	ut_asserteq(-ENODEV, uclass_get_device_by_seq(UCLASS_USB, 2, &bus));
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
DM_TEST(dm_test_usb_base, DM_TESTF_SCAN_PDATA | DM_TESTF_SCAN_FDT);
 | 
						|
 | 
						|
/*
 | 
						|
 * Test that we can use the flash stick. This is more of a functional test. It
 | 
						|
 * covers scanning the bug, setting up a hub and a flash stick and reading
 | 
						|
 * data from the flash stick.
 | 
						|
 */
 | 
						|
static int dm_test_usb_flash(struct unit_test_state *uts)
 | 
						|
{
 | 
						|
	struct udevice *dev;
 | 
						|
	struct blk_desc *dev_desc;
 | 
						|
	char cmp[1024];
 | 
						|
 | 
						|
	state_set_skip_delays(true);
 | 
						|
	ut_assertok(usb_init());
 | 
						|
	ut_assertok(uclass_get_device(UCLASS_MASS_STORAGE, 0, &dev));
 | 
						|
	ut_assertok(blk_get_device_by_str("usb", "0", &dev_desc));
 | 
						|
 | 
						|
	/* Read a few blocks and look for the string we expect */
 | 
						|
	ut_asserteq(512, dev_desc->blksz);
 | 
						|
	memset(cmp, '\0', sizeof(cmp));
 | 
						|
	ut_asserteq(2, blk_dread(dev_desc, 0, 2, cmp));
 | 
						|
	ut_assertok(strcmp(cmp, "this is a test"));
 | 
						|
	ut_assertok(usb_stop());
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
DM_TEST(dm_test_usb_flash, DM_TESTF_SCAN_PDATA | DM_TESTF_SCAN_FDT);
 | 
						|
 | 
						|
/* test that we can handle multiple storage devices */
 | 
						|
static int dm_test_usb_multi(struct unit_test_state *uts)
 | 
						|
{
 | 
						|
	struct udevice *dev;
 | 
						|
 | 
						|
	state_set_skip_delays(true);
 | 
						|
	ut_assertok(usb_init());
 | 
						|
	ut_assertok(uclass_get_device(UCLASS_MASS_STORAGE, 0, &dev));
 | 
						|
	ut_assertok(uclass_get_device(UCLASS_MASS_STORAGE, 1, &dev));
 | 
						|
	ut_assertok(uclass_get_device(UCLASS_MASS_STORAGE, 2, &dev));
 | 
						|
	ut_assertok(usb_stop());
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
DM_TEST(dm_test_usb_multi, DM_TESTF_SCAN_PDATA | DM_TESTF_SCAN_FDT);
 | 
						|
 | 
						|
static int count_usb_devices(void)
 | 
						|
{
 | 
						|
	struct udevice *hub;
 | 
						|
	struct uclass *uc;
 | 
						|
	int count = 0;
 | 
						|
	int ret;
 | 
						|
 | 
						|
	ret = uclass_get(UCLASS_USB_HUB, &uc);
 | 
						|
	if (ret)
 | 
						|
		return ret;
 | 
						|
 | 
						|
	uclass_foreach_dev(hub, uc) {
 | 
						|
		struct udevice *dev;
 | 
						|
 | 
						|
		count++;
 | 
						|
		for (device_find_first_child(hub, &dev);
 | 
						|
		     dev;
 | 
						|
		     device_find_next_child(&dev)) {
 | 
						|
			count++;
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	return count;
 | 
						|
}
 | 
						|
 | 
						|
/* test that we can remove an emulated device and it is then not found */
 | 
						|
static int dm_test_usb_remove(struct unit_test_state *uts)
 | 
						|
{
 | 
						|
	struct udevice *dev, *emul;
 | 
						|
 | 
						|
	/* Scan and check that all devices are present */
 | 
						|
	state_set_skip_delays(true);
 | 
						|
	ut_assertok(usb_init());
 | 
						|
	ut_assertok(uclass_get_device(UCLASS_MASS_STORAGE, 0, &dev));
 | 
						|
	ut_assertok(uclass_get_device(UCLASS_MASS_STORAGE, 1, &dev));
 | 
						|
	ut_assertok(uclass_get_device(UCLASS_MASS_STORAGE, 2, &dev));
 | 
						|
	ut_asserteq(6, count_usb_devices());
 | 
						|
	ut_assertok(usb_stop());
 | 
						|
	ut_asserteq(6, count_usb_devices());
 | 
						|
 | 
						|
	/* Remove the second emulation device */
 | 
						|
	ut_assertok(uclass_find_device_by_name(UCLASS_USB_EMUL, "flash-stick@1",
 | 
						|
					       &dev));
 | 
						|
	ut_assertok(device_unbind(dev));
 | 
						|
 | 
						|
	/* Rescan - only the first and third should be present */
 | 
						|
	ut_assertok(usb_init());
 | 
						|
	ut_assertok(uclass_get_device(UCLASS_MASS_STORAGE, 0, &dev));
 | 
						|
	ut_assertok(usb_emul_find_for_dev(dev, &emul));
 | 
						|
	ut_asserteq_str("flash-stick@0", emul->name);
 | 
						|
	ut_assertok(uclass_get_device(UCLASS_MASS_STORAGE, 1, &dev));
 | 
						|
	ut_assertok(usb_emul_find_for_dev(dev, &emul));
 | 
						|
	ut_asserteq_str("flash-stick@2", emul->name);
 | 
						|
 | 
						|
	ut_asserteq(-ENODEV, uclass_get_device(UCLASS_MASS_STORAGE, 2, &dev));
 | 
						|
 | 
						|
	ut_asserteq(5, count_usb_devices());
 | 
						|
	ut_assertok(usb_stop());
 | 
						|
	ut_asserteq(5, count_usb_devices());
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
DM_TEST(dm_test_usb_remove, DM_TESTF_SCAN_PDATA | DM_TESTF_SCAN_FDT);
 | 
						|
 | 
						|
const char usb_tree_base[] =
 | 
						|
"  1  Hub (12 Mb/s, 100mA)\n"
 | 
						|
"  |  sandbox hub 2345\n"
 | 
						|
"  |\n"
 | 
						|
"  |\b+-2  Mass Storage (12 Mb/s, 100mA)\n"
 | 
						|
"  |    sandbox flash flash-stick@0\n"
 | 
						|
"  |  \n"
 | 
						|
"  |\b+-3  Mass Storage (12 Mb/s, 100mA)\n"
 | 
						|
"  |    sandbox flash flash-stick@1\n"
 | 
						|
"  |  \n"
 | 
						|
"  |\b+-4  Mass Storage (12 Mb/s, 100mA)\n"
 | 
						|
"  |    sandbox flash flash-stick@2\n"
 | 
						|
"  |  \n"
 | 
						|
"  |\b+-5  Human Interface (12 Mb/s, 100mA)\n"
 | 
						|
"       sandbox keyboard keyb@3\n"
 | 
						|
"     \n";
 | 
						|
 | 
						|
/* test that the 'usb tree' command output looks correct */
 | 
						|
static int dm_test_usb_tree(struct unit_test_state *uts)
 | 
						|
{
 | 
						|
	char *data;
 | 
						|
	int len;
 | 
						|
 | 
						|
	state_set_skip_delays(true);
 | 
						|
	ut_assertok(usb_init());
 | 
						|
	console_record_reset_enable();
 | 
						|
	usb_show_tree();
 | 
						|
	len = membuff_getraw(&gd->console_out, -1, true, &data);
 | 
						|
	if (len)
 | 
						|
		data[len] = '\0';
 | 
						|
	ut_asserteq_str(usb_tree_base, data);
 | 
						|
	ut_assertok(usb_stop());
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
DM_TEST(dm_test_usb_tree, DM_TESTF_SCAN_PDATA | DM_TESTF_SCAN_FDT);
 | 
						|
 | 
						|
const char usb_tree_remove[] =
 | 
						|
"  1  Hub (12 Mb/s, 100mA)\n"
 | 
						|
"  |  sandbox hub 2345\n"
 | 
						|
"  |\n"
 | 
						|
"  |\b+-2  Mass Storage (12 Mb/s, 100mA)\n"
 | 
						|
"  |    sandbox flash flash-stick@0\n"
 | 
						|
"  |  \n"
 | 
						|
"  |\b+-3  Mass Storage (12 Mb/s, 100mA)\n"
 | 
						|
"  |    sandbox flash flash-stick@2\n"
 | 
						|
"  |  \n"
 | 
						|
"  |\b+-4  Human Interface (12 Mb/s, 100mA)\n"
 | 
						|
"       sandbox keyboard keyb@3\n"
 | 
						|
"     \n";
 | 
						|
 | 
						|
/*
 | 
						|
 * test that the 'usb tree' command output looks correct when we remove a
 | 
						|
 * device
 | 
						|
 */
 | 
						|
static int dm_test_usb_tree_remove(struct unit_test_state *uts)
 | 
						|
{
 | 
						|
	struct udevice *dev;
 | 
						|
	char *data;
 | 
						|
	int len;
 | 
						|
 | 
						|
	/* Remove the second emulation device */
 | 
						|
	ut_assertok(uclass_find_device_by_name(UCLASS_USB_EMUL, "flash-stick@1",
 | 
						|
					       &dev));
 | 
						|
	ut_assertok(device_unbind(dev));
 | 
						|
 | 
						|
	state_set_skip_delays(true);
 | 
						|
	ut_assertok(usb_init());
 | 
						|
	console_record_reset_enable();
 | 
						|
	usb_show_tree();
 | 
						|
	len = membuff_getraw(&gd->console_out, -1, true, &data);
 | 
						|
	if (len)
 | 
						|
		data[len] = '\0';
 | 
						|
	ut_asserteq_str(usb_tree_remove, data);
 | 
						|
	ut_assertok(usb_stop());
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
DM_TEST(dm_test_usb_tree_remove, DM_TESTF_SCAN_PDATA | DM_TESTF_SCAN_FDT);
 | 
						|
 | 
						|
const char usb_tree_reorder[] =
 | 
						|
"  1  Hub (12 Mb/s, 100mA)\n"
 | 
						|
"  |  sandbox hub 2345\n"
 | 
						|
"  |\n"
 | 
						|
"  |\b+-2  Mass Storage (12 Mb/s, 100mA)\n"
 | 
						|
"  |    sandbox flash flash-stick@0\n"
 | 
						|
"  |  \n"
 | 
						|
"  |\b+-3  Mass Storage (12 Mb/s, 100mA)\n"
 | 
						|
"  |    sandbox flash flash-stick@2\n"
 | 
						|
"  |  \n"
 | 
						|
"  |\b+-4  Human Interface (12 Mb/s, 100mA)\n"
 | 
						|
"  |    sandbox keyboard keyb@3\n"
 | 
						|
"  |  \n"
 | 
						|
"  |\b+-5  Mass Storage (12 Mb/s, 100mA)\n"
 | 
						|
"       sandbox flash flash-stick@1\n"
 | 
						|
"     \n";
 | 
						|
 | 
						|
/*
 | 
						|
 * test that the 'usb tree' command output looks correct when we reorder two
 | 
						|
 * devices.
 | 
						|
 */
 | 
						|
static int dm_test_usb_tree_reorder(struct unit_test_state *uts)
 | 
						|
{
 | 
						|
	struct udevice *dev, *parent;
 | 
						|
	char *data;
 | 
						|
	int len;
 | 
						|
 | 
						|
	/* Remove the second emulation device */
 | 
						|
	ut_assertok(uclass_find_device_by_name(UCLASS_USB_EMUL, "flash-stick@1",
 | 
						|
					       &dev));
 | 
						|
	parent = dev->parent;
 | 
						|
 | 
						|
	/* Reorder the devices in the parent list and uclass list */
 | 
						|
	list_del(&dev->sibling_node);
 | 
						|
	list_add_tail(&dev->sibling_node, &parent->child_head);
 | 
						|
 | 
						|
	list_del(&dev->uclass_node);
 | 
						|
	list_add_tail(&dev->uclass_node, &dev->uclass->dev_head);
 | 
						|
 | 
						|
	state_set_skip_delays(true);
 | 
						|
	ut_assertok(usb_init());
 | 
						|
	console_record_reset_enable();
 | 
						|
	usb_show_tree();
 | 
						|
	len = membuff_getraw(&gd->console_out, -1, true, &data);
 | 
						|
	if (len)
 | 
						|
		data[len] = '\0';
 | 
						|
	ut_asserteq_str(usb_tree_reorder, data);
 | 
						|
	ut_assertok(usb_stop());
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
DM_TEST(dm_test_usb_tree_reorder, DM_TESTF_SCAN_PDATA | DM_TESTF_SCAN_FDT);
 | 
						|
 | 
						|
static int dm_test_usb_keyb(struct unit_test_state *uts)
 | 
						|
{
 | 
						|
	struct udevice *dev;
 | 
						|
 | 
						|
	state_set_skip_delays(true);
 | 
						|
	ut_assertok(usb_init());
 | 
						|
 | 
						|
	/* Initially there should be no characters */
 | 
						|
	ut_asserteq(0, tstc());
 | 
						|
 | 
						|
	ut_assertok(uclass_get_device_by_name(UCLASS_USB_EMUL, "keyb",
 | 
						|
					      &dev));
 | 
						|
 | 
						|
	/*
 | 
						|
	 * Add a string to the USB keyboard buffer - it should appear in
 | 
						|
	 * stdin
 | 
						|
	 */
 | 
						|
	ut_assertok(sandbox_usb_keyb_add_string(dev, "ab"));
 | 
						|
	ut_asserteq(1, tstc());
 | 
						|
	ut_asserteq('a', getc());
 | 
						|
	ut_asserteq(1, tstc());
 | 
						|
	ut_asserteq('b', getc());
 | 
						|
	ut_asserteq(0, tstc());
 | 
						|
 | 
						|
	ut_assertok(usb_stop());
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
DM_TEST(dm_test_usb_keyb, DM_TESTF_SCAN_PDATA | DM_TESTF_SCAN_FDT);
 |