usbdevs.c revision 1.40 1 1.40 christos /* $NetBSD: usbdevs.c,v 1.40 2019/11/27 17:56:08 christos 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.40 christos __RCSID("$NetBSD: usbdevs.c,v 1.40 2019/11/27 17:56:08 christos Exp $");
35 1.33 skrll #endif
36 1.33 skrll
37 1.39 mrg #include <sys/types.h>
38 1.39 mrg #include <sys/drvctlio.h>
39 1.39 mrg
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.39 mrg #include <ctype.h>
51 1.40 christos #include <inttypes.h>
52 1.39 mrg
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.38 kamil icres = iconv(ic, __UNCONST(&utf8str), &insz, &termstr,
103 1.38 kamil &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.36 jmcneill 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.39 mrg static int
325 1.39 mrg getusbcount_device(int fd, const char *dev, int depth)
326 1.39 mrg {
327 1.39 mrg struct devlistargs laa = {
328 1.39 mrg .l_childname = NULL,
329 1.39 mrg .l_children = 0,
330 1.39 mrg };
331 1.39 mrg size_t i;
332 1.39 mrg size_t children;
333 1.39 mrg int nbusses = 0;
334 1.39 mrg
335 1.39 mrg if (depth && (dev == NULL || *dev == '\0'))
336 1.39 mrg return 0;
337 1.39 mrg
338 1.39 mrg /*
339 1.39 mrg * Look for children that match "usb[0-9]*". Could maybe
340 1.39 mrg * simply return 1 here, but there's always a chance that
341 1.39 mrg * someone has eg, a USB to PCI bridge, with a USB
342 1.39 mrg * controller behind PCI.
343 1.39 mrg */
344 1.39 mrg if (strncmp(dev, "usb", 3) == 0 && isdigit((int)dev[3]))
345 1.39 mrg nbusses++;
346 1.39 mrg
347 1.39 mrg strlcpy(laa.l_devname, dev, sizeof(laa.l_devname));
348 1.39 mrg
349 1.39 mrg if (ioctl(fd, DRVLISTDEV, &laa) == -1)
350 1.39 mrg err(EXIT_FAILURE, "DRVLISTDEV");
351 1.39 mrg children = laa.l_children;
352 1.39 mrg
353 1.39 mrg laa.l_childname = malloc(children * sizeof(laa.l_childname[0]));
354 1.39 mrg if (laa.l_childname == NULL)
355 1.39 mrg err(EXIT_FAILURE, "out of memory");
356 1.39 mrg if (ioctl(fd, DRVLISTDEV, &laa) == -1)
357 1.39 mrg err(EXIT_FAILURE, "DRVLISTDEV");
358 1.39 mrg if (laa.l_children > children)
359 1.39 mrg err(EXIT_FAILURE, "DRVLISTDEV: number of children grew");
360 1.39 mrg
361 1.39 mrg for (i = 0; i < laa.l_children; i++) {
362 1.39 mrg nbusses += getusbcount_device(fd, laa.l_childname[i], depth+1);
363 1.39 mrg }
364 1.39 mrg
365 1.39 mrg return nbusses;
366 1.39 mrg }
367 1.39 mrg
368 1.1 augustss int
369 1.14 augustss main(int argc, char **argv)
370 1.1 augustss {
371 1.40 christos 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.40 christos addr = strtoi(optarg, NULL, 10, 0, USB_MAX_DEVICES - 1,
381 1.40 christos &error);
382 1.40 christos if (error) {
383 1.40 christos errc(EXIT_FAILURE, error,
384 1.40 christos "Bad value for device address: `%s'",
385 1.40 christos optarg);
386 1.40 christos }
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.39 mrg int nbusses;
407 1.39 mrg int fd = open(DRVCTLDEV, O_RDONLY, 0);
408 1.39 mrg
409 1.39 mrg /* If no drvctl configured, default to 16. */
410 1.39 mrg if (fd != -1)
411 1.39 mrg nbusses = getusbcount_device(fd, "", 0);
412 1.39 mrg else
413 1.39 mrg nbusses = 16;
414 1.39 mrg close(fd);
415 1.39 mrg
416 1.39 mrg 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.40 christos return EXIT_SUCCESS;
440 1.1 augustss }
441