blob: b46ae6060207b53022b0679949eaf59687db7653 [file] [log] [blame]
Simon Glass31680482015-03-25 12:23:05 -06001/*
2 * Copyright (C) 2015 Google, Inc
3 *
4 * SPDX-License-Identifier: GPL-2.0+
5 */
6
7#include <common.h>
Simon Glassa73bda42015-11-08 23:47:45 -07008#include <console.h>
Simon Glass31680482015-03-25 12:23:05 -06009#include <dm.h>
10#include <usb.h>
11#include <asm/io.h>
Simon Glass2d7c4992015-11-08 23:47:44 -070012#include <asm/state.h>
Simon Glassc0ccc722015-11-08 23:48:08 -070013#include <asm/test.h>
Simon Glass2d7c4992015-11-08 23:47:44 -070014#include <dm/device-internal.h>
Simon Glass31680482015-03-25 12:23:05 -060015#include <dm/test.h>
Simon Glass4700fe52015-11-08 23:48:01 -070016#include <dm/uclass-internal.h>
Joe Hershberger3a77be52015-05-20 14:27:27 -050017#include <test/ut.h>
Simon Glass31680482015-03-25 12:23:05 -060018
Simon Glass4700fe52015-11-08 23:48:01 -070019DECLARE_GLOBAL_DATA_PTR;
20
Simon Glass31680482015-03-25 12:23:05 -060021/* Test that sandbox USB works correctly */
Joe Hershberger3a77be52015-05-20 14:27:27 -050022static int dm_test_usb_base(struct unit_test_state *uts)
Simon Glass31680482015-03-25 12:23:05 -060023{
24 struct udevice *bus;
25
26 ut_asserteq(-ENODEV, uclass_get_device_by_seq(UCLASS_USB, 0, &bus));
27 ut_assertok(uclass_get_device(UCLASS_USB, 0, &bus));
28 ut_asserteq(-ENODEV, uclass_get_device_by_seq(UCLASS_USB, 2, &bus));
29
30 return 0;
31}
32DM_TEST(dm_test_usb_base, DM_TESTF_SCAN_PDATA | DM_TESTF_SCAN_FDT);
33
34/*
35 * Test that we can use the flash stick. This is more of a functional test. It
36 * covers scanning the bug, setting up a hub and a flash stick and reading
37 * data from the flash stick.
38 */
Joe Hershberger3a77be52015-05-20 14:27:27 -050039static int dm_test_usb_flash(struct unit_test_state *uts)
Simon Glass31680482015-03-25 12:23:05 -060040{
41 struct udevice *dev;
Simon Glasse3394752016-02-29 15:25:34 -070042 struct blk_desc *dev_desc;
Simon Glass31680482015-03-25 12:23:05 -060043 char cmp[1024];
44
Simon Glass2d7c4992015-11-08 23:47:44 -070045 state_set_skip_delays(true);
Simon Glass31680482015-03-25 12:23:05 -060046 ut_assertok(usb_init());
47 ut_assertok(uclass_get_device(UCLASS_MASS_STORAGE, 0, &dev));
Simon Glasse6649a62016-02-29 15:25:43 -070048 ut_assertok(blk_get_device_by_str("usb", "0", &dev_desc));
Simon Glass31680482015-03-25 12:23:05 -060049
50 /* Read a few blocks and look for the string we expect */
51 ut_asserteq(512, dev_desc->blksz);
52 memset(cmp, '\0', sizeof(cmp));
Simon Glass2ee8ada2016-02-29 15:25:52 -070053 ut_asserteq(2, blk_dread(dev_desc, 0, 2, cmp));
Simon Glass31680482015-03-25 12:23:05 -060054 ut_assertok(strcmp(cmp, "this is a test"));
Simon Glass582b76e2016-02-29 15:26:02 -070055 ut_assertok(usb_stop());
Simon Glass31680482015-03-25 12:23:05 -060056
57 return 0;
58}
59DM_TEST(dm_test_usb_flash, DM_TESTF_SCAN_PDATA | DM_TESTF_SCAN_FDT);
Simon Glass4700fe52015-11-08 23:48:01 -070060
61/* test that we can handle multiple storage devices */
62static int dm_test_usb_multi(struct unit_test_state *uts)
63{
64 struct udevice *dev;
65
66 state_set_skip_delays(true);
67 ut_assertok(usb_init());
68 ut_assertok(uclass_get_device(UCLASS_MASS_STORAGE, 0, &dev));
69 ut_assertok(uclass_get_device(UCLASS_MASS_STORAGE, 1, &dev));
70 ut_assertok(uclass_get_device(UCLASS_MASS_STORAGE, 2, &dev));
Simon Glass582b76e2016-02-29 15:26:02 -070071 ut_assertok(usb_stop());
Simon Glass4700fe52015-11-08 23:48:01 -070072
73 return 0;
74}
75DM_TEST(dm_test_usb_multi, DM_TESTF_SCAN_PDATA | DM_TESTF_SCAN_FDT);
76
77static int count_usb_devices(void)
78{
79 struct udevice *hub;
80 struct uclass *uc;
81 int count = 0;
82 int ret;
83
84 ret = uclass_get(UCLASS_USB_HUB, &uc);
85 if (ret)
86 return ret;
87
88 uclass_foreach_dev(hub, uc) {
89 struct udevice *dev;
90
91 count++;
92 for (device_find_first_child(hub, &dev);
93 dev;
94 device_find_next_child(&dev)) {
95 count++;
96 }
97 }
98
99 return count;
100}
101
102/* test that we can remove an emulated device and it is then not found */
103static int dm_test_usb_remove(struct unit_test_state *uts)
104{
105 struct udevice *dev, *emul;
106
107 /* Scan and check that all devices are present */
108 state_set_skip_delays(true);
109 ut_assertok(usb_init());
110 ut_assertok(uclass_get_device(UCLASS_MASS_STORAGE, 0, &dev));
111 ut_assertok(uclass_get_device(UCLASS_MASS_STORAGE, 1, &dev));
112 ut_assertok(uclass_get_device(UCLASS_MASS_STORAGE, 2, &dev));
Simon Glassf0bfd6d2016-01-07 10:23:42 -0700113 ut_asserteq(6, count_usb_devices());
Simon Glass4700fe52015-11-08 23:48:01 -0700114 ut_assertok(usb_stop());
Simon Glassf0bfd6d2016-01-07 10:23:42 -0700115 ut_asserteq(6, count_usb_devices());
Simon Glass4700fe52015-11-08 23:48:01 -0700116
117 /* Remove the second emulation device */
118 ut_assertok(uclass_find_device_by_name(UCLASS_USB_EMUL, "flash-stick@1",
119 &dev));
120 ut_assertok(device_unbind(dev));
121
122 /* Rescan - only the first and third should be present */
123 ut_assertok(usb_init());
124 ut_assertok(uclass_get_device(UCLASS_MASS_STORAGE, 0, &dev));
125 ut_assertok(usb_emul_find_for_dev(dev, &emul));
126 ut_asserteq_str("flash-stick@0", emul->name);
127 ut_assertok(uclass_get_device(UCLASS_MASS_STORAGE, 1, &dev));
128 ut_assertok(usb_emul_find_for_dev(dev, &emul));
129 ut_asserteq_str("flash-stick@2", emul->name);
130
131 ut_asserteq(-ENODEV, uclass_get_device(UCLASS_MASS_STORAGE, 2, &dev));
132
Simon Glassf0bfd6d2016-01-07 10:23:42 -0700133 ut_asserteq(5, count_usb_devices());
Simon Glass4700fe52015-11-08 23:48:01 -0700134 ut_assertok(usb_stop());
Simon Glassf0bfd6d2016-01-07 10:23:42 -0700135 ut_asserteq(5, count_usb_devices());
Simon Glass4700fe52015-11-08 23:48:01 -0700136
137 return 0;
138}
139DM_TEST(dm_test_usb_remove, DM_TESTF_SCAN_PDATA | DM_TESTF_SCAN_FDT);
140
141const char usb_tree_base[] =
142" 1 Hub (12 Mb/s, 100mA)\n"
143" | sandbox hub 2345\n"
144" |\n"
145" |\b+-2 Mass Storage (12 Mb/s, 100mA)\n"
146" | sandbox flash flash-stick@0\n"
147" | \n"
148" |\b+-3 Mass Storage (12 Mb/s, 100mA)\n"
149" | sandbox flash flash-stick@1\n"
150" | \n"
151" |\b+-4 Mass Storage (12 Mb/s, 100mA)\n"
Simon Glassf0bfd6d2016-01-07 10:23:42 -0700152" | sandbox flash flash-stick@2\n"
153" | \n"
154" |\b+-5 Human Interface (12 Mb/s, 100mA)\n"
155" sandbox keyboard keyb@3\n"
Simon Glass4700fe52015-11-08 23:48:01 -0700156" \n";
157
158/* test that the 'usb tree' command output looks correct */
159static int dm_test_usb_tree(struct unit_test_state *uts)
160{
161 char *data;
162 int len;
163
164 state_set_skip_delays(true);
165 ut_assertok(usb_init());
166 console_record_reset_enable();
167 usb_show_tree();
168 len = membuff_getraw(&gd->console_out, -1, true, &data);
169 if (len)
170 data[len] = '\0';
171 ut_asserteq_str(usb_tree_base, data);
172 ut_assertok(usb_stop());
173
174 return 0;
175}
176DM_TEST(dm_test_usb_tree, DM_TESTF_SCAN_PDATA | DM_TESTF_SCAN_FDT);
177
178const char usb_tree_remove[] =
179" 1 Hub (12 Mb/s, 100mA)\n"
180" | sandbox hub 2345\n"
181" |\n"
182" |\b+-2 Mass Storage (12 Mb/s, 100mA)\n"
183" | sandbox flash flash-stick@0\n"
184" | \n"
185" |\b+-3 Mass Storage (12 Mb/s, 100mA)\n"
Simon Glassf0bfd6d2016-01-07 10:23:42 -0700186" | sandbox flash flash-stick@2\n"
187" | \n"
188" |\b+-4 Human Interface (12 Mb/s, 100mA)\n"
189" sandbox keyboard keyb@3\n"
Simon Glass4700fe52015-11-08 23:48:01 -0700190" \n";
191
192/*
193 * test that the 'usb tree' command output looks correct when we remove a
194 * device
195 */
196static int dm_test_usb_tree_remove(struct unit_test_state *uts)
197{
198 struct udevice *dev;
199 char *data;
200 int len;
201
202 /* Remove the second emulation device */
203 ut_assertok(uclass_find_device_by_name(UCLASS_USB_EMUL, "flash-stick@1",
204 &dev));
205 ut_assertok(device_unbind(dev));
206
207 state_set_skip_delays(true);
208 ut_assertok(usb_init());
209 console_record_reset_enable();
210 usb_show_tree();
211 len = membuff_getraw(&gd->console_out, -1, true, &data);
212 if (len)
213 data[len] = '\0';
214 ut_asserteq_str(usb_tree_remove, data);
215 ut_assertok(usb_stop());
216
217 return 0;
218}
219DM_TEST(dm_test_usb_tree_remove, DM_TESTF_SCAN_PDATA | DM_TESTF_SCAN_FDT);
Simon Glass846bb372015-11-08 23:48:02 -0700220
221const char usb_tree_reorder[] =
222" 1 Hub (12 Mb/s, 100mA)\n"
223" | sandbox hub 2345\n"
224" |\n"
225" |\b+-2 Mass Storage (12 Mb/s, 100mA)\n"
226" | sandbox flash flash-stick@0\n"
227" | \n"
228" |\b+-3 Mass Storage (12 Mb/s, 100mA)\n"
229" | sandbox flash flash-stick@2\n"
230" | \n"
Simon Glassf0bfd6d2016-01-07 10:23:42 -0700231" |\b+-4 Human Interface (12 Mb/s, 100mA)\n"
232" | sandbox keyboard keyb@3\n"
233" | \n"
234" |\b+-5 Mass Storage (12 Mb/s, 100mA)\n"
Simon Glass846bb372015-11-08 23:48:02 -0700235" sandbox flash flash-stick@1\n"
236" \n";
237
238/*
239 * test that the 'usb tree' command output looks correct when we reorder two
240 * devices.
241 */
242static int dm_test_usb_tree_reorder(struct unit_test_state *uts)
243{
244 struct udevice *dev, *parent;
245 char *data;
246 int len;
247
248 /* Remove the second emulation device */
249 ut_assertok(uclass_find_device_by_name(UCLASS_USB_EMUL, "flash-stick@1",
250 &dev));
251 parent = dev->parent;
252
253 /* Reorder the devices in the parent list and uclass list */
254 list_del(&dev->sibling_node);
255 list_add_tail(&dev->sibling_node, &parent->child_head);
256
257 list_del(&dev->uclass_node);
258 list_add_tail(&dev->uclass_node, &dev->uclass->dev_head);
259
260 state_set_skip_delays(true);
261 ut_assertok(usb_init());
262 console_record_reset_enable();
263 usb_show_tree();
264 len = membuff_getraw(&gd->console_out, -1, true, &data);
265 if (len)
266 data[len] = '\0';
267 ut_asserteq_str(usb_tree_reorder, data);
268 ut_assertok(usb_stop());
269
270 return 0;
271}
272DM_TEST(dm_test_usb_tree_reorder, DM_TESTF_SCAN_PDATA | DM_TESTF_SCAN_FDT);
Simon Glassc0ccc722015-11-08 23:48:08 -0700273
274static int dm_test_usb_keyb(struct unit_test_state *uts)
275{
276 struct udevice *dev;
277
278 state_set_skip_delays(true);
279 ut_assertok(usb_init());
280
281 /* Initially there should be no characters */
282 ut_asserteq(0, tstc());
283
284 ut_assertok(uclass_get_device_by_name(UCLASS_USB_EMUL, "keyb",
285 &dev));
286
287 /*
288 * Add a string to the USB keyboard buffer - it should appear in
289 * stdin
290 */
291 ut_assertok(sandbox_usb_keyb_add_string(dev, "ab"));
292 ut_asserteq(1, tstc());
293 ut_asserteq('a', getc());
294 ut_asserteq(1, tstc());
295 ut_asserteq('b', getc());
296 ut_asserteq(0, tstc());
297
298 ut_assertok(usb_stop());
299
300 return 0;
301}
302DM_TEST(dm_test_usb_keyb, DM_TESTF_SCAN_PDATA | DM_TESTF_SCAN_FDT);