Home | History | Annotate | Line # | Download | only in usbdevs
usbdevs.c revision 1.35.14.2
      1  1.35.14.2    martin /*	$NetBSD: usbdevs.c,v 1.35.14.2 2020/04/13 08:06:07 martin Exp $	*/
      2        1.2  augustss 
      3        1.2  augustss /*
      4        1.2  augustss  * Copyright (c) 1998 The NetBSD Foundation, Inc.
      5        1.2  augustss  * All rights reserved.
      6        1.2  augustss  *
      7        1.5  augustss  * This code is derived from software contributed to The NetBSD Foundation
      8       1.22     grant  * by Lennart Augustsson (augustss (at) NetBSD.org).
      9        1.2  augustss  *
     10        1.2  augustss  * Redistribution and use in source and binary forms, with or without
     11        1.2  augustss  * modification, are permitted provided that the following conditions
     12        1.2  augustss  * are met:
     13        1.2  augustss  * 1. Redistributions of source code must retain the above copyright
     14        1.2  augustss  *    notice, this list of conditions and the following disclaimer.
     15        1.2  augustss  * 2. Redistributions in binary form must reproduce the above copyright
     16        1.2  augustss  *    notice, this list of conditions and the following disclaimer in the
     17        1.2  augustss  *    documentation and/or other materials provided with the distribution.
     18        1.2  augustss  *
     19        1.2  augustss  * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
     20        1.2  augustss  * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
     21        1.2  augustss  * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
     22        1.2  augustss  * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
     23        1.2  augustss  * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
     24        1.2  augustss  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
     25        1.2  augustss  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
     26        1.2  augustss  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
     27        1.2  augustss  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
     28        1.2  augustss  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
     29        1.2  augustss  * POSSIBILITY OF SUCH DAMAGE.
     30        1.2  augustss  */
     31        1.2  augustss 
     32       1.33     skrll #include <sys/cdefs.h>
     33       1.33     skrll #ifndef lint
     34  1.35.14.2    martin __RCSID("$NetBSD: usbdevs.c,v 1.35.14.2 2020/04/13 08:06:07 martin Exp $");
     35       1.33     skrll #endif
     36       1.33     skrll 
     37  1.35.14.2    martin #include <sys/types.h>
     38  1.35.14.2    martin #include <sys/drvctlio.h>
     39  1.35.14.2    martin 
     40        1.1  augustss #include <stdio.h>
     41        1.1  augustss #include <stdlib.h>
     42        1.1  augustss #include <string.h>
     43        1.1  augustss #include <fcntl.h>
     44        1.1  augustss #include <unistd.h>
     45        1.1  augustss #include <err.h>
     46        1.3  augustss #include <errno.h>
     47       1.26  drochner #include <locale.h>
     48       1.26  drochner #include <langinfo.h>
     49       1.26  drochner #include <iconv.h>
     50  1.35.14.2    martin #include <ctype.h>
     51  1.35.14.2    martin #include <inttypes.h>
     52  1.35.14.2    martin 
     53        1.1  augustss #include <dev/usb/usb.h>
     54        1.1  augustss 
     55        1.1  augustss #define USBDEV "/dev/usb"
     56        1.1  augustss 
     57       1.28     joerg static int verbose = 0;
     58       1.28     joerg static int showdevs = 0;
     59        1.1  augustss 
     60       1.32   mlelstv struct stringtable {
     61       1.32   mlelstv 	int row, col;
     62       1.32   mlelstv 	const char *string;
     63       1.32   mlelstv };
     64       1.32   mlelstv 
     65       1.28     joerg __dead static void usage(void);
     66       1.32   mlelstv static void getstrings(const struct stringtable *, int, int, const char **, const char **);
     67       1.28     joerg static void usbdev(int f, int a, int rec);
     68       1.28     joerg static void usbdump(int f);
     69       1.28     joerg static void dumpone(char *name, int f, int addr);
     70        1.1  augustss 
     71       1.28     joerg static void
     72       1.28     joerg usage(void)
     73        1.1  augustss {
     74       1.20    simonb 
     75       1.27       wiz 	fprintf(stderr, "usage: %s [-dv] [-a addr] [-f dev]\n",
     76       1.17       cgd 	    getprogname());
     77       1.29  jakllsch 	exit(EXIT_FAILURE);
     78        1.1  augustss }
     79        1.1  augustss 
     80       1.28     joerg static char done[USB_MAX_DEVICES];
     81       1.28     joerg static int indent;
     82       1.26  drochner #define MAXLEN USB_MAX_ENCODED_STRING_LEN /* assume can't grow over UTF-8 */
     83       1.28     joerg static char vendor[MAXLEN], product[MAXLEN], serial[MAXLEN];
     84       1.26  drochner 
     85       1.26  drochner static void
     86       1.26  drochner u2t(const char *utf8str, char *termstr)
     87       1.26  drochner {
     88       1.26  drochner 	static iconv_t ic;
     89       1.26  drochner 	static int iconv_inited = 0;
     90       1.26  drochner 	size_t insz, outsz, icres;
     91       1.26  drochner 
     92       1.26  drochner 	if (!iconv_inited) {
     93       1.26  drochner 		setlocale(LC_ALL, "");
     94       1.26  drochner 		ic = iconv_open(nl_langinfo(CODESET), "UTF-8");
     95       1.26  drochner 		if (ic == (iconv_t)-1)
     96       1.26  drochner 			ic = iconv_open("ASCII", "UTF-8"); /* g.c.d. */
     97       1.26  drochner 		iconv_inited = 1;
     98       1.26  drochner 	}
     99       1.26  drochner 	if (ic != (iconv_t)-1) {
    100       1.26  drochner 		insz = strlen(utf8str);
    101       1.26  drochner 		outsz = MAXLEN - 1;
    102  1.35.14.2    martin 		icres = iconv(ic, __UNCONST(&utf8str), &insz, &termstr,
    103  1.35.14.2    martin 			&outsz);
    104       1.26  drochner 		if (icres != (size_t)-1) {
    105       1.26  drochner 			*termstr = '\0';
    106       1.26  drochner 			return;
    107       1.26  drochner 		}
    108       1.26  drochner 	}
    109       1.26  drochner 	strcpy(termstr, "(invalid)");
    110       1.26  drochner }
    111        1.1  augustss 
    112       1.32   mlelstv struct stringtable class_strings[] = {
    113       1.32   mlelstv 	{ UICLASS_UNSPEC,      -1, "Unspecified" },
    114       1.32   mlelstv 
    115       1.32   mlelstv 	{ UICLASS_AUDIO,       -1, "Audio" },
    116       1.32   mlelstv 	{ UICLASS_AUDIO,       UISUBCLASS_AUDIOCONTROL, "Audio Control" },
    117       1.32   mlelstv 	{ UICLASS_AUDIO,       UISUBCLASS_AUDIOSTREAM, "Audio Streaming" },
    118       1.32   mlelstv 	{ UICLASS_AUDIO,       UISUBCLASS_MIDISTREAM, "MIDI Streaming" },
    119       1.32   mlelstv 
    120       1.32   mlelstv 	{ UICLASS_CDC,         -1, "Communications and CDC Control" },
    121       1.32   mlelstv 	{ UICLASS_CDC,         UISUBCLASS_DIRECT_LINE_CONTROL_MODEL, "Direct Line" },
    122       1.32   mlelstv 	{ UICLASS_CDC,         UISUBCLASS_ABSTRACT_CONTROL_MODEL, "Abstract" },
    123       1.32   mlelstv 	{ UICLASS_CDC,         UISUBCLASS_TELEPHONE_CONTROL_MODEL, "Telephone" },
    124       1.32   mlelstv 	{ UICLASS_CDC,         UISUBCLASS_MULTICHANNEL_CONTROL_MODEL, "Multichannel" },
    125       1.32   mlelstv 	{ UICLASS_CDC,         UISUBCLASS_CAPI_CONTROLMODEL, "CAPI" },
    126       1.32   mlelstv 	{ UICLASS_CDC,         UISUBCLASS_ETHERNET_NETWORKING_CONTROL_MODEL, "Ethernet Networking" },
    127       1.32   mlelstv 	{ UICLASS_CDC,         UISUBCLASS_ATM_NETWORKING_CONTROL_MODEL, "ATM Networking" },
    128       1.32   mlelstv 
    129       1.32   mlelstv 	{ UICLASS_HID,         -1, "Human Interface Device" },
    130       1.32   mlelstv 	{ UICLASS_HID,         UISUBCLASS_BOOT, "Boot" },
    131       1.32   mlelstv 
    132       1.32   mlelstv 	{ UICLASS_PHYSICAL,    -1, "Physical" },
    133       1.32   mlelstv 
    134       1.32   mlelstv 	{ UICLASS_IMAGE,       -1, "Image" },
    135       1.32   mlelstv 
    136       1.32   mlelstv 	{ UICLASS_PRINTER,     -1, "Printer" },
    137       1.32   mlelstv 	{ UICLASS_PRINTER,     UISUBCLASS_PRINTER, "Printer" },
    138       1.32   mlelstv 
    139       1.32   mlelstv 	{ UICLASS_MASS,        -1, "Mass Storage" },
    140       1.32   mlelstv 	{ UICLASS_MASS,        UISUBCLASS_RBC, "RBC" },
    141       1.32   mlelstv 	{ UICLASS_MASS,        UISUBCLASS_SFF8020I, "SFF8020I" },
    142       1.32   mlelstv 	{ UICLASS_MASS,        UISUBCLASS_QIC157, "QIC157" },
    143       1.32   mlelstv 	{ UICLASS_MASS,        UISUBCLASS_UFI, "UFI" },
    144       1.32   mlelstv 	{ UICLASS_MASS,        UISUBCLASS_SFF8070I, "SFF8070I" },
    145       1.32   mlelstv 	{ UICLASS_MASS,        UISUBCLASS_SCSI, "SCSI" },
    146       1.32   mlelstv 	{ UICLASS_MASS,        UISUBCLASS_SCSI, "SCSI" },
    147       1.32   mlelstv 
    148       1.32   mlelstv 	{ UICLASS_HUB,         -1, "Hub" },
    149       1.32   mlelstv 	{ UICLASS_HUB,         UISUBCLASS_HUB, "Hub" },
    150       1.32   mlelstv 
    151       1.32   mlelstv 	{ UICLASS_CDC_DATA,    -1, "CDC-Data" },
    152       1.32   mlelstv 	{ UICLASS_CDC_DATA,    UISUBCLASS_DATA, "Data" },
    153       1.32   mlelstv 
    154       1.32   mlelstv 	{ UICLASS_SMARTCARD,   -1, "Smart Card" },
    155       1.32   mlelstv 
    156       1.32   mlelstv 	{ UICLASS_SECURITY,    -1, "Content Security" },
    157       1.32   mlelstv 
    158       1.32   mlelstv 	{ UICLASS_VIDEO,       -1, "Video" },
    159       1.32   mlelstv 	{ UICLASS_VIDEO,       UISUBCLASS_VIDEOCONTROL, "Video Control" },
    160       1.32   mlelstv 	{ UICLASS_VIDEO,       UISUBCLASS_VIDEOSTREAMING, "Video Streaming" },
    161       1.32   mlelstv 	{ UICLASS_VIDEO,       UISUBCLASS_VIDEOCOLLECTION, "Video Collection" },
    162       1.32   mlelstv 
    163       1.32   mlelstv #ifdef notyet
    164       1.32   mlelstv 	{ UICLASS_HEALTHCARE,  -1, "Personal Healthcare" },
    165       1.32   mlelstv 	{ UICLASS_AVDEVICE,    -1, "Audio/Video Device" },
    166       1.32   mlelstv 	{ UICLASS_BILLBOARD,   -1, "Billboard" },
    167       1.32   mlelstv #endif
    168       1.32   mlelstv 
    169       1.32   mlelstv 	{ UICLASS_DIAGNOSTIC,  -1, "Diagnostic" },
    170       1.32   mlelstv 	{ UICLASS_WIRELESS,    -1, "Wireless" },
    171       1.32   mlelstv 	{ UICLASS_WIRELESS,    UISUBCLASS_RF, "Radio Frequency" },
    172       1.32   mlelstv 
    173       1.32   mlelstv #ifdef notyet
    174       1.32   mlelstv 	{ UICLASS_MISC,        -1, "Miscellaneous" },
    175       1.32   mlelstv #endif
    176       1.32   mlelstv 
    177       1.32   mlelstv 	{ UICLASS_APPL_SPEC,   -1, "Application Specific" },
    178       1.32   mlelstv 	{ UICLASS_APPL_SPEC,   UISUBCLASS_FIRMWARE_DOWNLOAD, "Firmware Download" },
    179       1.32   mlelstv 	{ UICLASS_APPL_SPEC,   UISUBCLASS_IRDA,              "Irda" },
    180       1.32   mlelstv 
    181       1.32   mlelstv 	{ UICLASS_VENDOR,      -1, "Vendor Specific" },
    182       1.32   mlelstv 
    183       1.32   mlelstv 	{ -1, -1, NULL }
    184       1.32   mlelstv };
    185       1.32   mlelstv 
    186       1.32   mlelstv static void
    187       1.32   mlelstv getstrings(const struct stringtable *table,
    188       1.32   mlelstv            int row, int col, const char **rp, const char **cp) {
    189       1.32   mlelstv 	static char rbuf[5], cbuf[5];
    190       1.32   mlelstv 
    191       1.32   mlelstv 	snprintf(rbuf, sizeof(rbuf), "0x%02x", row);
    192       1.32   mlelstv 	snprintf(cbuf, sizeof(cbuf), "0x%02x", col);
    193       1.32   mlelstv 
    194       1.32   mlelstv 	*rp = rbuf;
    195       1.32   mlelstv 	*cp = cbuf;
    196       1.32   mlelstv 
    197       1.32   mlelstv 	while (table->string != NULL) {
    198       1.32   mlelstv 		if (table->row == row) {
    199       1.32   mlelstv 			if (table->col == -1)
    200       1.32   mlelstv 				*rp = table->string;
    201       1.32   mlelstv 			else if (table->col == col)
    202       1.32   mlelstv 				*cp = table->string;
    203       1.32   mlelstv 		} else if (table->row > row)
    204       1.32   mlelstv 			break;
    205       1.32   mlelstv 
    206       1.32   mlelstv 		++table;
    207       1.32   mlelstv 	}
    208       1.32   mlelstv }
    209       1.32   mlelstv 
    210       1.28     joerg static void
    211       1.14  augustss usbdev(int f, int a, int rec)
    212        1.1  augustss {
    213        1.1  augustss 	struct usb_device_info di;
    214       1.35     skrll 	int e, i;
    215        1.1  augustss 
    216       1.19  christos 	di.udi_addr = a;
    217        1.1  augustss 	e = ioctl(f, USB_DEVICEINFO, &di);
    218       1.10  augustss 	if (e) {
    219       1.10  augustss 		if (errno != ENXIO)
    220       1.10  augustss 			printf("addr %d: I/O error\n", a);
    221        1.1  augustss 		return;
    222       1.10  augustss 	}
    223       1.10  augustss 	printf("addr %d: ", a);
    224        1.1  augustss 	done[a] = 1;
    225        1.1  augustss 	if (verbose) {
    226       1.19  christos 		switch (di.udi_speed) {
    227       1.18  augustss 		case USB_SPEED_LOW:  printf("low speed, "); break;
    228       1.18  augustss 		case USB_SPEED_FULL: printf("full speed, "); break;
    229       1.18  augustss 		case USB_SPEED_HIGH: printf("high speed, "); break;
    230       1.31     skrll 		case USB_SPEED_SUPER: printf("super speed, "); break;
    231  1.35.14.1  christos 		case USB_SPEED_SUPER_PLUS: printf("super speed+, "); break;
    232       1.18  augustss 		default: break;
    233       1.18  augustss 		}
    234       1.19  christos 		if (di.udi_power)
    235       1.19  christos 			printf("power %d mA, ", di.udi_power);
    236        1.1  augustss 		else
    237        1.1  augustss 			printf("self powered, ");
    238       1.19  christos 		if (di.udi_config)
    239       1.19  christos 			printf("config %d, ", di.udi_config);
    240        1.1  augustss 		else
    241        1.1  augustss 			printf("unconfigured, ");
    242        1.1  augustss 	}
    243       1.26  drochner 	u2t(di.udi_product, product);
    244       1.26  drochner 	u2t(di.udi_vendor, vendor);
    245       1.26  drochner 	u2t(di.udi_serial, serial);
    246        1.6  augustss 	if (verbose) {
    247       1.32   mlelstv 		printf("%s(0x%04x), %s(0x%04x), rev %s(0x%04x)",
    248       1.26  drochner 		       product, di.udi_productNo,
    249       1.32   mlelstv 		       vendor, di.udi_vendorNo,
    250       1.32   mlelstv 			di.udi_release, di.udi_releaseNo);
    251       1.24  augustss 		if (di.udi_serial[0])
    252       1.26  drochner 			printf(", serial %s", serial);
    253        1.6  augustss 	} else
    254       1.26  drochner 		printf("%s, %s", product, vendor);
    255        1.1  augustss 	printf("\n");
    256       1.32   mlelstv 	if (verbose > 1 && di.udi_class != UICLASS_UNSPEC) {
    257       1.32   mlelstv 		const char *cstr, *sstr;
    258       1.32   mlelstv 		getstrings(class_strings, di.udi_class, di.udi_subclass, &cstr, &sstr);
    259       1.32   mlelstv 		printf("%*s  %s(0x%02x), %s(0x%02x), proto %u\n", indent, "",
    260       1.32   mlelstv 			cstr, di.udi_class, sstr, di.udi_subclass,
    261       1.32   mlelstv 			di.udi_protocol);
    262       1.32   mlelstv 	}
    263       1.15  augustss 	if (showdevs) {
    264       1.15  augustss 		for (i = 0; i < USB_MAX_DEVNAMES; i++)
    265       1.19  christos 			if (di.udi_devnames[i][0])
    266       1.15  augustss 				printf("%*s  %s\n", indent, "",
    267       1.19  christos 				       di.udi_devnames[i]);
    268       1.15  augustss 	}
    269        1.1  augustss 	if (!rec)
    270        1.1  augustss 		return;
    271       1.35     skrll 
    272       1.35     skrll 	unsigned int nports = di.udi_nports;
    273       1.35     skrll 
    274       1.35     skrll 	for (unsigned int p = 0; p < nports && p < __arraycount(di.udi_ports); p++) {
    275       1.19  christos 		int s = di.udi_ports[p];
    276        1.1  augustss 		if (s >= USB_MAX_DEVICES) {
    277        1.1  augustss 			if (verbose) {
    278        1.1  augustss 				printf("%*sport %d %s\n", indent+1, "", p+1,
    279        1.1  augustss 				       s == USB_PORT_ENABLED ? "enabled" :
    280        1.1  augustss 				       s == USB_PORT_SUSPENDED ? "suspended" :
    281        1.1  augustss 				       s == USB_PORT_POWERED ? "powered" :
    282        1.1  augustss 				       s == USB_PORT_DISABLED ? "disabled" :
    283        1.1  augustss 				       "???");
    284        1.1  augustss 
    285        1.1  augustss 			}
    286        1.1  augustss 			continue;
    287        1.1  augustss 		}
    288        1.1  augustss 		indent++;
    289        1.1  augustss 		printf("%*s", indent, "");
    290        1.1  augustss 		if (verbose)
    291        1.1  augustss 			printf("port %d ", p+1);
    292       1.10  augustss 		if (s == 0)
    293       1.10  augustss 			printf("addr 0 should never happen!\n");
    294       1.10  augustss 		else
    295       1.10  augustss 			usbdev(f, s, 1);
    296        1.1  augustss 		indent--;
    297        1.1  augustss 	}
    298        1.1  augustss }
    299        1.1  augustss 
    300       1.28     joerg static void
    301       1.14  augustss usbdump(int f)
    302        1.1  augustss {
    303        1.1  augustss 	int a;
    304        1.1  augustss 
    305       1.30  jakllsch 	for (a = 0; a < USB_MAX_DEVICES; a++) {
    306        1.1  augustss 		if (!done[a])
    307        1.1  augustss 			usbdev(f, a, 1);
    308        1.1  augustss 	}
    309        1.1  augustss }
    310        1.1  augustss 
    311       1.28     joerg static void
    312       1.14  augustss dumpone(char *name, int f, int addr)
    313        1.1  augustss {
    314        1.1  augustss 	if (verbose)
    315        1.1  augustss 		printf("Controller %s:\n", name);
    316        1.1  augustss 	indent = 0;
    317        1.1  augustss 	memset(done, 0, sizeof done);
    318       1.30  jakllsch 	if (addr >= 0)
    319        1.1  augustss 		usbdev(f, addr, 0);
    320        1.1  augustss 	else
    321        1.1  augustss 		usbdump(f);
    322        1.1  augustss }
    323        1.1  augustss 
    324  1.35.14.2    martin static int
    325  1.35.14.2    martin getusbcount_device(int fd, const char *dev, int depth)
    326  1.35.14.2    martin {
    327  1.35.14.2    martin 	struct devlistargs laa = {
    328  1.35.14.2    martin 	    .l_childname = NULL,
    329  1.35.14.2    martin 	    .l_children = 0,
    330  1.35.14.2    martin 	};
    331  1.35.14.2    martin 	size_t i;
    332  1.35.14.2    martin 	size_t children;
    333  1.35.14.2    martin 	int nbusses = 0;
    334  1.35.14.2    martin 
    335  1.35.14.2    martin 	if (depth && (dev == NULL || *dev == '\0'))
    336  1.35.14.2    martin 		return 0;
    337  1.35.14.2    martin 
    338  1.35.14.2    martin 	/*
    339  1.35.14.2    martin 	 * Look for children that match "usb[0-9]*".  Could maybe
    340  1.35.14.2    martin 	 * simply return 1 here, but there's always a chance that
    341  1.35.14.2    martin 	 * someone has eg, a USB to PCI bridge, with a USB
    342  1.35.14.2    martin 	 * controller behind PCI.
    343  1.35.14.2    martin 	 */
    344  1.35.14.2    martin 	if (strncmp(dev, "usb", 3) == 0 && isdigit((int)dev[3]))
    345  1.35.14.2    martin 		nbusses++;
    346  1.35.14.2    martin 
    347  1.35.14.2    martin 	strlcpy(laa.l_devname, dev, sizeof(laa.l_devname));
    348  1.35.14.2    martin 
    349  1.35.14.2    martin 	if (ioctl(fd, DRVLISTDEV, &laa) == -1)
    350  1.35.14.2    martin 		err(EXIT_FAILURE, "DRVLISTDEV");
    351  1.35.14.2    martin 	children = laa.l_children;
    352  1.35.14.2    martin 
    353  1.35.14.2    martin 	laa.l_childname = malloc(children * sizeof(laa.l_childname[0]));
    354  1.35.14.2    martin 	if (laa.l_childname == NULL)
    355  1.35.14.2    martin 		err(EXIT_FAILURE, "out of memory");
    356  1.35.14.2    martin 	if (ioctl(fd, DRVLISTDEV, &laa) == -1)
    357  1.35.14.2    martin 		err(EXIT_FAILURE, "DRVLISTDEV");
    358  1.35.14.2    martin 	if (laa.l_children > children)
    359  1.35.14.2    martin 		err(EXIT_FAILURE, "DRVLISTDEV: number of children grew");
    360  1.35.14.2    martin 
    361  1.35.14.2    martin 	for (i = 0; i < laa.l_children; i++) {
    362  1.35.14.2    martin 		nbusses += getusbcount_device(fd, laa.l_childname[i], depth+1);
    363  1.35.14.2    martin 	}
    364  1.35.14.2    martin 
    365  1.35.14.2    martin 	return nbusses;
    366  1.35.14.2    martin }
    367  1.35.14.2    martin 
    368        1.1  augustss int
    369       1.14  augustss main(int argc, char **argv)
    370        1.1  augustss {
    371  1.35.14.2    martin 	int ch, i, f, error;
    372        1.1  augustss 	char buf[50];
    373       1.29  jakllsch 	char *dev = NULL;
    374       1.30  jakllsch 	int addr = -1;
    375        1.4  augustss 	int ncont;
    376        1.1  augustss 
    377       1.16  augustss 	while ((ch = getopt(argc, argv, "a:df:v?")) != -1) {
    378        1.1  augustss 		switch(ch) {
    379        1.1  augustss 		case 'a':
    380  1.35.14.2    martin 			addr = strtoi(optarg, NULL, 10, 0, USB_MAX_DEVICES - 1,
    381  1.35.14.2    martin 			    &error);
    382  1.35.14.2    martin 			if (error) {
    383  1.35.14.2    martin 				errc(EXIT_FAILURE, error,
    384  1.35.14.2    martin 				    "Bad value for device address: `%s'",
    385  1.35.14.2    martin 				    optarg);
    386  1.35.14.2    martin 			}
    387       1.15  augustss 			break;
    388       1.15  augustss 		case 'd':
    389       1.15  augustss 			showdevs++;
    390        1.1  augustss 			break;
    391        1.1  augustss 		case 'f':
    392        1.1  augustss 			dev = optarg;
    393        1.1  augustss 			break;
    394        1.1  augustss 		case 'v':
    395       1.32   mlelstv 			verbose++;
    396        1.1  augustss 			break;
    397        1.1  augustss 		case '?':
    398        1.1  augustss 		default:
    399        1.1  augustss 			usage();
    400        1.1  augustss 		}
    401        1.1  augustss 	}
    402        1.1  augustss 	argc -= optind;
    403        1.1  augustss 	argv += optind;
    404        1.1  augustss 
    405       1.29  jakllsch 	if (dev == NULL) {
    406  1.35.14.2    martin 		int nbusses;
    407  1.35.14.2    martin 		int fd = open(DRVCTLDEV, O_RDONLY, 0);
    408  1.35.14.2    martin 
    409  1.35.14.2    martin 		/* If no drvctl configured, default to 16. */
    410  1.35.14.2    martin 		if (fd != -1)
    411  1.35.14.2    martin 			nbusses = getusbcount_device(fd, "", 0);
    412  1.35.14.2    martin 		else
    413  1.35.14.2    martin 			nbusses = 16;
    414  1.35.14.2    martin 		close(fd);
    415  1.35.14.2    martin 
    416  1.35.14.2    martin 		for (ncont = 0, i = 0; i < nbusses; i++) {
    417       1.21    itojun 			snprintf(buf, sizeof(buf), "%s%d", USBDEV, i);
    418        1.1  augustss 			f = open(buf, O_RDONLY);
    419        1.1  augustss 			if (f >= 0) {
    420        1.1  augustss 				dumpone(buf, f, addr);
    421        1.1  augustss 				close(f);
    422        1.3  augustss 			} else {
    423       1.11  augustss 				if (errno == ENOENT || errno == ENXIO)
    424       1.11  augustss 					continue;
    425       1.11  augustss 				warn("%s", buf);
    426        1.1  augustss 			}
    427       1.11  augustss 			ncont++;
    428        1.1  augustss 		}
    429        1.4  augustss 		if (verbose && ncont == 0)
    430       1.17       cgd 			printf("%s: no USB controllers found\n",
    431       1.17       cgd 			    getprogname());
    432        1.1  augustss 	} else {
    433        1.1  augustss 		f = open(dev, O_RDONLY);
    434        1.1  augustss 		if (f >= 0)
    435        1.1  augustss 			dumpone(dev, f, addr);
    436        1.1  augustss 		else
    437        1.1  augustss 			err(1, "%s", dev);
    438        1.1  augustss 	}
    439  1.35.14.2    martin 	return EXIT_SUCCESS;
    440        1.1  augustss }
    441