1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33#include <stdio.h>
34#include <string.h>
35#include <ftw.h>
36#include <stdlib.h>
37#include <pthread.h>
38#include <unistd.h>
39#include <errno.h>
40#include <limits.h>
41
42#include <sys/types.h>
43#include <sys/stat.h>
44#include <fcntl.h>
45
46#include <sys/ioctl.h>
47#include <linux/usbdevice_fs.h>
48
49
50
51#define TEST_CASES 30
52
53
54
55struct usbtest_param {
56
57 unsigned test_num;
58 unsigned iterations;
59 unsigned length;
60 unsigned vary;
61 unsigned sglen;
62
63
64 struct timeval duration;
65};
66#define USBTEST_REQUEST _IOWR('U', 100, struct usbtest_param)
67
68
69
70
71
72#define USB_DT_DEVICE 0x01
73#define USB_DT_INTERFACE 0x04
74
75#define USB_CLASS_PER_INTERFACE 0
76#define USB_CLASS_VENDOR_SPEC 0xff
77
78
79struct usb_device_descriptor {
80 __u8 bLength;
81 __u8 bDescriptorType;
82 __u16 bcdUSB;
83 __u8 bDeviceClass;
84 __u8 bDeviceSubClass;
85 __u8 bDeviceProtocol;
86 __u8 bMaxPacketSize0;
87 __u16 idVendor;
88 __u16 idProduct;
89 __u16 bcdDevice;
90 __u8 iManufacturer;
91 __u8 iProduct;
92 __u8 iSerialNumber;
93 __u8 bNumConfigurations;
94} __attribute__ ((packed));
95
96struct usb_interface_descriptor {
97 __u8 bLength;
98 __u8 bDescriptorType;
99
100 __u8 bInterfaceNumber;
101 __u8 bAlternateSetting;
102 __u8 bNumEndpoints;
103 __u8 bInterfaceClass;
104 __u8 bInterfaceSubClass;
105 __u8 bInterfaceProtocol;
106 __u8 iInterface;
107} __attribute__ ((packed));
108
109enum usb_device_speed {
110 USB_SPEED_UNKNOWN = 0,
111 USB_SPEED_LOW, USB_SPEED_FULL,
112 USB_SPEED_HIGH
113};
114
115
116
117static char *speed (enum usb_device_speed s)
118{
119 switch (s) {
120 case USB_SPEED_UNKNOWN: return "unknown";
121 case USB_SPEED_LOW: return "low";
122 case USB_SPEED_FULL: return "full";
123 case USB_SPEED_HIGH: return "high";
124 default: return "??";
125 }
126}
127
128struct testdev {
129 struct testdev *next;
130 char *name;
131 pthread_t thread;
132 enum usb_device_speed speed;
133 unsigned ifnum : 8;
134 unsigned forever : 1;
135 int test;
136
137 struct usbtest_param param;
138};
139static struct testdev *testdevs;
140
141static int testdev_ffs_ifnum(FILE *fd)
142{
143 union {
144 char buf[255];
145 struct usb_interface_descriptor intf;
146 } u;
147
148 for (;;) {
149 if (fread(u.buf, 1, 1, fd) != 1)
150 return -1;
151 if (fread(u.buf + 1, (unsigned char)u.buf[0] - 1, 1, fd) != 1)
152 return -1;
153
154 if (u.intf.bLength == sizeof u.intf
155 && u.intf.bDescriptorType == USB_DT_INTERFACE
156 && u.intf.bNumEndpoints == 2
157 && u.intf.bInterfaceClass == USB_CLASS_VENDOR_SPEC
158 && u.intf.bInterfaceSubClass == 0
159 && u.intf.bInterfaceProtocol == 0)
160 return (unsigned char)u.intf.bInterfaceNumber;
161 }
162}
163
164static int testdev_ifnum(FILE *fd)
165{
166 struct usb_device_descriptor dev;
167
168 if (fread(&dev, sizeof dev, 1, fd) != 1)
169 return -1;
170
171 if (dev.bLength != sizeof dev || dev.bDescriptorType != USB_DT_DEVICE)
172 return -1;
173
174
175 if (dev.idVendor == 0x0547 && dev.idProduct == 0x1002)
176 return 0;
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191 if (dev.idVendor == 0x0547 && dev.idProduct == 0x2235)
192 return 0;
193
194
195 if (dev.idVendor == 0x04b4 && dev.idProduct == 0x8613)
196 return 0;
197
198
199 if (dev.idVendor == 0x0547 && dev.idProduct == 0x0080)
200 return 0;
201
202
203 if (dev.idVendor == 0x06cd && dev.idProduct == 0x010b)
204 return 0;
205
206
207
208
209 if (dev.idVendor == 0x0525 && dev.idProduct == 0xa4a0)
210 return 0;
211
212
213 if (dev.idVendor == 0x0525 && dev.idProduct == 0xa4a4)
214 return testdev_ffs_ifnum(fd);
215
216
217
218 if (dev.idVendor == 0x0525 && dev.idProduct == 0xa4a3)
219 return 0;
220
221
222
223 if (dev.idVendor == 0xfff0 && dev.idProduct == 0xfff0)
224 return 0;
225
226
227
228
229 if (dev.idVendor == 0x0b62 && dev.idProduct == 0x0059)
230 return 0;
231
232
233
234
235
236
237
238 if (dev.idVendor == 0x0525 && dev.idProduct == 0xa4ac
239 && (dev.bDeviceClass == USB_CLASS_PER_INTERFACE
240 || dev.bDeviceClass == USB_CLASS_VENDOR_SPEC))
241 return testdev_ffs_ifnum(fd);
242
243 return -1;
244}
245
246static int find_testdev(const char *name, const struct stat *sb, int flag)
247{
248 FILE *fd;
249 int ifnum;
250 struct testdev *entry;
251
252 (void)sb;
253
254 if (flag != FTW_F)
255 return 0;
256
257 if (strrchr(name, '/')[1] == 'd')
258 return 0;
259
260 fd = fopen(name, "rb");
261 if (!fd) {
262 perror(name);
263 return 0;
264 }
265
266 ifnum = testdev_ifnum(fd);
267 fclose(fd);
268 if (ifnum < 0)
269 return 0;
270
271 entry = calloc(1, sizeof *entry);
272 if (!entry)
273 goto nomem;
274
275 entry->name = strdup(name);
276 if (!entry->name) {
277 free(entry);
278nomem:
279 perror("malloc");
280 return 0;
281 }
282
283 entry->ifnum = ifnum;
284
285
286
287
288 fprintf(stderr, "%s speed\t%s\t%u\n",
289 speed(entry->speed), entry->name, entry->ifnum);
290
291 entry->next = testdevs;
292 testdevs = entry;
293 return 0;
294}
295
296static int
297usbdev_ioctl (int fd, int ifno, unsigned request, void *param)
298{
299 struct usbdevfs_ioctl wrapper;
300
301 wrapper.ifno = ifno;
302 wrapper.ioctl_code = request;
303 wrapper.data = param;
304
305 return ioctl (fd, USBDEVFS_IOCTL, &wrapper);
306}
307
308static void *handle_testdev (void *arg)
309{
310 struct testdev *dev = arg;
311 int fd, i;
312 int status;
313
314 if ((fd = open (dev->name, O_RDWR)) < 0) {
315 perror ("can't open dev file r/w");
316 return 0;
317 }
318
319restart:
320 for (i = 0; i < TEST_CASES; i++) {
321 if (dev->test != -1 && dev->test != i)
322 continue;
323 dev->param.test_num = i;
324
325 status = usbdev_ioctl (fd, dev->ifnum,
326 USBTEST_REQUEST, &dev->param);
327 if (status < 0 && errno == EOPNOTSUPP)
328 continue;
329
330
331
332
333 if (status < 0) {
334 char buf [80];
335 int err = errno;
336
337 if (strerror_r (errno, buf, sizeof buf)) {
338 snprintf (buf, sizeof buf, "error %d", err);
339 errno = err;
340 }
341 printf ("%s test %d --> %d (%s)\n",
342 dev->name, i, errno, buf);
343 } else
344 printf ("%s test %d, %4d.%.06d secs\n", dev->name, i,
345 (int) dev->param.duration.tv_sec,
346 (int) dev->param.duration.tv_usec);
347
348 fflush (stdout);
349 }
350 if (dev->forever)
351 goto restart;
352
353 close (fd);
354 return arg;
355}
356
357static const char *usbfs_dir_find(void)
358{
359 static char usbfs_path_0[] = "/dev/usb/devices";
360 static char usbfs_path_1[] = "/proc/bus/usb/devices";
361 static char udev_usb_path[] = "/dev/bus/usb";
362
363 static char *const usbfs_paths[] = {
364 usbfs_path_0, usbfs_path_1
365 };
366
367 static char *const *
368 end = usbfs_paths + sizeof usbfs_paths / sizeof *usbfs_paths;
369
370 char *const *it = usbfs_paths;
371 do {
372 int fd = open(*it, O_RDONLY);
373 close(fd);
374 if (fd >= 0) {
375 strrchr(*it, '/')[0] = '\0';
376 return *it;
377 }
378 } while (++it != end);
379
380
381 if (access(udev_usb_path, F_OK) == 0)
382 return udev_usb_path;
383
384 return NULL;
385}
386
387static int parse_num(unsigned *num, const char *str)
388{
389 unsigned long val;
390 char *end;
391
392 errno = 0;
393 val = strtoul(str, &end, 0);
394 if (errno || *end || val > UINT_MAX)
395 return -1;
396 *num = val;
397 return 0;
398}
399
400int main (int argc, char **argv)
401{
402
403 int c;
404 struct testdev *entry;
405 char *device;
406 const char *usbfs_dir = NULL;
407 int all = 0, forever = 0, not = 0;
408 int test = -1 ;
409 struct usbtest_param param;
410
411
412
413
414
415
416
417
418
419
420 param.iterations = 1000;
421 param.length = 512;
422 param.vary = 512;
423 param.sglen = 32;
424
425
426 device = getenv ("DEVICE");
427
428 while ((c = getopt (argc, argv, "D:aA:c:g:hlns:t:v:")) != EOF)
429 switch (c) {
430 case 'D':
431 device = optarg;
432 continue;
433 case 'A':
434 usbfs_dir = optarg;
435
436 case 'a':
437 device = NULL;
438 all = 1;
439 continue;
440 case 'c':
441 if (parse_num(¶m.iterations, optarg))
442 goto usage;
443 continue;
444 case 'g':
445 if (parse_num(¶m.sglen, optarg))
446 goto usage;
447 continue;
448 case 'l':
449 forever = 1;
450 continue;
451 case 'n':
452 not = 1;
453 continue;
454 case 's':
455 if (parse_num(¶m.length, optarg))
456 goto usage;
457 continue;
458 case 't':
459 test = atoi (optarg);
460 if (test < 0)
461 goto usage;
462 continue;
463 case 'v':
464 if (parse_num(¶m.vary, optarg))
465 goto usage;
466 continue;
467 case '?':
468 case 'h':
469 default:
470usage:
471 fprintf (stderr,
472 "usage: %s [options]\n"
473 "Options:\n"
474 "\t-D dev only test specific device\n"
475 "\t-A usbfs-dir\n"
476 "\t-a test all recognized devices\n"
477 "\t-l loop forever(for stress test)\n"
478 "\t-t testnum only run specified case\n"
479 "\t-n no test running, show devices to be tested\n"
480 "Case arguments:\n"
481 "\t-c iterations default 1000\n"
482 "\t-s packetsize default 512\n"
483 "\t-g sglen default 32\n"
484 "\t-v vary default 512\n",
485 argv[0]);
486 return 1;
487 }
488 if (optind != argc)
489 goto usage;
490 if (!all && !device) {
491 fprintf (stderr, "must specify '-a' or '-D dev', "
492 "or DEVICE=/proc/bus/usb/BBB/DDD in env\n");
493 goto usage;
494 }
495
496
497 if (!usbfs_dir) {
498 usbfs_dir = usbfs_dir_find();
499 if (!usbfs_dir) {
500 fputs ("usbfs files are missing\n", stderr);
501 return -1;
502 }
503 }
504
505
506 if (ftw (usbfs_dir, find_testdev, 3) != 0) {
507 fputs ("ftw failed; is usbfs missing?\n", stderr);
508 return -1;
509 }
510
511
512 if (!testdevs && !device) {
513 fputs ("no test devices recognized\n", stderr);
514 return -1;
515 }
516 if (not)
517 return 0;
518 if (testdevs && testdevs->next == 0 && !device)
519 device = testdevs->name;
520 for (entry = testdevs; entry; entry = entry->next) {
521 int status;
522
523 entry->param = param;
524 entry->forever = forever;
525 entry->test = test;
526
527 if (device) {
528 if (strcmp (entry->name, device))
529 continue;
530 return handle_testdev (entry) != entry;
531 }
532 status = pthread_create (&entry->thread, 0, handle_testdev, entry);
533 if (status) {
534 perror ("pthread_create");
535 continue;
536 }
537 }
538 if (device) {
539 struct testdev dev;
540
541
542 fprintf (stderr, "%s: %s may see only control tests\n",
543 argv [0], device);
544
545 memset (&dev, 0, sizeof dev);
546 dev.name = device;
547 dev.param = param;
548 dev.forever = forever;
549 dev.test = test;
550 return handle_testdev (&dev) != &dev;
551 }
552
553
554 for (entry = testdevs; entry; entry = entry->next) {
555 void *retval;
556
557 if (pthread_join (entry->thread, &retval))
558 perror ("pthread_join");
559
560 }
561
562 return 0;
563}
564