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

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

usbinfo can print (raw) HID report descriptor

  • Property mode set to 100644
File size: 5.5 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/hc.h>
46#include <usb/dev/pipes.h>
47#include "usbinfo.h"
48
49static void print_usage(char *app_name)
50{
51#define _INDENT " "
52#define _OPTION(opt, description) \
53 printf(_INDENT opt "\n" _INDENT _INDENT description "\n")
54
55 printf(NAME ": query USB devices for descriptors\n\n");
56 printf("Usage: %s [options] device [device [device [ ... ]]]\n",
57 app_name);
58 printf(_INDENT "The device is a devman path to the device.\n");
59
60 _OPTION("-h --help", "Print this help and exit.");
61 _OPTION("-i --identification", "Brief device identification.");
62 _OPTION("-m --match-ids", "Print match ids generated for the device.");
63 _OPTION("-t --descriptor-tree", "Print descriptor tree.");
64 _OPTION("-T --descriptor-tree-full", "Print detailed descriptor tree");
65 _OPTION("-s --strings", "Try to print all string descriptors.");
66 _OPTION("-S --status", "Get status of the device.");
67 _OPTION("-r --hid-report", "Dump HID report descriptor.");
68
69 printf("\n");
70 printf("If no option is specified, `-i' is considered default.\n");
71 printf("\n");
72
73#undef _OPTION
74#undef _INDENT
75}
76
77static struct option long_options[] = {
78 {"help", no_argument, NULL, 'h'},
79 {"identification", no_argument, NULL, 'i'},
80 {"match-ids", no_argument, NULL, 'm'},
81 {"descriptor-tree", no_argument, NULL, 't'},
82 {"descriptor-tree-full", no_argument, NULL, 'T'},
83 {"strings", no_argument, NULL, 's'},
84 {"status", no_argument, NULL, 'S'},
85 {"hid-report", no_argument, NULL, 'r'},
86 {0, 0, NULL, 0}
87};
88static const char *short_options = "himtTsSr";
89
90static usbinfo_action_t actions[] = {
91 {
92 .opt = 'i',
93 .action = dump_short_device_identification,
94 .active = false
95 },
96 {
97 .opt = 'm',
98 .action = dump_device_match_ids,
99 .active = false
100 },
101 {
102 .opt = 't',
103 .action = dump_descriptor_tree_brief,
104 .active = false
105 },
106 {
107 .opt = 'T',
108 .action = dump_descriptor_tree_full,
109 .active = false
110 },
111 {
112 .opt = 's',
113 .action = dump_strings,
114 .active = false
115 },
116 {
117 .opt = 'S',
118 .action = dump_status,
119 .active = false
120 },
121 {
122 .opt = 'r',
123 .action = dump_hidreport,
124 .active = false
125 },
126 {
127 .opt = 0
128 }
129};
130
131int main(int argc, char *argv[])
132{
133 if (argc <= 1) {
134 print_usage(argv[0]);
135 return -1;
136 }
137
138 /*
139 * Process command-line options. They determine what shall be
140 * done with the device.
141 */
142 int opt;
143 do {
144 opt = getopt_long(argc, argv,
145 short_options, long_options, NULL);
146 switch (opt) {
147 case -1:
148 break;
149 case '?':
150 print_usage(argv[0]);
151 return 1;
152 case 'h':
153 print_usage(argv[0]);
154 return 0;
155 default: {
156 int idx = 0;
157 while (actions[idx].opt != 0) {
158 if (actions[idx].opt == opt) {
159 actions[idx].active = true;
160 break;
161 }
162 idx++;
163 }
164 break;
165 }
166 }
167 } while (opt > 0);
168
169 /* Set the default action. */
170 int idx = 0;
171 bool something_active = false;
172 while (actions[idx].opt != 0) {
173 if (actions[idx].active) {
174 something_active = true;
175 break;
176 }
177 idx++;
178 }
179 if (!something_active) {
180 actions[0].active = true;
181 }
182
183 /*
184 * Go through all devices given on the command line and run the
185 * specified actions.
186 */
187 int i;
188 for (i = optind; i < argc; i++) {
189 char *devpath = argv[i];
190
191 /* The initialization is here only to make compiler happy. */
192 devman_handle_t hc_handle = 0;
193 usb_address_t dev_addr = 0;
194 int rc = usb_resolve_device_handle(devpath,
195 &hc_handle, &dev_addr, NULL);
196 if (rc != EOK) {
197 fprintf(stderr, NAME ": device `%s' not found "
198 "or not of USB kind, skipping.\n",
199 devpath);
200 continue;
201 }
202
203 usbinfo_device_t *dev = prepare_device(devpath,
204 hc_handle, dev_addr);
205 if (dev == NULL) {
206 continue;
207 }
208
209 /* Run actions the user specified. */
210 printf("%s\n", devpath);
211
212 int action = 0;
213 while (actions[action].opt != 0) {
214 if (actions[action].active) {
215 actions[action].action(dev);
216 }
217 action++;
218 }
219
220 /* Destroy the control pipe (close the session etc.). */
221 destroy_device(dev);
222 }
223
224 return 0;
225}
226
227
228/** @}
229 */
Note: See TracBrowser for help on using the repository browser.