xref: /aosp_15_r20/external/libusb/android/examples/unrooted_android.c (revision 86b64dcb59b3a0b37502ecd56e119234366a6f7e)
1*86b64dcbSAndroid Build Coastguard Worker /*
2*86b64dcbSAndroid Build Coastguard Worker  *  libusb example program for reading out USB descriptors on unrooted Android
3*86b64dcbSAndroid Build Coastguard Worker  *  (based on testlibusb.c)
4*86b64dcbSAndroid Build Coastguard Worker  *
5*86b64dcbSAndroid Build Coastguard Worker  *  Copyright 2020-2021 Peter Stoiber
6*86b64dcbSAndroid Build Coastguard Worker  *
7*86b64dcbSAndroid Build Coastguard Worker  *  This library is free software; you can redistribute it and/or
8*86b64dcbSAndroid Build Coastguard Worker  *  modify it under the terms of the GNU Lesser General Public
9*86b64dcbSAndroid Build Coastguard Worker  *  License as published by the Free Software Foundation; either
10*86b64dcbSAndroid Build Coastguard Worker  *  version 2.1 of the License, or (at your option) any later version.
11*86b64dcbSAndroid Build Coastguard Worker  *
12*86b64dcbSAndroid Build Coastguard Worker  *  This library is distributed in the hope that it will be useful,
13*86b64dcbSAndroid Build Coastguard Worker  *  but WITHOUT ANY WARRANTY; without even the implied warranty of
14*86b64dcbSAndroid Build Coastguard Worker  *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
15*86b64dcbSAndroid Build Coastguard Worker  *  Lesser General Public License for more details.
16*86b64dcbSAndroid Build Coastguard Worker  *
17*86b64dcbSAndroid Build Coastguard Worker  *  You should have received a copy of the GNU Lesser General Public
18*86b64dcbSAndroid Build Coastguard Worker  *  License along with this library; if not, write to the Free Software
19*86b64dcbSAndroid Build Coastguard Worker  *  Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA
20*86b64dcbSAndroid Build Coastguard Worker  *
21*86b64dcbSAndroid Build Coastguard Worker  *  Please contact the author if you need another license.
22*86b64dcbSAndroid Build Coastguard Worker  *  This Repository is provided "as is", without warranties of any kind.
23*86b64dcbSAndroid Build Coastguard Worker */
24*86b64dcbSAndroid Build Coastguard Worker 
25*86b64dcbSAndroid Build Coastguard Worker /*
26*86b64dcbSAndroid Build Coastguard Worker  * This example creates a shared object which can be accessed over JNA or JNI from Java or Kotlin in Android.
27*86b64dcbSAndroid Build Coastguard Worker  * Hint: If you are using Android Studio, set the "Debug type" to "Java Only" to receive debug messages.
28*86b64dcbSAndroid Build Coastguard Worker  */
29*86b64dcbSAndroid Build Coastguard Worker 
30*86b64dcbSAndroid Build Coastguard Worker /*
31*86b64dcbSAndroid Build Coastguard Worker  * Usage:
32*86b64dcbSAndroid Build Coastguard Worker  * First, you have to connect your USB device from the Java side.
33*86b64dcbSAndroid Build Coastguard Worker  * Use the android.hardware.usb class to find the USB device, claim the interfaces, and open the usb_device_connection
34*86b64dcbSAndroid Build Coastguard Worker  * Obtain the native File Descriptor --> usb_device_connection.getFileDescriptor()
35*86b64dcbSAndroid Build Coastguard Worker  * Pass the received int value to the unrooted_usb_description method of this code (over JNA)
36*86b64dcbSAndroid Build Coastguard Worker  */
37*86b64dcbSAndroid Build Coastguard Worker 
38*86b64dcbSAndroid Build Coastguard Worker /*
39*86b64dcbSAndroid Build Coastguard Worker  * libusb can only be included in Android projects using NDK for now. (CMake is not supported at the moment)
40*86b64dcbSAndroid Build Coastguard Worker  * Clone the libusb git repo into your Android project and include the Android.mk file in your build.gradle.
41*86b64dcbSAndroid Build Coastguard Worker  */
42*86b64dcbSAndroid Build Coastguard Worker 
43*86b64dcbSAndroid Build Coastguard Worker /*
44*86b64dcbSAndroid Build Coastguard Worker  Example JNA Approach:
45*86b64dcbSAndroid Build Coastguard Worker     public interface unrooted_sample extends Library {
46*86b64dcbSAndroid Build Coastguard Worker         public static final unrooted_sample INSTANCE = Native.load("unrooted_android", unrooted_sample.class);
47*86b64dcbSAndroid Build Coastguard Worker         public int unrooted_usb_description (int fileDescriptor);
48*86b64dcbSAndroid Build Coastguard Worker     }
49*86b64dcbSAndroid Build Coastguard Worker     unrooted_sample.INSTANCE.unrooted_usb_description( usbDeviceConnection.getFileDescriptor());
50*86b64dcbSAndroid Build Coastguard Worker  */
51*86b64dcbSAndroid Build Coastguard Worker 
52*86b64dcbSAndroid Build Coastguard Worker #include <jni.h>
53*86b64dcbSAndroid Build Coastguard Worker #include <string.h>
54*86b64dcbSAndroid Build Coastguard Worker #include "unrooted_android.h"
55*86b64dcbSAndroid Build Coastguard Worker #include "libusb.h"
56*86b64dcbSAndroid Build Coastguard Worker #include <android/log.h>
57*86b64dcbSAndroid Build Coastguard Worker #define  LOG_TAG    "LibUsb"
58*86b64dcbSAndroid Build Coastguard Worker #define  LOGD(...)  __android_log_print(ANDROID_LOG_DEBUG, LOG_TAG, __VA_ARGS__)
59*86b64dcbSAndroid Build Coastguard Worker 
60*86b64dcbSAndroid Build Coastguard Worker int verbose = 0;
61*86b64dcbSAndroid Build Coastguard Worker 
print_endpoint_comp(const struct libusb_ss_endpoint_companion_descriptor * ep_comp)62*86b64dcbSAndroid Build Coastguard Worker static void print_endpoint_comp(const struct libusb_ss_endpoint_companion_descriptor *ep_comp)
63*86b64dcbSAndroid Build Coastguard Worker {
64*86b64dcbSAndroid Build Coastguard Worker     LOGD("      USB 3.0 Endpoint Companion:\n");
65*86b64dcbSAndroid Build Coastguard Worker     LOGD("        bMaxBurst:           %u\n", ep_comp->bMaxBurst);
66*86b64dcbSAndroid Build Coastguard Worker     LOGD("        bmAttributes:        %02xh\n", ep_comp->bmAttributes);
67*86b64dcbSAndroid Build Coastguard Worker     LOGD("        wBytesPerInterval:   %u\n", ep_comp->wBytesPerInterval);
68*86b64dcbSAndroid Build Coastguard Worker }
69*86b64dcbSAndroid Build Coastguard Worker 
print_endpoint(const struct libusb_endpoint_descriptor * endpoint)70*86b64dcbSAndroid Build Coastguard Worker static void print_endpoint(const struct libusb_endpoint_descriptor *endpoint)
71*86b64dcbSAndroid Build Coastguard Worker {
72*86b64dcbSAndroid Build Coastguard Worker     int i, ret;
73*86b64dcbSAndroid Build Coastguard Worker 
74*86b64dcbSAndroid Build Coastguard Worker     LOGD("      Endpoint:\n");
75*86b64dcbSAndroid Build Coastguard Worker     LOGD("        bEndpointAddress:    %02xh\n", endpoint->bEndpointAddress);
76*86b64dcbSAndroid Build Coastguard Worker     LOGD("        bmAttributes:        %02xh\n", endpoint->bmAttributes);
77*86b64dcbSAndroid Build Coastguard Worker     LOGD("        wMaxPacketSize:      %u\n", endpoint->wMaxPacketSize);
78*86b64dcbSAndroid Build Coastguard Worker     LOGD("        bInterval:           %u\n", endpoint->bInterval);
79*86b64dcbSAndroid Build Coastguard Worker     LOGD("        bRefresh:            %u\n", endpoint->bRefresh);
80*86b64dcbSAndroid Build Coastguard Worker     LOGD("        bSynchAddress:       %u\n", endpoint->bSynchAddress);
81*86b64dcbSAndroid Build Coastguard Worker 
82*86b64dcbSAndroid Build Coastguard Worker     for (i = 0; i < endpoint->extra_length;) {
83*86b64dcbSAndroid Build Coastguard Worker         if (LIBUSB_DT_SS_ENDPOINT_COMPANION == endpoint->extra[i + 1]) {
84*86b64dcbSAndroid Build Coastguard Worker             struct libusb_ss_endpoint_companion_descriptor *ep_comp;
85*86b64dcbSAndroid Build Coastguard Worker 
86*86b64dcbSAndroid Build Coastguard Worker             ret = libusb_get_ss_endpoint_companion_descriptor(NULL, endpoint, &ep_comp);
87*86b64dcbSAndroid Build Coastguard Worker             if (LIBUSB_SUCCESS != ret)
88*86b64dcbSAndroid Build Coastguard Worker                 continue;
89*86b64dcbSAndroid Build Coastguard Worker 
90*86b64dcbSAndroid Build Coastguard Worker             print_endpoint_comp(ep_comp);
91*86b64dcbSAndroid Build Coastguard Worker 
92*86b64dcbSAndroid Build Coastguard Worker             libusb_free_ss_endpoint_companion_descriptor(ep_comp);
93*86b64dcbSAndroid Build Coastguard Worker         }
94*86b64dcbSAndroid Build Coastguard Worker 
95*86b64dcbSAndroid Build Coastguard Worker         i += endpoint->extra[i];
96*86b64dcbSAndroid Build Coastguard Worker     }
97*86b64dcbSAndroid Build Coastguard Worker }
98*86b64dcbSAndroid Build Coastguard Worker 
print_altsetting(const struct libusb_interface_descriptor * interface)99*86b64dcbSAndroid Build Coastguard Worker static void print_altsetting(const struct libusb_interface_descriptor *interface)
100*86b64dcbSAndroid Build Coastguard Worker {
101*86b64dcbSAndroid Build Coastguard Worker     uint8_t i;
102*86b64dcbSAndroid Build Coastguard Worker 
103*86b64dcbSAndroid Build Coastguard Worker     LOGD("    Interface:\n");
104*86b64dcbSAndroid Build Coastguard Worker     LOGD("      bInterfaceNumber:      %u\n", interface->bInterfaceNumber);
105*86b64dcbSAndroid Build Coastguard Worker     LOGD("      bAlternateSetting:     %u\n", interface->bAlternateSetting);
106*86b64dcbSAndroid Build Coastguard Worker     LOGD("      bNumEndpoints:         %u\n", interface->bNumEndpoints);
107*86b64dcbSAndroid Build Coastguard Worker     LOGD("      bInterfaceClass:       %u\n", interface->bInterfaceClass);
108*86b64dcbSAndroid Build Coastguard Worker     LOGD("      bInterfaceSubClass:    %u\n", interface->bInterfaceSubClass);
109*86b64dcbSAndroid Build Coastguard Worker     LOGD("      bInterfaceProtocol:    %u\n", interface->bInterfaceProtocol);
110*86b64dcbSAndroid Build Coastguard Worker     LOGD("      iInterface:            %u\n", interface->iInterface);
111*86b64dcbSAndroid Build Coastguard Worker 
112*86b64dcbSAndroid Build Coastguard Worker     for (i = 0; i < interface->bNumEndpoints; i++)
113*86b64dcbSAndroid Build Coastguard Worker         print_endpoint(&interface->endpoint[i]);
114*86b64dcbSAndroid Build Coastguard Worker }
115*86b64dcbSAndroid Build Coastguard Worker 
print_2_0_ext_cap(struct libusb_usb_2_0_extension_descriptor * usb_2_0_ext_cap)116*86b64dcbSAndroid Build Coastguard Worker static void print_2_0_ext_cap(struct libusb_usb_2_0_extension_descriptor *usb_2_0_ext_cap)
117*86b64dcbSAndroid Build Coastguard Worker {
118*86b64dcbSAndroid Build Coastguard Worker     LOGD("    USB 2.0 Extension Capabilities:\n");
119*86b64dcbSAndroid Build Coastguard Worker     LOGD("      bDevCapabilityType:    %u\n", usb_2_0_ext_cap->bDevCapabilityType);
120*86b64dcbSAndroid Build Coastguard Worker     LOGD("      bmAttributes:          %08xh\n", usb_2_0_ext_cap->bmAttributes);
121*86b64dcbSAndroid Build Coastguard Worker }
122*86b64dcbSAndroid Build Coastguard Worker 
print_ss_usb_cap(struct libusb_ss_usb_device_capability_descriptor * ss_usb_cap)123*86b64dcbSAndroid Build Coastguard Worker static void print_ss_usb_cap(struct libusb_ss_usb_device_capability_descriptor *ss_usb_cap)
124*86b64dcbSAndroid Build Coastguard Worker {
125*86b64dcbSAndroid Build Coastguard Worker     LOGD("    USB 3.0 Capabilities:\n");
126*86b64dcbSAndroid Build Coastguard Worker     LOGD("      bDevCapabilityType:    %u\n", ss_usb_cap->bDevCapabilityType);
127*86b64dcbSAndroid Build Coastguard Worker     LOGD("      bmAttributes:          %02xh\n", ss_usb_cap->bmAttributes);
128*86b64dcbSAndroid Build Coastguard Worker     LOGD("      wSpeedSupported:       %u\n", ss_usb_cap->wSpeedSupported);
129*86b64dcbSAndroid Build Coastguard Worker     LOGD("      bFunctionalitySupport: %u\n", ss_usb_cap->bFunctionalitySupport);
130*86b64dcbSAndroid Build Coastguard Worker     LOGD("      bU1devExitLat:         %u\n", ss_usb_cap->bU1DevExitLat);
131*86b64dcbSAndroid Build Coastguard Worker     LOGD("      bU2devExitLat:         %u\n", ss_usb_cap->bU2DevExitLat);
132*86b64dcbSAndroid Build Coastguard Worker }
133*86b64dcbSAndroid Build Coastguard Worker 
print_bos(libusb_device_handle * handle)134*86b64dcbSAndroid Build Coastguard Worker static void print_bos(libusb_device_handle *handle)
135*86b64dcbSAndroid Build Coastguard Worker {
136*86b64dcbSAndroid Build Coastguard Worker     struct libusb_bos_descriptor *bos;
137*86b64dcbSAndroid Build Coastguard Worker     uint8_t i;
138*86b64dcbSAndroid Build Coastguard Worker     int ret;
139*86b64dcbSAndroid Build Coastguard Worker 
140*86b64dcbSAndroid Build Coastguard Worker     ret = libusb_get_bos_descriptor(handle, &bos);
141*86b64dcbSAndroid Build Coastguard Worker     if (ret < 0)
142*86b64dcbSAndroid Build Coastguard Worker         return;
143*86b64dcbSAndroid Build Coastguard Worker 
144*86b64dcbSAndroid Build Coastguard Worker     LOGD("  Binary Object Store (BOS):\n");
145*86b64dcbSAndroid Build Coastguard Worker     LOGD("    wTotalLength:            %u\n", bos->wTotalLength);
146*86b64dcbSAndroid Build Coastguard Worker     LOGD("    bNumDeviceCaps:          %u\n", bos->bNumDeviceCaps);
147*86b64dcbSAndroid Build Coastguard Worker 
148*86b64dcbSAndroid Build Coastguard Worker     for (i = 0; i < bos->bNumDeviceCaps; i++) {
149*86b64dcbSAndroid Build Coastguard Worker         struct libusb_bos_dev_capability_descriptor *dev_cap = bos->dev_capability[i];
150*86b64dcbSAndroid Build Coastguard Worker 
151*86b64dcbSAndroid Build Coastguard Worker         if (dev_cap->bDevCapabilityType == LIBUSB_BT_USB_2_0_EXTENSION) {
152*86b64dcbSAndroid Build Coastguard Worker             struct libusb_usb_2_0_extension_descriptor *usb_2_0_extension;
153*86b64dcbSAndroid Build Coastguard Worker 
154*86b64dcbSAndroid Build Coastguard Worker             ret = libusb_get_usb_2_0_extension_descriptor(NULL, dev_cap, &usb_2_0_extension);
155*86b64dcbSAndroid Build Coastguard Worker             if (ret < 0)
156*86b64dcbSAndroid Build Coastguard Worker                 return;
157*86b64dcbSAndroid Build Coastguard Worker 
158*86b64dcbSAndroid Build Coastguard Worker             print_2_0_ext_cap(usb_2_0_extension);
159*86b64dcbSAndroid Build Coastguard Worker             libusb_free_usb_2_0_extension_descriptor(usb_2_0_extension);
160*86b64dcbSAndroid Build Coastguard Worker         } else if (dev_cap->bDevCapabilityType == LIBUSB_BT_SS_USB_DEVICE_CAPABILITY) {
161*86b64dcbSAndroid Build Coastguard Worker             struct libusb_ss_usb_device_capability_descriptor *ss_dev_cap;
162*86b64dcbSAndroid Build Coastguard Worker 
163*86b64dcbSAndroid Build Coastguard Worker             ret = libusb_get_ss_usb_device_capability_descriptor(NULL, dev_cap, &ss_dev_cap);
164*86b64dcbSAndroid Build Coastguard Worker             if (ret < 0)
165*86b64dcbSAndroid Build Coastguard Worker                 return;
166*86b64dcbSAndroid Build Coastguard Worker 
167*86b64dcbSAndroid Build Coastguard Worker             print_ss_usb_cap(ss_dev_cap);
168*86b64dcbSAndroid Build Coastguard Worker             libusb_free_ss_usb_device_capability_descriptor(ss_dev_cap);
169*86b64dcbSAndroid Build Coastguard Worker         }
170*86b64dcbSAndroid Build Coastguard Worker     }
171*86b64dcbSAndroid Build Coastguard Worker 
172*86b64dcbSAndroid Build Coastguard Worker     libusb_free_bos_descriptor(bos);
173*86b64dcbSAndroid Build Coastguard Worker }
174*86b64dcbSAndroid Build Coastguard Worker 
print_interface(const struct libusb_interface * interface)175*86b64dcbSAndroid Build Coastguard Worker static void print_interface(const struct libusb_interface *interface)
176*86b64dcbSAndroid Build Coastguard Worker {
177*86b64dcbSAndroid Build Coastguard Worker     int i;
178*86b64dcbSAndroid Build Coastguard Worker 
179*86b64dcbSAndroid Build Coastguard Worker     for (i = 0; i < interface->num_altsetting; i++)
180*86b64dcbSAndroid Build Coastguard Worker         print_altsetting(&interface->altsetting[i]);
181*86b64dcbSAndroid Build Coastguard Worker }
182*86b64dcbSAndroid Build Coastguard Worker 
print_configuration(struct libusb_config_descriptor * config)183*86b64dcbSAndroid Build Coastguard Worker static void print_configuration(struct libusb_config_descriptor *config)
184*86b64dcbSAndroid Build Coastguard Worker {
185*86b64dcbSAndroid Build Coastguard Worker     uint8_t i;
186*86b64dcbSAndroid Build Coastguard Worker 
187*86b64dcbSAndroid Build Coastguard Worker     LOGD("  Configuration:\n");
188*86b64dcbSAndroid Build Coastguard Worker     LOGD("    wTotalLength:            %u\n", config->wTotalLength);
189*86b64dcbSAndroid Build Coastguard Worker     LOGD("    bNumInterfaces:          %u\n", config->bNumInterfaces);
190*86b64dcbSAndroid Build Coastguard Worker     LOGD("    bConfigurationValue:     %u\n", config->bConfigurationValue);
191*86b64dcbSAndroid Build Coastguard Worker     LOGD("    iConfiguration:          %u\n", config->iConfiguration);
192*86b64dcbSAndroid Build Coastguard Worker     LOGD("    bmAttributes:            %02xh\n", config->bmAttributes);
193*86b64dcbSAndroid Build Coastguard Worker     LOGD("    MaxPower:                %u\n", config->MaxPower);
194*86b64dcbSAndroid Build Coastguard Worker 
195*86b64dcbSAndroid Build Coastguard Worker     for (i = 0; i < config->bNumInterfaces; i++)
196*86b64dcbSAndroid Build Coastguard Worker         print_interface(&config->interface[i]);
197*86b64dcbSAndroid Build Coastguard Worker }
198*86b64dcbSAndroid Build Coastguard Worker 
print_device(libusb_device * dev,libusb_device_handle * handle)199*86b64dcbSAndroid Build Coastguard Worker static void print_device(libusb_device *dev, libusb_device_handle *handle)
200*86b64dcbSAndroid Build Coastguard Worker {
201*86b64dcbSAndroid Build Coastguard Worker     struct libusb_device_descriptor desc;
202*86b64dcbSAndroid Build Coastguard Worker     unsigned char string[256];
203*86b64dcbSAndroid Build Coastguard Worker     const char *speed;
204*86b64dcbSAndroid Build Coastguard Worker     int ret;
205*86b64dcbSAndroid Build Coastguard Worker     uint8_t i;
206*86b64dcbSAndroid Build Coastguard Worker 
207*86b64dcbSAndroid Build Coastguard Worker     switch (libusb_get_device_speed(dev)) {
208*86b64dcbSAndroid Build Coastguard Worker         case LIBUSB_SPEED_LOW:		speed = "1.5M"; break;
209*86b64dcbSAndroid Build Coastguard Worker         case LIBUSB_SPEED_FULL:		speed = "12M"; break;
210*86b64dcbSAndroid Build Coastguard Worker         case LIBUSB_SPEED_HIGH:		speed = "480M"; break;
211*86b64dcbSAndroid Build Coastguard Worker         case LIBUSB_SPEED_SUPER:	speed = "5G"; break;
212*86b64dcbSAndroid Build Coastguard Worker         case LIBUSB_SPEED_SUPER_PLUS:	speed = "10G"; break;
213*86b64dcbSAndroid Build Coastguard Worker         case LIBUSB_SPEED_SUPER_PLUS_X2:	speed = "20G"; break;
214*86b64dcbSAndroid Build Coastguard Worker         default:			speed = "Unknown";
215*86b64dcbSAndroid Build Coastguard Worker     }
216*86b64dcbSAndroid Build Coastguard Worker 
217*86b64dcbSAndroid Build Coastguard Worker     ret = libusb_get_device_descriptor(dev, &desc);
218*86b64dcbSAndroid Build Coastguard Worker     if (ret < 0) {
219*86b64dcbSAndroid Build Coastguard Worker         LOGD("failed to get device descriptor");
220*86b64dcbSAndroid Build Coastguard Worker         return;
221*86b64dcbSAndroid Build Coastguard Worker     }
222*86b64dcbSAndroid Build Coastguard Worker 
223*86b64dcbSAndroid Build Coastguard Worker     LOGD("Dev (bus %u, device %u): %04X - %04X speed: %s\n",
224*86b64dcbSAndroid Build Coastguard Worker            libusb_get_bus_number(dev), libusb_get_device_address(dev),
225*86b64dcbSAndroid Build Coastguard Worker            desc.idVendor, desc.idProduct, speed);
226*86b64dcbSAndroid Build Coastguard Worker 
227*86b64dcbSAndroid Build Coastguard Worker     if (!handle)
228*86b64dcbSAndroid Build Coastguard Worker         libusb_open(dev, &handle);
229*86b64dcbSAndroid Build Coastguard Worker 
230*86b64dcbSAndroid Build Coastguard Worker     if (handle) {
231*86b64dcbSAndroid Build Coastguard Worker         if (desc.iManufacturer) {
232*86b64dcbSAndroid Build Coastguard Worker             ret = libusb_get_string_descriptor_ascii(handle, desc.iManufacturer, string, sizeof(string));
233*86b64dcbSAndroid Build Coastguard Worker             if (ret > 0)
234*86b64dcbSAndroid Build Coastguard Worker                 LOGD("  Manufacturer:              %s\n", (char *)string);
235*86b64dcbSAndroid Build Coastguard Worker         }
236*86b64dcbSAndroid Build Coastguard Worker 
237*86b64dcbSAndroid Build Coastguard Worker         if (desc.iProduct) {
238*86b64dcbSAndroid Build Coastguard Worker             ret = libusb_get_string_descriptor_ascii(handle, desc.iProduct, string, sizeof(string));
239*86b64dcbSAndroid Build Coastguard Worker             if (ret > 0)
240*86b64dcbSAndroid Build Coastguard Worker                 LOGD("  Product:                   %s\n", (char *)string);
241*86b64dcbSAndroid Build Coastguard Worker         }
242*86b64dcbSAndroid Build Coastguard Worker 
243*86b64dcbSAndroid Build Coastguard Worker         if (desc.iSerialNumber && verbose) {
244*86b64dcbSAndroid Build Coastguard Worker             ret = libusb_get_string_descriptor_ascii(handle, desc.iSerialNumber, string, sizeof(string));
245*86b64dcbSAndroid Build Coastguard Worker             if (ret > 0)
246*86b64dcbSAndroid Build Coastguard Worker                 LOGD("  Serial Number:             %s\n", (char *)string);
247*86b64dcbSAndroid Build Coastguard Worker         }
248*86b64dcbSAndroid Build Coastguard Worker     }
249*86b64dcbSAndroid Build Coastguard Worker 
250*86b64dcbSAndroid Build Coastguard Worker     if (verbose) {
251*86b64dcbSAndroid Build Coastguard Worker         for (i = 0; i < desc.bNumConfigurations; i++) {
252*86b64dcbSAndroid Build Coastguard Worker             struct libusb_config_descriptor *config;
253*86b64dcbSAndroid Build Coastguard Worker 
254*86b64dcbSAndroid Build Coastguard Worker             ret = libusb_get_config_descriptor(dev, i, &config);
255*86b64dcbSAndroid Build Coastguard Worker             if (LIBUSB_SUCCESS != ret) {
256*86b64dcbSAndroid Build Coastguard Worker                 LOGD("  Couldn't retrieve descriptors\n");
257*86b64dcbSAndroid Build Coastguard Worker                 continue;
258*86b64dcbSAndroid Build Coastguard Worker             }
259*86b64dcbSAndroid Build Coastguard Worker 
260*86b64dcbSAndroid Build Coastguard Worker             print_configuration(config);
261*86b64dcbSAndroid Build Coastguard Worker 
262*86b64dcbSAndroid Build Coastguard Worker             libusb_free_config_descriptor(config);
263*86b64dcbSAndroid Build Coastguard Worker         }
264*86b64dcbSAndroid Build Coastguard Worker 
265*86b64dcbSAndroid Build Coastguard Worker         if (handle && desc.bcdUSB >= 0x0201)
266*86b64dcbSAndroid Build Coastguard Worker             print_bos(handle);
267*86b64dcbSAndroid Build Coastguard Worker     }
268*86b64dcbSAndroid Build Coastguard Worker 
269*86b64dcbSAndroid Build Coastguard Worker     if (handle)
270*86b64dcbSAndroid Build Coastguard Worker         libusb_close(handle);
271*86b64dcbSAndroid Build Coastguard Worker }
272*86b64dcbSAndroid Build Coastguard Worker 
273*86b64dcbSAndroid Build Coastguard Worker 
274*86b64dcbSAndroid Build Coastguard Worker /* fileDescriptor = the native file descriptor obtained in Java and transferred to native over JNA, for example */
unrooted_usb_description(int fileDescriptor)275*86b64dcbSAndroid Build Coastguard Worker int unrooted_usb_description(int fileDescriptor)
276*86b64dcbSAndroid Build Coastguard Worker {
277*86b64dcbSAndroid Build Coastguard Worker     libusb_context *ctx = NULL;
278*86b64dcbSAndroid Build Coastguard Worker     libusb_device_handle *devh = NULL;
279*86b64dcbSAndroid Build Coastguard Worker     int r = 0;
280*86b64dcbSAndroid Build Coastguard Worker     verbose = 1;
281*86b64dcbSAndroid Build Coastguard Worker     r = libusb_set_option(NULL, LIBUSB_OPTION_NO_DEVICE_DISCOVERY, NULL);
282*86b64dcbSAndroid Build Coastguard Worker     if (r != LIBUSB_SUCCESS) {
283*86b64dcbSAndroid Build Coastguard Worker         LOGD("libusb_set_option failed: %d\n", r);
284*86b64dcbSAndroid Build Coastguard Worker         return -1;
285*86b64dcbSAndroid Build Coastguard Worker     }
286*86b64dcbSAndroid Build Coastguard Worker     r = libusb_init(&ctx);
287*86b64dcbSAndroid Build Coastguard Worker     if (r < 0) {
288*86b64dcbSAndroid Build Coastguard Worker         LOGD("libusb_init failed: %d\n", r);
289*86b64dcbSAndroid Build Coastguard Worker         return r;
290*86b64dcbSAndroid Build Coastguard Worker     }
291*86b64dcbSAndroid Build Coastguard Worker     r = libusb_wrap_sys_device(ctx, (intptr_t)fileDescriptor, &devh);
292*86b64dcbSAndroid Build Coastguard Worker     if (r < 0) {
293*86b64dcbSAndroid Build Coastguard Worker         LOGD("libusb_wrap_sys_device failed: %d\n", r);
294*86b64dcbSAndroid Build Coastguard Worker         return r;
295*86b64dcbSAndroid Build Coastguard Worker     } else if (devh == NULL) {
296*86b64dcbSAndroid Build Coastguard Worker         LOGD("libusb_wrap_sys_device returned invalid handle\n");
297*86b64dcbSAndroid Build Coastguard Worker         return r;
298*86b64dcbSAndroid Build Coastguard Worker     }
299*86b64dcbSAndroid Build Coastguard Worker     print_device(libusb_get_device(devh), devh);
300*86b64dcbSAndroid Build Coastguard Worker     return r;
301*86b64dcbSAndroid Build Coastguard Worker }
302