source: mainline/uspace/app/usbinfo/main.c@ 7b13d8e

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

Add printing of descriptor tree to usbinfo

  • Property mode set to 100644
File size: 6.2 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
137 printf("\n");
138 printf("If no option is specified, `-i' is considered default.\n");
139 printf("\n");
140
141#undef _OPTION
142#undef _INDENT
143}
144
145static struct option long_options[] = {
146 {"help", no_argument, NULL, 'h'},
147 {"identification", no_argument, NULL, 'i'},
148 {"match-ids", no_argument, NULL, 'm'},
149 {"descriptor-tree", no_argument, NULL, 't'},
150 {0, 0, NULL, 0}
151};
152static const char *short_options = "himt";
153
154int main(int argc, char *argv[])
155{
156 if (argc <= 1) {
157 print_usage(argv[0]);
158 return -1;
159 }
160
161 bool action_print_short_identification = false;
162 bool action_print_match_ids = false;
163 bool action_print_descriptor_tree = false;
164
165 /*
166 * Process command-line options. They determine what shall be
167 * done with the device.
168 */
169 int opt;
170 do {
171 opt = getopt_long(argc, argv,
172 short_options, long_options, NULL);
173 switch (opt) {
174 case -1:
175 break;
176 case '?':
177 print_usage(argv[0]);
178 return 1;
179 case 'h':
180 print_usage(argv[0]);
181 return 0;
182 case 'i':
183 action_print_short_identification = true;
184 break;
185 case 'm':
186 action_print_match_ids = true;
187 break;
188 case 't':
189 action_print_descriptor_tree = true;
190 break;
191 default:
192 assert(false && "unreachable code");
193 break;
194 }
195 } while (opt > 0);
196
197 /* Set the default action. */
198 if (!action_print_match_ids
199 && !action_print_short_identification
200 && !action_print_descriptor_tree) {
201 action_print_short_identification = true;
202 }
203
204 /*
205 * Go through all devices given on the command line and run the
206 * specified actions.
207 */
208 int i;
209 for (i = optind; i < argc; i++) {
210 char *devpath = argv[i];
211
212 /* The initialization is here only to make compiler happy. */
213 devman_handle_t hc_handle = 0;
214 usb_address_t dev_addr = 0;
215 bool found = resolve_hc_handle_and_dev_addr(devpath,
216 &hc_handle, &dev_addr);
217 if (!found) {
218 fprintf(stderr, NAME ": device `%s' not found "
219 "or not of USB kind, skipping.\n",
220 devpath);
221 continue;
222 }
223
224 usbinfo_device_t *dev = prepare_device(hc_handle, dev_addr);
225 if (dev == NULL) {
226 continue;
227 }
228
229 /* Run actions the user specified. */
230 printf("%s\n", devpath);
231
232 if (action_print_short_identification) {
233 dump_short_device_identification(dev);
234 }
235 if (action_print_match_ids) {
236 dump_device_match_ids(dev);
237 }
238 if (action_print_descriptor_tree) {
239 dump_descriptor_tree_brief(dev);
240 }
241
242 /* Destroy the control pipe (close the session etc.). */
243 destroy_device(dev);
244 }
245
246 return 0;
247}
248
249
250/** @}
251 */
Note: See TracBrowser for help on using the repository browser.