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

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

Add back full descriptor dump to usbinfo

  • Property mode set to 100644
File size: 6.6 KB
RevLine 
[07b9203e]1/*
[c377bc50]2 * Copyright (c) 2010-2011 Vojtech Horky
[07b9203e]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
[632ed68]29/** @addtogroup usbinfo
[07b9203e]30 * @{
31 */
32/**
33 * @file
[632ed68]34 * USB querying.
[07b9203e]35 */
36
37#include <stdio.h>
38#include <stdlib.h>
39#include <errno.h>
40#include <str_error.h>
41#include <bool.h>
[c377bc50]42#include <getopt.h>
[07b9203e]43#include <devman.h>
[c377bc50]44#include <devmap.h>
[7ffe82f]45#include <usb/usbdevice.h>
46#include <usb/pipes.h>
[07b9203e]47#include "usbinfo.h"
48
[7ffe82f]49static bool resolve_hc_handle_and_dev_addr(const char *devpath,
50 devman_handle_t *out_hc_handle, usb_address_t *out_device_address)
[07b9203e]51{
52 int rc;
53
[2180979]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
[7ffe82f]59 char *path = str_dup(devpath);
60 if (path == NULL) {
61 return ENOMEM;
[608afb9]62 }
63
[7ffe82f]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 }
[07b9203e]78
[7ffe82f]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 }
[07b9203e]93
[7ffe82f]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 }
[07b9203e]104 }
105
[7ffe82f]106 free(path);
[07b9203e]107
[7ffe82f]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}
[07b9203e]120
[e160da4d]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.");
[e387d0f]136 _OPTION("-T --descriptor-tree-full", "Print detailed descriptor tree");
[aad3587]137 _OPTION("-s --strings", "Try to print all string descriptors.");
[e160da4d]138
139 printf("\n");
140 printf("If no option is specified, `-i' is considered default.\n");
141 printf("\n");
142
143#undef _OPTION
144#undef _INDENT
145}
146
[81ca204]147static struct option long_options[] = {
148 {"help", no_argument, NULL, 'h'},
149 {"identification", no_argument, NULL, 'i'},
150 {"match-ids", no_argument, NULL, 'm'},
[e160da4d]151 {"descriptor-tree", no_argument, NULL, 't'},
[e387d0f]152 {"descriptor-tree-full", no_argument, NULL, 'T'},
[aad3587]153 {"strings", no_argument, NULL, 's'},
[81ca204]154 {0, 0, NULL, 0}
155};
[e387d0f]156static const char *short_options = "himtTs";
[81ca204]157
[a458bc9]158static usbinfo_action_t actions[] = {
159 {
160 .opt = 'i',
161 .action = dump_short_device_identification,
162 .active = false
163 },
164 {
165 .opt = 'm',
166 .action = dump_device_match_ids,
167 .active = false
168 },
169 {
170 .opt = 't',
171 .action = dump_descriptor_tree_brief,
172 .active = false
173 },
[e387d0f]174 {
175 .opt = 'T',
176 .action = dump_descriptor_tree_full,
177 .active = false
178 },
[a458bc9]179 {
180 .opt = 's',
181 .action = dump_strings,
182 .active = false
183 },
184 {
185 .opt = 0
186 }
187};
188
[c377bc50]189int main(int argc, char *argv[])
190{
191 if (argc <= 1) {
192 print_usage(argv[0]);
193 return -1;
[2c5cefa]194 }
[07b9203e]195
[7ffe82f]196 /*
197 * Process command-line options. They determine what shall be
198 * done with the device.
199 */
[81ca204]200 int opt;
201 do {
202 opt = getopt_long(argc, argv,
203 short_options, long_options, NULL);
204 switch (opt) {
205 case -1:
206 break;
207 case '?':
208 print_usage(argv[0]);
209 return 1;
210 case 'h':
211 print_usage(argv[0]);
212 return 0;
[a458bc9]213 default: {
214 int idx = 0;
215 while (actions[idx].opt != 0) {
216 if (actions[idx].opt == opt) {
217 actions[idx].active = true;
218 break;
219 }
220 idx++;
221 }
[81ca204]222 break;
[a458bc9]223 }
[81ca204]224 }
225 } while (opt > 0);
[c377bc50]226
[81ca204]227 /* Set the default action. */
[a458bc9]228 int idx = 0;
229 bool something_active = false;
230 while (actions[idx].opt != 0) {
231 if (actions[idx].active) {
232 something_active = true;
233 break;
234 }
235 idx++;
236 }
237 if (!something_active) {
238 actions[0].active = true;
[81ca204]239 }
[c377bc50]240
[7ffe82f]241 /*
242 * Go through all devices given on the command line and run the
243 * specified actions.
244 */
245 int i;
[81ca204]246 for (i = optind; i < argc; i++) {
[7ffe82f]247 char *devpath = argv[i];
248
249 /* The initialization is here only to make compiler happy. */
250 devman_handle_t hc_handle = 0;
251 usb_address_t dev_addr = 0;
252 bool found = resolve_hc_handle_and_dev_addr(devpath,
253 &hc_handle, &dev_addr);
254 if (!found) {
255 fprintf(stderr, NAME ": device `%s' not found "
256 "or not of USB kind, skipping.\n",
257 devpath);
258 continue;
[c377bc50]259 }
260
[3100ebe]261 usbinfo_device_t *dev = prepare_device(hc_handle, dev_addr);
262 if (dev == NULL) {
263 continue;
264 }
265
266 /* Run actions the user specified. */
267 printf("%s\n", devpath);
[81ca204]268
[a458bc9]269 int action = 0;
270 while (actions[action].opt != 0) {
271 if (actions[action].active) {
272 actions[action].action(dev);
273 }
274 action++;
[aad3587]275 }
[3100ebe]276
277 /* Destroy the control pipe (close the session etc.). */
278 destroy_device(dev);
[a12917e]279 }
280
[c377bc50]281 return 0;
[07b9203e]282}
283
284
285/** @}
286 */
Note: See TracBrowser for help on using the repository browser.