usbdevs.c revision 1.26 1 1.26 drochner /* $NetBSD: usbdevs.c,v 1.26 2010/02/02 16:25:30 drochner 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.1 augustss #include <stdio.h>
33 1.1 augustss #include <stdlib.h>
34 1.1 augustss #include <string.h>
35 1.1 augustss #include <sys/types.h>
36 1.1 augustss #include <fcntl.h>
37 1.1 augustss #include <unistd.h>
38 1.1 augustss #include <err.h>
39 1.3 augustss #include <errno.h>
40 1.26 drochner #include <locale.h>
41 1.26 drochner #include <langinfo.h>
42 1.26 drochner #include <iconv.h>
43 1.1 augustss #include <dev/usb/usb.h>
44 1.1 augustss
45 1.1 augustss #define USBDEV "/dev/usb"
46 1.1 augustss
47 1.15 augustss int verbose = 0;
48 1.15 augustss int showdevs = 0;
49 1.1 augustss
50 1.14 augustss void usage(void);
51 1.14 augustss void usbdev(int f, int a, int rec);
52 1.14 augustss void usbdump(int f);
53 1.14 augustss void dumpone(char *name, int f, int addr);
54 1.14 augustss int main(int, char **);
55 1.1 augustss
56 1.1 augustss void
57 1.1 augustss usage()
58 1.1 augustss {
59 1.20 simonb
60 1.23 jmmv fprintf(stderr, "usage: %s [-a addr] [-d] [-f dev] [-v]\n",
61 1.17 cgd getprogname());
62 1.1 augustss exit(1);
63 1.1 augustss }
64 1.1 augustss
65 1.1 augustss char done[USB_MAX_DEVICES];
66 1.1 augustss int indent;
67 1.26 drochner #define MAXLEN USB_MAX_ENCODED_STRING_LEN /* assume can't grow over UTF-8 */
68 1.26 drochner char vendor[MAXLEN], product[MAXLEN], serial[MAXLEN];
69 1.26 drochner
70 1.26 drochner static void
71 1.26 drochner u2t(const char *utf8str, char *termstr)
72 1.26 drochner {
73 1.26 drochner static iconv_t ic;
74 1.26 drochner static int iconv_inited = 0;
75 1.26 drochner size_t insz, outsz, icres;
76 1.26 drochner
77 1.26 drochner if (!iconv_inited) {
78 1.26 drochner setlocale(LC_ALL, "");
79 1.26 drochner ic = iconv_open(nl_langinfo(CODESET), "UTF-8");
80 1.26 drochner if (ic == (iconv_t)-1)
81 1.26 drochner ic = iconv_open("ASCII", "UTF-8"); /* g.c.d. */
82 1.26 drochner iconv_inited = 1;
83 1.26 drochner }
84 1.26 drochner if (ic != (iconv_t)-1) {
85 1.26 drochner insz = strlen(utf8str);
86 1.26 drochner outsz = MAXLEN - 1;
87 1.26 drochner icres = iconv(ic, &utf8str, &insz, &termstr, &outsz);
88 1.26 drochner if (icres != (size_t)-1) {
89 1.26 drochner *termstr = '\0';
90 1.26 drochner return;
91 1.26 drochner }
92 1.26 drochner }
93 1.26 drochner strcpy(termstr, "(invalid)");
94 1.26 drochner }
95 1.1 augustss
96 1.1 augustss void
97 1.14 augustss usbdev(int f, int a, int rec)
98 1.1 augustss {
99 1.1 augustss struct usb_device_info di;
100 1.15 augustss int e, p, i;
101 1.1 augustss
102 1.19 christos di.udi_addr = a;
103 1.1 augustss e = ioctl(f, USB_DEVICEINFO, &di);
104 1.10 augustss if (e) {
105 1.10 augustss if (errno != ENXIO)
106 1.10 augustss printf("addr %d: I/O error\n", a);
107 1.1 augustss return;
108 1.10 augustss }
109 1.10 augustss printf("addr %d: ", a);
110 1.1 augustss done[a] = 1;
111 1.1 augustss if (verbose) {
112 1.19 christos switch (di.udi_speed) {
113 1.18 augustss case USB_SPEED_LOW: printf("low speed, "); break;
114 1.18 augustss case USB_SPEED_FULL: printf("full speed, "); break;
115 1.18 augustss case USB_SPEED_HIGH: printf("high speed, "); break;
116 1.18 augustss default: break;
117 1.18 augustss }
118 1.19 christos if (di.udi_power)
119 1.19 christos printf("power %d mA, ", di.udi_power);
120 1.1 augustss else
121 1.1 augustss printf("self powered, ");
122 1.19 christos if (di.udi_config)
123 1.19 christos printf("config %d, ", di.udi_config);
124 1.1 augustss else
125 1.1 augustss printf("unconfigured, ");
126 1.1 augustss }
127 1.26 drochner u2t(di.udi_product, product);
128 1.26 drochner u2t(di.udi_vendor, vendor);
129 1.26 drochner u2t(di.udi_serial, serial);
130 1.6 augustss if (verbose) {
131 1.6 augustss printf("%s(0x%04x), %s(0x%04x), rev %s",
132 1.26 drochner product, di.udi_productNo,
133 1.26 drochner vendor, di.udi_vendorNo, di.udi_release);
134 1.24 augustss if (di.udi_serial[0])
135 1.26 drochner printf(", serial %s", serial);
136 1.6 augustss } else
137 1.26 drochner printf("%s, %s", product, vendor);
138 1.1 augustss printf("\n");
139 1.15 augustss if (showdevs) {
140 1.15 augustss for (i = 0; i < USB_MAX_DEVNAMES; i++)
141 1.19 christos if (di.udi_devnames[i][0])
142 1.15 augustss printf("%*s %s\n", indent, "",
143 1.19 christos di.udi_devnames[i]);
144 1.15 augustss }
145 1.1 augustss if (!rec)
146 1.1 augustss return;
147 1.19 christos for (p = 0; p < di.udi_nports; p++) {
148 1.19 christos int s = di.udi_ports[p];
149 1.1 augustss if (s >= USB_MAX_DEVICES) {
150 1.1 augustss if (verbose) {
151 1.1 augustss printf("%*sport %d %s\n", indent+1, "", p+1,
152 1.1 augustss s == USB_PORT_ENABLED ? "enabled" :
153 1.1 augustss s == USB_PORT_SUSPENDED ? "suspended" :
154 1.1 augustss s == USB_PORT_POWERED ? "powered" :
155 1.1 augustss s == USB_PORT_DISABLED ? "disabled" :
156 1.1 augustss "???");
157 1.1 augustss
158 1.1 augustss }
159 1.1 augustss continue;
160 1.1 augustss }
161 1.1 augustss indent++;
162 1.1 augustss printf("%*s", indent, "");
163 1.1 augustss if (verbose)
164 1.1 augustss printf("port %d ", p+1);
165 1.10 augustss if (s == 0)
166 1.10 augustss printf("addr 0 should never happen!\n");
167 1.10 augustss else
168 1.10 augustss usbdev(f, s, 1);
169 1.1 augustss indent--;
170 1.1 augustss }
171 1.1 augustss }
172 1.1 augustss
173 1.1 augustss void
174 1.14 augustss usbdump(int f)
175 1.1 augustss {
176 1.1 augustss int a;
177 1.1 augustss
178 1.1 augustss for (a = 1; a < USB_MAX_DEVICES; a++) {
179 1.1 augustss if (!done[a])
180 1.1 augustss usbdev(f, a, 1);
181 1.1 augustss }
182 1.1 augustss }
183 1.1 augustss
184 1.1 augustss void
185 1.14 augustss dumpone(char *name, int f, int addr)
186 1.1 augustss {
187 1.1 augustss if (verbose)
188 1.1 augustss printf("Controller %s:\n", name);
189 1.1 augustss indent = 0;
190 1.1 augustss memset(done, 0, sizeof done);
191 1.1 augustss if (addr)
192 1.1 augustss usbdev(f, addr, 0);
193 1.1 augustss else
194 1.1 augustss usbdump(f);
195 1.1 augustss }
196 1.1 augustss
197 1.1 augustss int
198 1.14 augustss main(int argc, char **argv)
199 1.1 augustss {
200 1.1 augustss int ch, i, f;
201 1.1 augustss char buf[50];
202 1.1 augustss char *dev = 0;
203 1.1 augustss int addr = 0;
204 1.4 augustss int ncont;
205 1.1 augustss
206 1.16 augustss while ((ch = getopt(argc, argv, "a:df:v?")) != -1) {
207 1.1 augustss switch(ch) {
208 1.1 augustss case 'a':
209 1.1 augustss addr = atoi(optarg);
210 1.15 augustss break;
211 1.15 augustss case 'd':
212 1.15 augustss showdevs++;
213 1.1 augustss break;
214 1.1 augustss case 'f':
215 1.1 augustss dev = optarg;
216 1.1 augustss break;
217 1.1 augustss case 'v':
218 1.1 augustss verbose = 1;
219 1.1 augustss break;
220 1.1 augustss case '?':
221 1.1 augustss default:
222 1.1 augustss usage();
223 1.1 augustss }
224 1.1 augustss }
225 1.1 augustss argc -= optind;
226 1.1 augustss argv += optind;
227 1.1 augustss
228 1.1 augustss if (dev == 0) {
229 1.4 augustss for (ncont = 0, i = 0; i < 10; i++) {
230 1.21 itojun snprintf(buf, sizeof(buf), "%s%d", USBDEV, i);
231 1.1 augustss f = open(buf, O_RDONLY);
232 1.1 augustss if (f >= 0) {
233 1.1 augustss dumpone(buf, f, addr);
234 1.1 augustss close(f);
235 1.3 augustss } else {
236 1.11 augustss if (errno == ENOENT || errno == ENXIO)
237 1.11 augustss continue;
238 1.11 augustss warn("%s", buf);
239 1.1 augustss }
240 1.11 augustss ncont++;
241 1.1 augustss }
242 1.4 augustss if (verbose && ncont == 0)
243 1.17 cgd printf("%s: no USB controllers found\n",
244 1.17 cgd getprogname());
245 1.1 augustss } else {
246 1.1 augustss f = open(dev, O_RDONLY);
247 1.1 augustss if (f >= 0)
248 1.1 augustss dumpone(dev, f, addr);
249 1.1 augustss else
250 1.1 augustss err(1, "%s", dev);
251 1.1 augustss }
252 1.1 augustss exit(0);
253 1.1 augustss }
254