source: mainline/uspace/app/usbinfo/main.c@ c709eb0

lfn serial ticket/834-toolchain-update topic/msim-upgrade topic/simplify-dev-export
Last change on this file since c709eb0 was cb61e8f, checked in by Vojtech Horky <vojtechhorky@…>, 14 years ago

usbinfo prints device status as well

Device (and control endpoint zero) status is printed via `-S' switch.

  • Property mode set to 100644
File size: 6.8 KB
Line 
1/*
2 * Copyright (c) 2010-2011 Vojtech Horky
3 * All rights reserved.
4 *
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions
7 * are met:
8 *
9 * - Redistributions of source code must retain the above copyright
10 * notice, this list of conditions and the following disclaimer.
11 * - Redistributions in binary form must reproduce the above copyright
12 * notice, this list of conditions and the following disclaimer in the
13 * documentation and/or other materials provided with the distribution.
14 * - The name of the author may not be used to endorse or promote products
15 * derived from this software without specific prior written permission.
16 *
17 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
18 * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
19 * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
20 * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
21 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
22 * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
23 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
24 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
25 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
26 * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
27 */
28
29/** @addtogroup usbinfo
30 * @{
31 */
32/**
33 * @file
34 * USB querying.
35 */
36
37#include <stdio.h>
38#include <stdlib.h>
39#include <errno.h>
40#include <str_error.h>
41#include <bool.h>
42#include <getopt.h>
43#include <devman.h>
44#include <devmap.h>
45#include <usb/usbdevice.h>
46#include <usb/pipes.h>
47#include "usbinfo.h"
48
49static bool resolve_hc_handle_and_dev_addr(const char *devpath,
50 devman_handle_t *out_hc_handle, usb_address_t *out_device_address)
51{
52 int rc;
53
54 /* Hack for QEMU to save-up on typing ;-). */
55 if (str_cmp(devpath, "qemu") == 0) {
56 devpath = "/hw/pci0/00:01.2/uhci-rh/usb00_a1";
57 }
58
59 char *path = str_dup(devpath);
60 if (path == NULL) {
61 return ENOMEM;
62 }
63
64 devman_handle_t hc = 0;
65 bool hc_found = false;
66 usb_address_t addr = 0;
67 bool addr_found = false;
68
69 /* Remove suffixes and hope that we will encounter device node. */
70 while (str_length(path) > 0) {
71 /* Get device handle first. */
72 devman_handle_t dev_handle;
73 rc = devman_device_get_handle(path, &dev_handle, 0);
74 if (rc != EOK) {
75 free(path);
76 return false;
77 }
78
79 /* Try to find its host controller. */
80 if (!hc_found) {
81 rc = usb_hc_find(dev_handle, &hc);
82 if (rc == EOK) {
83 hc_found = true;
84 }
85 }
86 /* Try to get its address. */
87 if (!addr_found) {
88 addr = usb_device_get_assigned_address(dev_handle);
89 if (addr >= 0) {
90 addr_found = true;
91 }
92 }
93
94 /* Speed-up. */
95 if (hc_found && addr_found) {
96 break;
97 }
98
99 /* Remove the last suffix. */
100 char *slash_pos = str_rchr(path, '/');
101 if (slash_pos != NULL) {
102 *slash_pos = 0;
103 }
104 }
105
106 free(path);
107
108 if (hc_found && addr_found) {
109 if (out_hc_handle != NULL) {
110 *out_hc_handle = hc;
111 }
112 if (out_device_address != NULL) {
113 *out_device_address = addr;
114 }
115 return true;
116 } else {
117 return false;
118 }
119}
120
121static void print_usage(char *app_name)
122{
123#define _INDENT " "
124#define _OPTION(opt, description) \
125 printf(_INDENT opt "\n" _INDENT _INDENT description "\n")
126
127 printf(NAME ": query USB devices for descriptors\n\n");
128 printf("Usage: %s [options] device [device [device [ ... ]]]\n",
129 app_name);
130 printf(_INDENT "The device is a devman path to the device.\n");
131
132 _OPTION("-h --help", "Print this help and exit.");
133 _OPTION("-i --identification", "Brief device identification.");
134 _OPTION("-m --match-ids", "Print match ids generated for the device.");
135 _OPTION("-t --descriptor-tree", "Print descriptor tree.");
136 _OPTION("-T --descriptor-tree-full", "Print detailed descriptor tree");
137 _OPTION("-s --strings", "Try to print all string descriptors.");
138 _OPTION("-S --status", "Get status of the device.");
139
140 printf("\n");
141 printf("If no option is specified, `-i' is considered default.\n");
142 printf("\n");
143
144#undef _OPTION
145#undef _INDENT
146}
147
148static struct option long_options[] = {
149 {"help", no_argument, NULL, 'h'},
150 {"identification", no_argument, NULL, 'i'},
151 {"match-ids", no_argument, NULL, 'm'},
152 {"descriptor-tree", no_argument, NULL, 't'},
153 {"descriptor-tree-full", no_argument, NULL, 'T'},
154 {"strings", no_argument, NULL, 's'},
155 {"status", no_argument, NULL, 'S'},
156 {0, 0, NULL, 0}
157};
158static const char *short_options = "himtTsS";
159
160static usbinfo_action_t actions[] = {
161 {
162 .opt = 'i',
163 .action = dump_short_device_identification,
164 .active = false
165 },
166 {
167 .opt = 'm',
168 .action = dump_device_match_ids,
169 .active = false
170 },
171 {
172 .opt = 't',
173 .action = dump_descriptor_tree_brief,
174 .active = false
175 },
176 {
177 .opt = 'T',
178 .action = dump_descriptor_tree_full,
179 .active = false
180 },
181 {
182 .opt = 's',
183 .action = dump_strings,
184 .active = false
185 },
186 {
187 .opt = 'S',
188 .action = dump_status,
189 .active = false
190 },
191 {
192 .opt = 0
193 }
194};
195
196int main(int argc, char *argv[])
197{
198 if (argc <= 1) {
199 print_usage(argv[0]);
200 return -1;
201 }
202
203 /*
204 * Process command-line options. They determine what shall be
205 * done with the device.
206 */
207 int opt;
208 do {
209 opt = getopt_long(argc, argv,
210 short_options, long_options, NULL);
211 switch (opt) {
212 case -1:
213 break;
214 case '?':
215 print_usage(argv[0]);
216 return 1;
217 case 'h':
218 print_usage(argv[0]);
219 return 0;
220 default: {
221 int idx = 0;
222 while (actions[idx].opt != 0) {
223 if (actions[idx].opt == opt) {
224 actions[idx].active = true;
225 break;
226 }
227 idx++;
228 }
229 break;
230 }
231 }
232 } while (opt > 0);
233
234 /* Set the default action. */
235 int idx = 0;
236 bool something_active = false;
237 while (actions[idx].opt != 0) {
238 if (actions[idx].active) {
239 something_active = true;
240 break;
241 }
242 idx++;
243 }
244 if (!something_active) {
245 actions[0].active = true;
246 }
247
248 /*
249 * Go through all devices given on the command line and run the
250 * specified actions.
251 */
252 int i;
253 for (i = optind; i < argc; i++) {
254 char *devpath = argv[i];
255
256 /* The initialization is here only to make compiler happy. */
257 devman_handle_t hc_handle = 0;
258 usb_address_t dev_addr = 0;
259 bool found = resolve_hc_handle_and_dev_addr(devpath,
260 &hc_handle, &dev_addr);
261 if (!found) {
262 fprintf(stderr, NAME ": device `%s' not found "
263 "or not of USB kind, skipping.\n",
264 devpath);
265 continue;
266 }
267
268 usbinfo_device_t *dev = prepare_device(hc_handle, dev_addr);
269 if (dev == NULL) {
270 continue;
271 }
272
273 /* Run actions the user specified. */
274 printf("%s\n", devpath);
275
276 int action = 0;
277 while (actions[action].opt != 0) {
278 if (actions[action].active) {
279 actions[action].action(dev);
280 }
281 action++;
282 }
283
284 /* Destroy the control pipe (close the session etc.). */
285 destroy_device(dev);
286 }
287
288 return 0;
289}
290
291
292/** @}
293 */
Note: See TracBrowser for help on using the repository browser.