VirtualBox

source: vbox/trunk/src/VBox/Main/src-server/linux/USBGetDevices.cpp@ 58650

Last change on this file since 58650 was 58170, checked in by vboxsync, 9 years ago

doxygen: fixes.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 59.6 KB
Line 
1/* $Id: USBGetDevices.cpp 58170 2015-10-12 09:27:14Z vboxsync $ */
2/** @file
3 * VirtualBox Linux host USB device enumeration.
4 */
5
6/*
7 * Copyright (C) 2006-2012 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 */
17
18
19/*********************************************************************************************************************************
20* Header Files *
21*********************************************************************************************************************************/
22
23#include "USBGetDevices.h"
24
25#include <VBox/err.h>
26#include <VBox/usb.h>
27#include <VBox/usblib.h>
28
29#include <iprt/linux/sysfs.h>
30#include <iprt/cdefs.h>
31#include <iprt/ctype.h>
32#include <iprt/dir.h>
33#include <iprt/env.h>
34#include <iprt/file.h>
35#include <iprt/fs.h>
36#include <iprt/log.h>
37#include <iprt/mem.h>
38#include <iprt/param.h>
39#include <iprt/path.h>
40#include <iprt/string.h>
41#include "vector.h"
42
43#ifdef VBOX_WITH_LINUX_COMPILER_H
44# include <linux/compiler.h>
45#endif
46#include <linux/usbdevice_fs.h>
47
48#include <sys/types.h>
49#include <sys/stat.h>
50#include <sys/vfs.h>
51
52#include <dirent.h>
53#include <dlfcn.h>
54#include <errno.h>
55#include <fcntl.h>
56#include <stdio.h>
57#include <string.h>
58#include <unistd.h>
59
60
61/*********************************************************************************************************************************
62* Structures and Typedefs *
63*********************************************************************************************************************************/
64/** Suffix translation. */
65typedef struct USBSUFF
66{
67 char szSuff[4];
68 unsigned cchSuff;
69 unsigned uMul;
70 unsigned uDiv;
71} USBSUFF, *PUSBSUFF;
72typedef const USBSUFF *PCUSBSUFF;
73
74/** Structure describing a host USB device */
75typedef struct USBDeviceInfo
76{
77 /** The device node of the device. */
78 char *mDevice;
79 /** The system identifier of the device. Specific to the probing
80 * method. */
81 char *mSysfsPath;
82 /** List of interfaces as sysfs paths */
83 VECTOR_PTR(char *) mvecpszInterfaces;
84} USBDeviceInfo;
85
86
87/*********************************************************************************************************************************
88* Global Variables *
89*********************************************************************************************************************************/
90/**
91 * Suffixes for the endpoint polling interval.
92 */
93static const USBSUFF s_aIntervalSuff[] =
94{
95 { "ms", 2, 1, 0 },
96 { "us", 2, 1, 1000 },
97 { "ns", 2, 1, 1000000 },
98 { "s", 1, 1000, 0 },
99 { "", 0, 0, 0 } /* term */
100};
101
102
103/**
104 * "reads" the number suffix. It's more like validating it and
105 * skipping the necessary number of chars.
106 */
107static int usbReadSkipSuffix(char **ppszNext)
108{
109 char *pszNext = *ppszNext;
110 if (!RT_C_IS_SPACE(*pszNext) && *pszNext)
111 {
112 /* skip unit */
113 if (pszNext[0] == 'm' && pszNext[1] == 's')
114 pszNext += 2;
115 else if (pszNext[0] == 'm' && pszNext[1] == 'A')
116 pszNext += 2;
117
118 /* skip parenthesis */
119 if (*pszNext == '(')
120 {
121 pszNext = strchr(pszNext, ')');
122 if (!pszNext++)
123 {
124 AssertMsgFailed(("*ppszNext=%s\n", *ppszNext));
125 return VERR_PARSE_ERROR;
126 }
127 }
128
129 /* blank or end of the line. */
130 if (!RT_C_IS_SPACE(*pszNext) && *pszNext)
131 {
132 AssertMsgFailed(("pszNext=%s\n", pszNext));
133 return VERR_PARSE_ERROR;
134 }
135
136 /* it's ok. */
137 *ppszNext = pszNext;
138 }
139
140 return VINF_SUCCESS;
141}
142
143
144/**
145 * Reads a USB number returning the number and the position of the next character to parse.
146 */
147static int usbReadNum(const char *pszValue, unsigned uBase, uint32_t u32Mask, PCUSBSUFF paSuffs, void *pvNum, char **ppszNext)
148{
149 /*
150 * Initialize return value to zero and strip leading spaces.
151 */
152 switch (u32Mask)
153 {
154 case 0xff: *(uint8_t *)pvNum = 0; break;
155 case 0xffff: *(uint16_t *)pvNum = 0; break;
156 case 0xffffffff: *(uint32_t *)pvNum = 0; break;
157 }
158 pszValue = RTStrStripL(pszValue);
159 if (*pszValue)
160 {
161 /*
162 * Try convert the number.
163 */
164 char *pszNext;
165 uint32_t u32 = 0;
166 RTStrToUInt32Ex(pszValue, &pszNext, uBase, &u32);
167 if (pszNext == pszValue)
168 {
169 AssertMsgFailed(("pszValue=%d\n", pszValue));
170 return VERR_NO_DATA;
171 }
172
173 /*
174 * Check the range.
175 */
176 if (u32 & ~u32Mask)
177 {
178 AssertMsgFailed(("pszValue=%d u32=%#x lMask=%#x\n", pszValue, u32, u32Mask));
179 return VERR_OUT_OF_RANGE;
180 }
181
182 /*
183 * Validate and skip stuff following the number.
184 */
185 if (paSuffs)
186 {
187 if (!RT_C_IS_SPACE(*pszNext) && *pszNext)
188 {
189 for (PCUSBSUFF pSuff = paSuffs; pSuff->szSuff[0]; pSuff++)
190 {
191 if ( !strncmp(pSuff->szSuff, pszNext, pSuff->cchSuff)
192 && (!pszNext[pSuff->cchSuff] || RT_C_IS_SPACE(pszNext[pSuff->cchSuff])))
193 {
194 if (pSuff->uDiv)
195 u32 /= pSuff->uDiv;
196 else
197 u32 *= pSuff->uMul;
198 break;
199 }
200 }
201 }
202 }
203 else
204 {
205 int rc = usbReadSkipSuffix(&pszNext);
206 if (RT_FAILURE(rc))
207 return rc;
208 }
209
210 *ppszNext = pszNext;
211
212 /*
213 * Set the value.
214 */
215 switch (u32Mask)
216 {
217 case 0xff: *(uint8_t *)pvNum = (uint8_t)u32; break;
218 case 0xffff: *(uint16_t *)pvNum = (uint16_t)u32; break;
219 case 0xffffffff: *(uint32_t *)pvNum = (uint32_t)u32; break;
220 }
221 }
222 return VINF_SUCCESS;
223}
224
225
226static int usbRead8(const char *pszValue, unsigned uBase, uint8_t *pu8, char **ppszNext)
227{
228 return usbReadNum(pszValue, uBase, 0xff, NULL, pu8, ppszNext);
229}
230
231
232static int usbRead16(const char *pszValue, unsigned uBase, uint16_t *pu16, char **ppszNext)
233{
234 return usbReadNum(pszValue, uBase, 0xffff, NULL, pu16, ppszNext);
235}
236
237
238#if 0
239static int usbRead16Suff(const char *pszValue, unsigned uBase, PCUSBSUFF paSuffs, uint16_t *pu16, char **ppszNext)
240{
241 return usbReadNum(pszValue, uBase, 0xffff, paSuffs, pu16, ppszNext);
242}
243#endif
244
245
246/**
247 * Reads a USB BCD number returning the number and the position of the next character to parse.
248 * The returned number contains the integer part in the high byte and the decimal part in the low byte.
249 */
250static int usbReadBCD(const char *pszValue, unsigned uBase, uint16_t *pu16, char **ppszNext)
251{
252 /*
253 * Initialize return value to zero and strip leading spaces.
254 */
255 *pu16 = 0;
256 pszValue = RTStrStripL(pszValue);
257 if (*pszValue)
258 {
259 /*
260 * Try convert the number.
261 */
262 /* integer part */
263 char *pszNext;
264 uint32_t u32Int = 0;
265 RTStrToUInt32Ex(pszValue, &pszNext, uBase, &u32Int);
266 if (pszNext == pszValue)
267 {
268 AssertMsgFailed(("pszValue=%s\n", pszValue));
269 return VERR_NO_DATA;
270 }
271 if (u32Int & ~0xff)
272 {
273 AssertMsgFailed(("pszValue=%s u32Int=%#x (int)\n", pszValue, u32Int));
274 return VERR_OUT_OF_RANGE;
275 }
276
277 /* skip dot and read decimal part */
278 if (*pszNext != '.')
279 {
280 AssertMsgFailed(("pszValue=%s pszNext=%s (int)\n", pszValue, pszNext));
281 return VERR_PARSE_ERROR;
282 }
283 char *pszValue2 = RTStrStripL(pszNext + 1);
284 uint32_t u32Dec = 0;
285 RTStrToUInt32Ex(pszValue2, &pszNext, uBase, &u32Dec);
286 if (pszNext == pszValue)
287 {
288 AssertMsgFailed(("pszValue=%s\n", pszValue));
289 return VERR_NO_DATA;
290 }
291 if (u32Dec & ~0xff)
292 {
293 AssertMsgFailed(("pszValue=%s u32Dec=%#x\n", pszValue, u32Dec));
294 return VERR_OUT_OF_RANGE;
295 }
296
297 /*
298 * Validate and skip stuff following the number.
299 */
300 int rc = usbReadSkipSuffix(&pszNext);
301 if (RT_FAILURE(rc))
302 return rc;
303 *ppszNext = pszNext;
304
305 /*
306 * Set the value.
307 */
308 *pu16 = (uint16_t)u32Int << 8 | (uint16_t)u32Dec;
309 }
310 return VINF_SUCCESS;
311}
312
313
314/**
315 * Reads a string, i.e. allocates memory and copies it.
316 *
317 * We assume that a string is Utf8 and if that's not the case
318 * (pre-2.6.32-kernels used Latin-1, but so few devices return non-ASCII that
319 * this usually goes unnoticed) then we mercilessly force it to be so.
320 */
321static int usbReadStr(const char *pszValue, const char **ppsz)
322{
323 char *psz;
324
325 if (*ppsz)
326 RTStrFree((char *)*ppsz);
327 psz = RTStrDup(pszValue);
328 if (psz)
329 {
330 RTStrPurgeEncoding(psz);
331 *ppsz = psz;
332 return VINF_SUCCESS;
333 }
334 return VERR_NO_MEMORY;
335}
336
337
338/**
339 * Skips the current property.
340 */
341static char *usbReadSkip(char *pszValue)
342{
343 char *psz = strchr(pszValue, '=');
344 if (psz)
345 psz = strchr(psz + 1, '=');
346 if (!psz)
347 return strchr(pszValue, '\0');
348 while (psz > pszValue && !RT_C_IS_SPACE(psz[-1]))
349 psz--;
350 Assert(psz > pszValue);
351 return psz;
352}
353
354
355/**
356 * Determine the USB speed.
357 */
358static int usbReadSpeed(const char *pszValue, USBDEVICESPEED *pSpd, char **ppszNext)
359{
360 pszValue = RTStrStripL(pszValue);
361 /* verified with Linux 2.4.0 ... Linux 2.6.25 */
362 if (!strncmp(pszValue, RT_STR_TUPLE("1.5")))
363 *pSpd = USBDEVICESPEED_LOW;
364 else if (!strncmp(pszValue, RT_STR_TUPLE("12 ")))
365 *pSpd = USBDEVICESPEED_FULL;
366 else if (!strncmp(pszValue, RT_STR_TUPLE("480")))
367 *pSpd = USBDEVICESPEED_HIGH;
368 else if (!strncmp(pszValue, RT_STR_TUPLE("5000")))
369 *pSpd = USBDEVICESPEED_SUPER;
370 else
371 *pSpd = USBDEVICESPEED_UNKNOWN;
372 while (pszValue[0] != '\0' && !RT_C_IS_SPACE(pszValue[0]))
373 pszValue++;
374 *ppszNext = (char *)pszValue;
375 return VINF_SUCCESS;
376}
377
378
379/**
380 * Compare a prefix and returns pointer to the char following it if it matches.
381 */
382static char *usbPrefix(char *psz, const char *pszPref, size_t cchPref)
383{
384 if (strncmp(psz, pszPref, cchPref))
385 return NULL;
386 return psz + cchPref;
387}
388
389
390/**
391 * Does some extra checks to improve the detected device state.
392 *
393 * We cannot distinguish between USED_BY_HOST_CAPTURABLE and
394 * USED_BY_GUEST, HELD_BY_PROXY all that well and it shouldn't be
395 * necessary either.
396 *
397 * We will however, distinguish between the device we have permissions
398 * to open and those we don't. This is necessary for two reasons.
399 *
400 * Firstly, because it's futile to even attempt opening a device which we
401 * don't have access to, it only serves to confuse the user. (That said,
402 * it might also be a bit confusing for the user to see that a USB device
403 * is grayed out with no further explanation, and no way of generating an
404 * error hinting at why this is the case.)
405 *
406 * Secondly and more importantly, we're racing against udevd with respect
407 * to permissions and group settings on newly plugged devices. When we
408 * detect a new device that we cannot access we will poll on it for a few
409 * seconds to give udevd time to fix it. The polling is actually triggered
410 * in the 'new device' case in the compare loop.
411 *
412 * The USBDEVICESTATE_USED_BY_HOST state is only used for this no-access
413 * case, while USBDEVICESTATE_UNSUPPORTED is only used in the 'hub' case.
414 * When it's neither of these, we set USBDEVICESTATE_UNUSED or
415 * USBDEVICESTATE_USED_BY_HOST_CAPTURABLE depending on whether there is
416 * a driver associated with any of the interfaces.
417 *
418 * All except the access check and a special idVendor == 0 precaution
419 * is handled at parse time.
420 *
421 * @returns The adjusted state.
422 * @param pDevice The device.
423 */
424static USBDEVICESTATE usbDeterminState(PCUSBDEVICE pDevice)
425{
426 /*
427 * If it's already flagged as unsupported, there is nothing to do.
428 */
429 USBDEVICESTATE enmState = pDevice->enmState;
430 if (enmState == USBDEVICESTATE_UNSUPPORTED)
431 return USBDEVICESTATE_UNSUPPORTED;
432
433 /*
434 * Root hubs and similar doesn't have any vendor id, just
435 * refuse these device.
436 */
437 if (!pDevice->idVendor)
438 return USBDEVICESTATE_UNSUPPORTED;
439
440 /*
441 * Check if we've got access to the device, if we haven't flag
442 * it as used-by-host.
443 */
444#ifndef VBOX_USB_WITH_SYSFS
445 const char *pszAddress = pDevice->pszAddress;
446#else
447 if (pDevice->pszAddress == NULL)
448 /* We can't do much with the device without an address. */
449 return USBDEVICESTATE_UNSUPPORTED;
450 const char *pszAddress = strstr(pDevice->pszAddress, "//device:");
451 pszAddress = pszAddress != NULL
452 ? pszAddress + sizeof("//device:") - 1
453 : pDevice->pszAddress;
454#endif
455 if ( access(pszAddress, R_OK | W_OK) != 0
456 && errno == EACCES)
457 return USBDEVICESTATE_USED_BY_HOST;
458
459#ifdef VBOX_USB_WITH_SYSFS
460 /**
461 * @todo Check that any other essential fields are present and mark as
462 * invalid if not. Particularly to catch the case where the device was
463 * unplugged while we were reading in its properties.
464 */
465#endif
466
467 return enmState;
468}
469
470
471/** Just a worker for USBProxyServiceLinux::getDevices that avoids some code duplication. */
472static int addDeviceToChain(PUSBDEVICE pDev, PUSBDEVICE *ppFirst, PUSBDEVICE **pppNext, const char *pcszUsbfsRoot,
473 bool testfs, int rc)
474{
475 /* usbDeterminState requires the address. */
476 PUSBDEVICE pDevNew = (PUSBDEVICE)RTMemDup(pDev, sizeof(*pDev));
477 if (pDevNew)
478 {
479 RTStrAPrintf((char **)&pDevNew->pszAddress, "%s/%03d/%03d", pcszUsbfsRoot, pDevNew->bBus, pDevNew->bDevNum);
480 if (pDevNew->pszAddress)
481 {
482 pDevNew->enmState = usbDeterminState(pDevNew);
483 if (pDevNew->enmState != USBDEVICESTATE_UNSUPPORTED || testfs)
484 {
485 if (*pppNext)
486 **pppNext = pDevNew;
487 else
488 *ppFirst = pDevNew;
489 *pppNext = &pDevNew->pNext;
490 }
491 else
492 deviceFree(pDevNew);
493 }
494 else
495 {
496 deviceFree(pDevNew);
497 rc = VERR_NO_MEMORY;
498 }
499 }
500 else
501 {
502 rc = VERR_NO_MEMORY;
503 deviceFreeMembers(pDev);
504 }
505
506 return rc;
507}
508
509
510static int openDevicesFile(const char *pcszUsbfsRoot, FILE **ppFile)
511{
512 char *pszPath;
513 FILE *pFile;
514 RTStrAPrintf(&pszPath, "%s/devices", pcszUsbfsRoot);
515 if (!pszPath)
516 return VERR_NO_MEMORY;
517 pFile = fopen(pszPath, "r");
518 RTStrFree(pszPath);
519 if (!pFile)
520 return RTErrConvertFromErrno(errno);
521 *ppFile = pFile;
522 return VINF_SUCCESS;
523}
524
525/**
526 * USBProxyService::getDevices() implementation for usbfs. The @a testfs flag
527 * tells the function to return information about unsupported devices as well.
528 * This is used as a sanity test to check that a devices file is really what
529 * we expect.
530 */
531static PUSBDEVICE getDevicesFromUsbfs(const char *pcszUsbfsRoot, bool testfs)
532{
533 PUSBDEVICE pFirst = NULL;
534 FILE *pFile = NULL;
535 int rc;
536 rc = openDevicesFile(pcszUsbfsRoot, &pFile);
537 if (RT_SUCCESS(rc))
538 {
539 PUSBDEVICE *ppNext = NULL;
540 int cHits = 0;
541 char szLine[1024];
542 USBDEVICE Dev;
543 RT_ZERO(Dev);
544 Dev.enmState = USBDEVICESTATE_UNUSED;
545
546 /* Set close on exit and hope no one is racing us. */
547 rc = fcntl(fileno(pFile), F_SETFD, FD_CLOEXEC) >= 0
548 ? VINF_SUCCESS
549 : RTErrConvertFromErrno(errno);
550 while ( RT_SUCCESS(rc)
551 && fgets(szLine, sizeof(szLine), pFile))
552 {
553 char *psz;
554 char *pszValue;
555
556 /* validate and remove the trailing newline. */
557 psz = strchr(szLine, '\0');
558 if (psz[-1] != '\n' && !feof(pFile))
559 {
560 AssertMsgFailed(("Line too long. (cch=%d)\n", strlen(szLine)));
561 continue;
562 }
563
564 /* strip */
565 psz = RTStrStrip(szLine);
566 if (!*psz)
567 continue;
568
569 /*
570 * Interpret the line.
571 * (Ordered by normal occurrence.)
572 */
573 char ch = psz[0];
574 if (psz[1] != ':')
575 continue;
576 psz = RTStrStripL(psz + 3);
577#define PREFIX(str) ( (pszValue = usbPrefix(psz, str, sizeof(str) - 1)) != NULL )
578 switch (ch)
579 {
580 /*
581 * T: Bus=dd Lev=dd Prnt=dd Port=dd Cnt=dd Dev#=ddd Spd=ddd MxCh=dd
582 * | | | | | | | | |__MaxChildren
583 * | | | | | | | |__Device Speed in Mbps
584 * | | | | | | |__DeviceNumber
585 * | | | | | |__Count of devices at this level
586 * | | | | |__Connector/Port on Parent for this device
587 * | | | |__Parent DeviceNumber
588 * | | |__Level in topology for this bus
589 * | |__Bus number
590 * |__Topology info tag
591 */
592 case 'T':
593 /* add */
594 AssertMsg(cHits >= 3 || cHits == 0, ("cHits=%d\n", cHits));
595 if (cHits >= 3)
596 rc = addDeviceToChain(&Dev, &pFirst, &ppNext, pcszUsbfsRoot, testfs, rc);
597 else
598 deviceFreeMembers(&Dev);
599
600 /* Reset device state */
601 RT_ZERO(Dev);
602 Dev.enmState = USBDEVICESTATE_UNUSED;
603 cHits = 1;
604
605 /* parse the line. */
606 while (*psz && RT_SUCCESS(rc))
607 {
608 if (PREFIX("Bus="))
609 rc = usbRead8(pszValue, 10, &Dev.bBus, &psz);
610 else if (PREFIX("Port="))
611 rc = usbRead8(pszValue, 10, &Dev.bPort, &psz);
612 else if (PREFIX("Spd="))
613 rc = usbReadSpeed(pszValue, &Dev.enmSpeed, &psz);
614 else if (PREFIX("Dev#="))
615 rc = usbRead8(pszValue, 10, &Dev.bDevNum, &psz);
616 else
617 psz = usbReadSkip(psz);
618 psz = RTStrStripL(psz);
619 }
620 break;
621
622 /*
623 * Bandwidth info:
624 * B: Alloc=ddd/ddd us (xx%), #Int=ddd, #Iso=ddd
625 * | | | |__Number of isochronous requests
626 * | | |__Number of interrupt requests
627 * | |__Total Bandwidth allocated to this bus
628 * |__Bandwidth info tag
629 */
630 case 'B':
631 break;
632
633 /*
634 * D: Ver=x.xx Cls=xx(sssss) Sub=xx Prot=xx MxPS=dd #Cfgs=dd
635 * | | | | | | |__NumberConfigurations
636 * | | | | | |__MaxPacketSize of Default Endpoint
637 * | | | | |__DeviceProtocol
638 * | | | |__DeviceSubClass
639 * | | |__DeviceClass
640 * | |__Device USB version
641 * |__Device info tag #1
642 */
643 case 'D':
644 while (*psz && RT_SUCCESS(rc))
645 {
646 if (PREFIX("Ver="))
647 rc = usbReadBCD(pszValue, 16, &Dev.bcdUSB, &psz);
648 else if (PREFIX("Cls="))
649 {
650 rc = usbRead8(pszValue, 16, &Dev.bDeviceClass, &psz);
651 if (RT_SUCCESS(rc) && Dev.bDeviceClass == 9 /* HUB */)
652 Dev.enmState = USBDEVICESTATE_UNSUPPORTED;
653 }
654 else if (PREFIX("Sub="))
655 rc = usbRead8(pszValue, 16, &Dev.bDeviceSubClass, &psz);
656 else if (PREFIX("Prot="))
657 rc = usbRead8(pszValue, 16, &Dev.bDeviceProtocol, &psz);
658 //else if (PREFIX("MxPS="))
659 // rc = usbRead16(pszValue, 10, &Dev.wMaxPacketSize, &psz);
660 else if (PREFIX("#Cfgs="))
661 rc = usbRead8(pszValue, 10, &Dev.bNumConfigurations, &psz);
662 else
663 psz = usbReadSkip(psz);
664 psz = RTStrStripL(psz);
665 }
666 cHits++;
667 break;
668
669 /*
670 * P: Vendor=xxxx ProdID=xxxx Rev=xx.xx
671 * | | | |__Product revision number
672 * | | |__Product ID code
673 * | |__Vendor ID code
674 * |__Device info tag #2
675 */
676 case 'P':
677 while (*psz && RT_SUCCESS(rc))
678 {
679 if (PREFIX("Vendor="))
680 rc = usbRead16(pszValue, 16, &Dev.idVendor, &psz);
681 else if (PREFIX("ProdID="))
682 rc = usbRead16(pszValue, 16, &Dev.idProduct, &psz);
683 else if (PREFIX("Rev="))
684 rc = usbReadBCD(pszValue, 16, &Dev.bcdDevice, &psz);
685 else
686 psz = usbReadSkip(psz);
687 psz = RTStrStripL(psz);
688 }
689 cHits++;
690 break;
691
692 /*
693 * String.
694 */
695 case 'S':
696 if (PREFIX("Manufacturer="))
697 rc = usbReadStr(pszValue, &Dev.pszManufacturer);
698 else if (PREFIX("Product="))
699 rc = usbReadStr(pszValue, &Dev.pszProduct);
700 else if (PREFIX("SerialNumber="))
701 {
702 rc = usbReadStr(pszValue, &Dev.pszSerialNumber);
703 if (RT_SUCCESS(rc))
704 Dev.u64SerialHash = USBLibHashSerial(pszValue);
705 }
706 break;
707
708 /*
709 * C:* #Ifs=dd Cfg#=dd Atr=xx MPwr=dddmA
710 * | | | | | |__MaxPower in mA
711 * | | | | |__Attributes
712 * | | | |__ConfiguratioNumber
713 * | | |__NumberOfInterfaces
714 * | |__ "*" indicates the active configuration (others are " ")
715 * |__Config info tag
716 */
717 case 'C':
718 break;
719
720 /*
721 * I: If#=dd Alt=dd #EPs=dd Cls=xx(sssss) Sub=xx Prot=xx Driver=ssss
722 * | | | | | | | |__Driver name
723 * | | | | | | | or "(none)"
724 * | | | | | | |__InterfaceProtocol
725 * | | | | | |__InterfaceSubClass
726 * | | | | |__InterfaceClass
727 * | | | |__NumberOfEndpoints
728 * | | |__AlternateSettingNumber
729 * | |__InterfaceNumber
730 * |__Interface info tag
731 */
732 case 'I':
733 {
734 /* Check for thing we don't support. */
735 while (*psz && RT_SUCCESS(rc))
736 {
737 if (PREFIX("Driver="))
738 {
739 const char *pszDriver = NULL;
740 rc = usbReadStr(pszValue, &pszDriver);
741 if ( !pszDriver
742 || !*pszDriver
743 || !strcmp(pszDriver, "(none)")
744 || !strcmp(pszDriver, "(no driver)"))
745 /* no driver */;
746 else if (!strcmp(pszDriver, "hub"))
747 Dev.enmState = USBDEVICESTATE_UNSUPPORTED;
748 else if (Dev.enmState == USBDEVICESTATE_UNUSED)
749 Dev.enmState = USBDEVICESTATE_USED_BY_HOST_CAPTURABLE;
750 RTStrFree((char *)pszDriver);
751 break; /* last attrib */
752 }
753 else if (PREFIX("Cls="))
754 {
755 uint8_t bInterfaceClass;
756 rc = usbRead8(pszValue, 16, &bInterfaceClass, &psz);
757 if (RT_SUCCESS(rc) && bInterfaceClass == 9 /* HUB */)
758 Dev.enmState = USBDEVICESTATE_UNSUPPORTED;
759 }
760 else
761 psz = usbReadSkip(psz);
762 psz = RTStrStripL(psz);
763 }
764 break;
765 }
766
767
768 /*
769 * E: Ad=xx(s) Atr=xx(ssss) MxPS=dddd Ivl=dddms
770 * | | | | |__Interval (max) between transfers
771 * | | | |__EndpointMaxPacketSize
772 * | | |__Attributes(EndpointType)
773 * | |__EndpointAddress(I=In,O=Out)
774 * |__Endpoint info tag
775 */
776 case 'E':
777 break;
778
779 }
780#undef PREFIX
781 } /* parse loop */
782 fclose(pFile);
783
784 /*
785 * Add the current entry.
786 */
787 AssertMsg(cHits >= 3 || cHits == 0, ("cHits=%d\n", cHits));
788 if (cHits >= 3)
789 rc = addDeviceToChain(&Dev, &pFirst, &ppNext, pcszUsbfsRoot, testfs, rc);
790
791 /*
792 * Success?
793 */
794 if (RT_FAILURE(rc))
795 {
796 while (pFirst)
797 {
798 PUSBDEVICE pFree = pFirst;
799 pFirst = pFirst->pNext;
800 deviceFree(pFree);
801 }
802 }
803 }
804 if (RT_FAILURE(rc))
805 LogFlow(("USBProxyServiceLinux::getDevices: rc=%Rrc\n", rc));
806 return pFirst;
807}
808
809#ifdef VBOX_USB_WITH_SYSFS
810
811static void USBDevInfoCleanup(USBDeviceInfo *pSelf)
812{
813 RTStrFree(pSelf->mDevice);
814 RTStrFree(pSelf->mSysfsPath);
815 pSelf->mDevice = pSelf->mSysfsPath = NULL;
816 VEC_CLEANUP_PTR(&pSelf->mvecpszInterfaces);
817}
818
819static int USBDevInfoInit(USBDeviceInfo *pSelf, const char *aDevice,
820 const char *aSystemID)
821{
822 pSelf->mDevice = aDevice ? RTStrDup(aDevice) : NULL;
823 pSelf->mSysfsPath = aSystemID ? RTStrDup(aSystemID) : NULL;
824 VEC_INIT_PTR(&pSelf->mvecpszInterfaces, char *, RTStrFree);
825 if ((aDevice && !pSelf->mDevice) || (aSystemID && ! pSelf->mSysfsPath))
826 {
827 USBDevInfoCleanup(pSelf);
828 return 0;
829 }
830 return 1;
831}
832
833#define USBDEVICE_MAJOR 189
834
835/** Calculate the bus (a.k.a root hub) number of a USB device from it's sysfs
836 * path. sysfs nodes representing root hubs have file names of the form
837 * usb<n>, where n is the bus number; other devices start with that number.
838 * See [http://www.linux-usb.org/FAQ.html#i6] and
839 * [http://www.kernel.org/doc/Documentation/usb/proc_usb_info.txt] for
840 * equivalent information about usbfs.
841 * @returns a bus number greater than 0 on success or 0 on failure.
842 */
843static unsigned usbGetBusFromSysfsPath(const char *pcszPath)
844{
845 const char *pcszFile = strrchr(pcszPath, '/');
846 if (!pcszFile)
847 return 0;
848 unsigned bus = RTStrToUInt32(pcszFile + 1);
849 if ( !bus
850 && pcszFile[1] == 'u' && pcszFile[2] == 's' && pcszFile[3] == 'b')
851 bus = RTStrToUInt32(pcszFile + 4);
852 return bus;
853}
854
855/** Calculate the device number of a USB device. See
856 * drivers/usb/core/hub.c:usb_new_device as of Linux 2.6.20. */
857static dev_t usbMakeDevNum(unsigned bus, unsigned device)
858{
859 AssertReturn(bus > 0, 0);
860 AssertReturn(((device - 1) & ~127) == 0, 0);
861 AssertReturn(device > 0, 0);
862 return makedev(USBDEVICE_MAJOR, ((bus - 1) << 7) + device - 1);
863}
864
865/**
866 * If a file @a pcszNode from /sys/bus/usb/devices is a device rather than an
867 * interface add an element for the device to @a pvecDevInfo.
868 */
869static int addIfDevice(const char *pcszDevicesRoot,
870 const char *pcszNode,
871 VECTOR_OBJ(USBDeviceInfo) *pvecDevInfo)
872{
873 const char *pcszFile = strrchr(pcszNode, '/');
874 if (!pcszFile)
875 return VERR_INVALID_PARAMETER;
876 if (strchr(pcszFile, ':'))
877 return VINF_SUCCESS;
878 unsigned bus = usbGetBusFromSysfsPath(pcszNode);
879 if (!bus)
880 return VINF_SUCCESS;
881 int device = RTLinuxSysFsReadIntFile(10, "%s/devnum", pcszNode);
882 if (device < 0)
883 return VINF_SUCCESS;
884 dev_t devnum = usbMakeDevNum(bus, device);
885 if (!devnum)
886 return VINF_SUCCESS;
887 char szDevPath[RTPATH_MAX];
888 ssize_t cchDevPath;
889 cchDevPath = RTLinuxCheckDevicePath(devnum, RTFS_TYPE_DEV_CHAR,
890 szDevPath, sizeof(szDevPath),
891 "%s/%.3d/%.3d",
892 pcszDevicesRoot, bus, device);
893 if (cchDevPath < 0)
894 return VINF_SUCCESS;
895
896 USBDeviceInfo info;
897 if (USBDevInfoInit(&info, szDevPath, pcszNode))
898 if (RT_SUCCESS(VEC_PUSH_BACK_OBJ(pvecDevInfo, USBDeviceInfo,
899 &info)))
900 return VINF_SUCCESS;
901 USBDevInfoCleanup(&info);
902 return VERR_NO_MEMORY;
903}
904
905/** The logic for testing whether a sysfs address corresponds to an
906 * interface of a device. Both must be referenced by their canonical
907 * sysfs paths. This is not tested, as the test requires file-system
908 * interaction. */
909static bool muiIsAnInterfaceOf(const char *pcszIface, const char *pcszDev)
910{
911 size_t cchDev = strlen(pcszDev);
912
913 AssertPtr(pcszIface);
914 AssertPtr(pcszDev);
915 Assert(pcszIface[0] == '/');
916 Assert(pcszDev[0] == '/');
917 Assert(pcszDev[cchDev - 1] != '/');
918 /* If this passes, pcszIface is at least cchDev long */
919 if (strncmp(pcszIface, pcszDev, cchDev))
920 return false;
921 /* If this passes, pcszIface is longer than cchDev */
922 if (pcszIface[cchDev] != '/')
923 return false;
924 /* In sysfs an interface is an immediate subdirectory of the device */
925 if (strchr(pcszIface + cchDev + 1, '/'))
926 return false;
927 /* And it always has a colon in its name */
928 if (!strchr(pcszIface + cchDev + 1, ':'))
929 return false;
930 /* And hopefully we have now elimitated everything else */
931 return true;
932}
933
934#ifdef DEBUG
935# ifdef __cplusplus
936/** Unit test the logic in muiIsAnInterfaceOf in debug builds. */
937class testIsAnInterfaceOf
938{
939public:
940 testIsAnInterfaceOf()
941 {
942 Assert(muiIsAnInterfaceOf("/sys/devices/pci0000:00/0000:00:1a.0/usb3/3-0:1.0",
943 "/sys/devices/pci0000:00/0000:00:1a.0/usb3"));
944 Assert(!muiIsAnInterfaceOf("/sys/devices/pci0000:00/0000:00:1a.0/usb3/3-1",
945 "/sys/devices/pci0000:00/0000:00:1a.0/usb3"));
946 Assert(!muiIsAnInterfaceOf("/sys/devices/pci0000:00/0000:00:1a.0/usb3/3-0:1.0/driver",
947 "/sys/devices/pci0000:00/0000:00:1a.0/usb3"));
948 }
949};
950static testIsAnInterfaceOf testIsAnInterfaceOfInst;
951# endif /* __cplusplus */
952#endif /* DEBUG */
953
954/**
955 * Tell whether a file in /sys/bus/usb/devices is an interface rather than a
956 * device. To be used with getDeviceInfoFromSysfs().
957 */
958static int addIfInterfaceOf(const char *pcszNode, USBDeviceInfo *pInfo)
959{
960 if (!muiIsAnInterfaceOf(pcszNode, pInfo->mSysfsPath))
961 return VINF_SUCCESS;
962 char *pszDup = (char *)RTStrDup(pcszNode);
963 if (pszDup)
964 if (RT_SUCCESS(VEC_PUSH_BACK_PTR(&pInfo->mvecpszInterfaces,
965 char *, pszDup)))
966 return VINF_SUCCESS;
967 RTStrFree(pszDup);
968 return VERR_NO_MEMORY;
969}
970
971/** Helper for readFilePaths(). Adds the entries from the open directory
972 * @a pDir to the vector @a pvecpchDevs using either the full path or the
973 * realpath() and skipping hidden files and files on which realpath() fails. */
974static int readFilePathsFromDir(const char *pcszPath, DIR *pDir,
975 VECTOR_PTR(char *) *pvecpchDevs)
976{
977 struct dirent entry, *pResult;
978 int err, rc;
979
980 for (err = readdir_r(pDir, &entry, &pResult); pResult;
981 err = readdir_r(pDir, &entry, &pResult))
982 {
983 char szPath[RTPATH_MAX + 1], szRealPath[RTPATH_MAX + 1], *pszPath;
984 if (entry.d_name[0] == '.')
985 continue;
986 if (snprintf(szPath, sizeof(szPath), "%s/%s", pcszPath,
987 entry.d_name) < 0)
988 return RTErrConvertFromErrno(errno);
989 if (!realpath(szPath, szRealPath))
990 return RTErrConvertFromErrno(errno);
991 pszPath = RTStrDup(szRealPath);
992 if (!pszPath)
993 return VERR_NO_MEMORY;
994 if (RT_FAILURE(rc = VEC_PUSH_BACK_PTR(pvecpchDevs, char *, pszPath)))
995 return rc;
996 }
997 return RTErrConvertFromErrno(err);
998}
999
1000/**
1001 * Dump the names of a directory's entries into a vector of char pointers.
1002 *
1003 * @returns zero on success or (positive) posix error value.
1004 * @param pcszPath the path to dump.
1005 * @param pvecpchDevs an empty vector of char pointers - must be cleaned up
1006 * by the caller even on failure.
1007 * @param withRealPath whether to canonicalise the filename with realpath
1008 */
1009static int readFilePaths(const char *pcszPath, VECTOR_PTR(char *) *pvecpchDevs)
1010{
1011 DIR *pDir;
1012 int rc;
1013
1014 AssertPtrReturn(pvecpchDevs, EINVAL);
1015 AssertReturn(VEC_SIZE_PTR(pvecpchDevs) == 0, EINVAL);
1016 AssertPtrReturn(pcszPath, EINVAL);
1017
1018 pDir = opendir(pcszPath);
1019 if (!pDir)
1020 return RTErrConvertFromErrno(errno);
1021 rc = readFilePathsFromDir(pcszPath, pDir, pvecpchDevs);
1022 if (closedir(pDir) < 0 && RT_SUCCESS(rc))
1023 rc = RTErrConvertFromErrno(errno);
1024 return rc;
1025}
1026
1027/**
1028 * Logic for USBSysfsEnumerateHostDevices.
1029 * @param pvecDevInfo vector of device information structures to add device
1030 * information to
1031 * @param pvecpchDevs empty scratch vector which will be freed by the caller,
1032 * to simplify exit logic
1033 */
1034static int doSysfsEnumerateHostDevices(const char *pcszDevicesRoot,
1035 VECTOR_OBJ(USBDeviceInfo) *pvecDevInfo,
1036 VECTOR_PTR(char *) *pvecpchDevs)
1037{
1038 char **ppszEntry;
1039 USBDeviceInfo *pInfo;
1040 int rc;
1041
1042 AssertPtrReturn(pvecDevInfo, VERR_INVALID_POINTER);
1043 LogFlowFunc (("pvecDevInfo=%p\n", pvecDevInfo));
1044
1045 rc = readFilePaths("/sys/bus/usb/devices", pvecpchDevs);
1046 if (RT_FAILURE(rc))
1047 return rc;
1048 VEC_FOR_EACH(pvecpchDevs, char *, ppszEntry)
1049 if (RT_FAILURE(rc = addIfDevice(pcszDevicesRoot, *ppszEntry,
1050 pvecDevInfo)))
1051 return rc;
1052 VEC_FOR_EACH(pvecDevInfo, USBDeviceInfo, pInfo)
1053 VEC_FOR_EACH(pvecpchDevs, char *, ppszEntry)
1054 if (RT_FAILURE(rc = addIfInterfaceOf(*ppszEntry, pInfo)))
1055 return rc;
1056 return VINF_SUCCESS;
1057}
1058
1059static int USBSysfsEnumerateHostDevices(const char *pcszDevicesRoot,
1060 VECTOR_OBJ(USBDeviceInfo) *pvecDevInfo)
1061{
1062 VECTOR_PTR(char *) vecpchDevs;
1063 int rc = VERR_NOT_IMPLEMENTED;
1064
1065 AssertReturn(VEC_SIZE_OBJ(pvecDevInfo) == 0, VERR_INVALID_PARAMETER);
1066 LogFlowFunc(("entered\n"));
1067 VEC_INIT_PTR(&vecpchDevs, char *, RTStrFree);
1068 rc = doSysfsEnumerateHostDevices(pcszDevicesRoot, pvecDevInfo,
1069 &vecpchDevs);
1070 VEC_CLEANUP_PTR(&vecpchDevs);
1071 LogFlowFunc(("rc=%Rrc\n", rc));
1072 return rc;
1073}
1074
1075/**
1076 * Helper function for extracting the port number on the parent device from
1077 * the sysfs path value.
1078 *
1079 * The sysfs path is a chain of elements separated by forward slashes, and for
1080 * USB devices, the last element in the chain takes the form
1081 * <port>-<port>.[...].<port>[:<config>.<interface>]
1082 * where the first <port> is the port number on the root hub, and the following
1083 * (optional) ones are the port numbers on any other hubs between the device
1084 * and the root hub. The last part (:<config.interface>) is only present for
1085 * interfaces, not for devices. This API should only be called for devices.
1086 * For compatibility with usbfs, which enumerates from zero up, we subtract one
1087 * from the port number.
1088 *
1089 * For root hubs, the last element in the chain takes the form
1090 * usb<hub number>
1091 * and usbfs always returns port number zero.
1092 *
1093 * @returns VBox status code. pu8Port is set on success.
1094 * @param pszPath The sysfs path to parse.
1095 * @param pu8Port Where to store the port number.
1096 */
1097static int usbGetPortFromSysfsPath(const char *pszPath, uint8_t *pu8Port)
1098{
1099 AssertPtrReturn(pszPath, VERR_INVALID_POINTER);
1100 AssertPtrReturn(pu8Port, VERR_INVALID_POINTER);
1101
1102 /*
1103 * This should not be possible until we get PCs with USB as their primary bus.
1104 * Note: We don't assert this, as we don't expect the caller to validate the
1105 * sysfs path.
1106 */
1107 const char *pszLastComp = strrchr(pszPath, '/');
1108 if (!pszLastComp)
1109 {
1110 Log(("usbGetPortFromSysfsPath(%s): failed [1]\n", pszPath));
1111 return VERR_INVALID_PARAMETER;
1112 }
1113 pszLastComp++; /* skip the slash */
1114
1115 /*
1116 * This API should not be called for interfaces, so the last component
1117 * of the path should not contain a colon. We *do* assert this, as it
1118 * might indicate a caller bug.
1119 */
1120 AssertMsgReturn(strchr(pszLastComp, ':') == NULL, ("%s\n", pszPath), VERR_INVALID_PARAMETER);
1121
1122 /*
1123 * Look for the start of the last number.
1124 */
1125 const char *pchDash = strrchr(pszLastComp, '-');
1126 const char *pchDot = strrchr(pszLastComp, '.');
1127 if (!pchDash && !pchDot)
1128 {
1129 /* No -/. so it must be a root hub. Check that it's usb<something>. */
1130 if (strncmp(pszLastComp, RT_STR_TUPLE("usb")) != 0)
1131 {
1132 Log(("usbGetPortFromSysfsPath(%s): failed [2]\n", pszPath));
1133 return VERR_INVALID_PARAMETER;
1134 }
1135 return VERR_NOT_SUPPORTED;
1136 }
1137 else
1138 {
1139 const char *pszLastPort = pchDot != NULL
1140 ? pchDot + 1
1141 : pchDash + 1;
1142 int rc = RTStrToUInt8Full(pszLastPort, 10, pu8Port);
1143 if (rc != VINF_SUCCESS)
1144 {
1145 Log(("usbGetPortFromSysfsPath(%s): failed [3], rc=%Rrc\n", pszPath, rc));
1146 return VERR_INVALID_PARAMETER;
1147 }
1148 if (*pu8Port == 0)
1149 {
1150 Log(("usbGetPortFromSysfsPath(%s): failed [4]\n", pszPath));
1151 return VERR_INVALID_PARAMETER;
1152 }
1153
1154 /* usbfs compatibility, 0-based port number. */
1155 *pu8Port -= 1;
1156 }
1157 return VINF_SUCCESS;
1158}
1159
1160
1161/**
1162 * Dumps a USBDEVICE structure to the log using LogLevel 3.
1163 * @param pDev The structure to log.
1164 * @todo This is really common code.
1165 */
1166DECLINLINE(void) usbLogDevice(PUSBDEVICE pDev)
1167{
1168 NOREF(pDev);
1169
1170 Log3(("USB device:\n"));
1171 Log3(("Product: %s (%x)\n", pDev->pszProduct, pDev->idProduct));
1172 Log3(("Manufacturer: %s (Vendor ID %x)\n", pDev->pszManufacturer, pDev->idVendor));
1173 Log3(("Serial number: %s (%llx)\n", pDev->pszSerialNumber, pDev->u64SerialHash));
1174 Log3(("Device revision: %d\n", pDev->bcdDevice));
1175 Log3(("Device class: %x\n", pDev->bDeviceClass));
1176 Log3(("Device subclass: %x\n", pDev->bDeviceSubClass));
1177 Log3(("Device protocol: %x\n", pDev->bDeviceProtocol));
1178 Log3(("USB version number: %d\n", pDev->bcdUSB));
1179 Log3(("Device speed: %s\n",
1180 pDev->enmSpeed == USBDEVICESPEED_UNKNOWN ? "unknown"
1181 : pDev->enmSpeed == USBDEVICESPEED_LOW ? "1.5 MBit/s"
1182 : pDev->enmSpeed == USBDEVICESPEED_FULL ? "12 MBit/s"
1183 : pDev->enmSpeed == USBDEVICESPEED_HIGH ? "480 MBit/s"
1184 : pDev->enmSpeed == USBDEVICESPEED_SUPER ? "5.0 GBit/s"
1185 : pDev->enmSpeed == USBDEVICESPEED_VARIABLE ? "variable"
1186 : "invalid"));
1187 Log3(("Number of configurations: %d\n", pDev->bNumConfigurations));
1188 Log3(("Bus number: %d\n", pDev->bBus));
1189 Log3(("Port number: %d\n", pDev->bPort));
1190 Log3(("Device number: %d\n", pDev->bDevNum));
1191 Log3(("Device state: %s\n",
1192 pDev->enmState == USBDEVICESTATE_UNSUPPORTED ? "unsupported"
1193 : pDev->enmState == USBDEVICESTATE_USED_BY_HOST ? "in use by host"
1194 : pDev->enmState == USBDEVICESTATE_USED_BY_HOST_CAPTURABLE ? "in use by host, possibly capturable"
1195 : pDev->enmState == USBDEVICESTATE_UNUSED ? "not in use"
1196 : pDev->enmState == USBDEVICESTATE_HELD_BY_PROXY ? "held by proxy"
1197 : pDev->enmState == USBDEVICESTATE_USED_BY_GUEST ? "used by guest"
1198 : "invalid"));
1199 Log3(("OS device address: %s\n", pDev->pszAddress));
1200}
1201
1202/**
1203 * In contrast to usbReadBCD() this function can handle BCD values without
1204 * a decimal separator. This is necessary for parsing bcdDevice.
1205 * @param pszBuf Pointer to the string buffer.
1206 * @param pu15 Pointer to the return value.
1207 * @returns IPRT status code.
1208 */
1209static int convertSysfsStrToBCD(const char *pszBuf, uint16_t *pu16)
1210{
1211 char *pszNext;
1212 int32_t i32;
1213
1214 pszBuf = RTStrStripL(pszBuf);
1215 int rc = RTStrToInt32Ex(pszBuf, &pszNext, 16, &i32);
1216 if ( RT_FAILURE(rc)
1217 || rc == VWRN_NUMBER_TOO_BIG
1218 || i32 < 0)
1219 return VERR_NUMBER_TOO_BIG;
1220 if (*pszNext == '.')
1221 {
1222 if (i32 > 255)
1223 return VERR_NUMBER_TOO_BIG;
1224 int32_t i32Lo;
1225 rc = RTStrToInt32Ex(pszNext+1, &pszNext, 16, &i32Lo);
1226 if ( RT_FAILURE(rc)
1227 || rc == VWRN_NUMBER_TOO_BIG
1228 || i32Lo > 255
1229 || i32Lo < 0)
1230 return VERR_NUMBER_TOO_BIG;
1231 i32 = (i32 << 8) | i32Lo;
1232 }
1233 if ( i32 > 65535
1234 || (*pszNext != '\0' && *pszNext != ' '))
1235 return VERR_NUMBER_TOO_BIG;
1236
1237 *pu16 = (uint16_t)i32;
1238 return VINF_SUCCESS;
1239}
1240
1241#endif /* VBOX_USB_WITH_SYSFS */
1242
1243static void fillInDeviceFromSysfs(USBDEVICE *Dev, USBDeviceInfo *pInfo)
1244{
1245 int rc;
1246 const char *pszSysfsPath = pInfo->mSysfsPath;
1247
1248 /* Fill in the simple fields */
1249 Dev->enmState = USBDEVICESTATE_UNUSED;
1250 Dev->bBus = usbGetBusFromSysfsPath(pszSysfsPath);
1251 Dev->bDeviceClass = RTLinuxSysFsReadIntFile(16, "%s/bDeviceClass", pszSysfsPath);
1252 Dev->bDeviceSubClass = RTLinuxSysFsReadIntFile(16, "%s/bDeviceSubClass", pszSysfsPath);
1253 Dev->bDeviceProtocol = RTLinuxSysFsReadIntFile(16, "%s/bDeviceProtocol", pszSysfsPath);
1254 Dev->bNumConfigurations = RTLinuxSysFsReadIntFile(10, "%s/bNumConfigurations", pszSysfsPath);
1255 Dev->idVendor = RTLinuxSysFsReadIntFile(16, "%s/idVendor", pszSysfsPath);
1256 Dev->idProduct = RTLinuxSysFsReadIntFile(16, "%s/idProduct", pszSysfsPath);
1257 Dev->bDevNum = RTLinuxSysFsReadIntFile(10, "%s/devnum", pszSysfsPath);
1258
1259 /* Now deal with the non-numeric bits. */
1260 char szBuf[1024]; /* Should be larger than anything a sane device
1261 * will need, and insane devices can be unsupported
1262 * until further notice. */
1263 ssize_t cchRead;
1264
1265 /* For simplicity, we just do strcmps on the next one. */
1266 cchRead = RTLinuxSysFsReadStrFile(szBuf, sizeof(szBuf), "%s/speed",
1267 pszSysfsPath);
1268 if (cchRead <= 0 || (size_t) cchRead == sizeof(szBuf))
1269 Dev->enmState = USBDEVICESTATE_UNSUPPORTED;
1270 else
1271 Dev->enmSpeed = !strcmp(szBuf, "1.5") ? USBDEVICESPEED_LOW
1272 : !strcmp(szBuf, "12") ? USBDEVICESPEED_FULL
1273 : !strcmp(szBuf, "480") ? USBDEVICESPEED_HIGH
1274 : !strcmp(szBuf, "5000") ? USBDEVICESPEED_SUPER
1275 : USBDEVICESPEED_UNKNOWN;
1276
1277 cchRead = RTLinuxSysFsReadStrFile(szBuf, sizeof(szBuf), "%s/version",
1278 pszSysfsPath);
1279 if (cchRead <= 0 || (size_t) cchRead == sizeof(szBuf))
1280 Dev->enmState = USBDEVICESTATE_UNSUPPORTED;
1281 else
1282 {
1283 rc = convertSysfsStrToBCD(szBuf, &Dev->bcdUSB);
1284 if (RT_FAILURE(rc))
1285 {
1286 Dev->enmState = USBDEVICESTATE_UNSUPPORTED;
1287 Dev->bcdUSB = (uint16_t)-1;
1288 }
1289 }
1290
1291 cchRead = RTLinuxSysFsReadStrFile(szBuf, sizeof(szBuf), "%s/bcdDevice",
1292 pszSysfsPath);
1293 if (cchRead <= 0 || (size_t) cchRead == sizeof(szBuf))
1294 Dev->bcdDevice = (uint16_t)-1;
1295 else
1296 {
1297 rc = convertSysfsStrToBCD(szBuf, &Dev->bcdDevice);
1298 if (RT_FAILURE(rc))
1299 Dev->bcdDevice = (uint16_t)-1;
1300 }
1301
1302 /* Now do things that need string duplication */
1303 cchRead = RTLinuxSysFsReadStrFile(szBuf, sizeof(szBuf), "%s/product",
1304 pszSysfsPath);
1305 if (cchRead > 0 && (size_t) cchRead < sizeof(szBuf))
1306 {
1307 RTStrPurgeEncoding(szBuf);
1308 Dev->pszProduct = RTStrDup(szBuf);
1309 }
1310
1311 cchRead = RTLinuxSysFsReadStrFile(szBuf, sizeof(szBuf), "%s/serial",
1312 pszSysfsPath);
1313 if (cchRead > 0 && (size_t) cchRead < sizeof(szBuf))
1314 {
1315 RTStrPurgeEncoding(szBuf);
1316 Dev->pszSerialNumber = RTStrDup(szBuf);
1317 Dev->u64SerialHash = USBLibHashSerial(szBuf);
1318 }
1319
1320 cchRead = RTLinuxSysFsReadStrFile(szBuf, sizeof(szBuf), "%s/manufacturer",
1321 pszSysfsPath);
1322 if (cchRead > 0 && (size_t) cchRead < sizeof(szBuf))
1323 {
1324 RTStrPurgeEncoding(szBuf);
1325 Dev->pszManufacturer = RTStrDup(szBuf);
1326 }
1327
1328 /* Work out the port number */
1329 if (RT_FAILURE(usbGetPortFromSysfsPath(pszSysfsPath, &Dev->bPort)))
1330 Dev->enmState = USBDEVICESTATE_UNSUPPORTED;
1331
1332 /* Check the interfaces to see if we can support the device. */
1333 char **ppszIf;
1334 VEC_FOR_EACH(&pInfo->mvecpszInterfaces, char *, ppszIf)
1335 {
1336 ssize_t cb = RTLinuxSysFsGetLinkDest(szBuf, sizeof(szBuf), "%s/driver",
1337 *ppszIf);
1338 if (cb > 0 && Dev->enmState != USBDEVICESTATE_UNSUPPORTED)
1339 Dev->enmState = (strcmp(szBuf, "hub") == 0)
1340 ? USBDEVICESTATE_UNSUPPORTED
1341 : USBDEVICESTATE_USED_BY_HOST_CAPTURABLE;
1342 if (RTLinuxSysFsReadIntFile(16, "%s/bInterfaceClass",
1343 *ppszIf) == 9 /* hub */)
1344 Dev->enmState = USBDEVICESTATE_UNSUPPORTED;
1345 }
1346
1347 /* We use a double slash as a separator in the pszAddress field. This is
1348 * alright as the two paths can't contain a slash due to the way we build
1349 * them. */
1350 char *pszAddress = NULL;
1351 RTStrAPrintf(&pszAddress, "sysfs:%s//device:%s", pszSysfsPath,
1352 pInfo->mDevice);
1353 Dev->pszAddress = pszAddress;
1354
1355 /* Work out from the data collected whether we can support this device. */
1356 Dev->enmState = usbDeterminState(Dev);
1357 usbLogDevice(Dev);
1358}
1359
1360/**
1361 * USBProxyService::getDevices() implementation for sysfs.
1362 */
1363static PUSBDEVICE getDevicesFromSysfs(const char *pcszDevicesRoot, bool testfs)
1364{
1365#ifdef VBOX_USB_WITH_SYSFS
1366 /* Add each of the devices found to the chain. */
1367 PUSBDEVICE pFirst = NULL;
1368 PUSBDEVICE pLast = NULL;
1369 VECTOR_OBJ(USBDeviceInfo) vecDevInfo;
1370 USBDeviceInfo *pInfo;
1371 int rc;
1372
1373 VEC_INIT_OBJ(&vecDevInfo, USBDeviceInfo, USBDevInfoCleanup);
1374 rc = USBSysfsEnumerateHostDevices(pcszDevicesRoot, &vecDevInfo);
1375 if (RT_FAILURE(rc))
1376 return NULL;
1377 VEC_FOR_EACH(&vecDevInfo, USBDeviceInfo, pInfo)
1378 {
1379 USBDEVICE *Dev = (USBDEVICE *)RTMemAllocZ(sizeof(USBDEVICE));
1380 if (!Dev)
1381 rc = VERR_NO_MEMORY;
1382 if (RT_SUCCESS(rc))
1383 {
1384 fillInDeviceFromSysfs(Dev, pInfo);
1385 }
1386 if ( RT_SUCCESS(rc)
1387 && ( Dev->enmState != USBDEVICESTATE_UNSUPPORTED
1388 || testfs)
1389 && Dev->pszAddress != NULL
1390 )
1391 {
1392 if (pLast != NULL)
1393 {
1394 pLast->pNext = Dev;
1395 pLast = pLast->pNext;
1396 }
1397 else
1398 pFirst = pLast = Dev;
1399 }
1400 else
1401 deviceFree(Dev);
1402 if (RT_FAILURE(rc))
1403 break;
1404 }
1405 if (RT_FAILURE(rc))
1406 deviceListFree(&pFirst);
1407
1408 VEC_CLEANUP_OBJ(&vecDevInfo);
1409 return pFirst;
1410#else /* !VBOX_USB_WITH_SYSFS */
1411 return NULL;
1412#endif /* !VBOX_USB_WITH_SYSFS */
1413}
1414
1415#ifdef UNIT_TEST
1416/* Set up mock functions for USBProxyLinuxCheckDeviceRoot - here dlsym and close
1417 * for the inotify presence check. */
1418static int testInotifyInitGood(void) { return 0; }
1419static int testInotifyInitBad(void) { return -1; }
1420static bool s_fHaveInotifyLibC = true;
1421static bool s_fHaveInotifyKernel = true;
1422
1423static void *testDLSym(void *handle, const char *symbol)
1424{
1425 Assert(handle == RTLD_DEFAULT);
1426 Assert(!RTStrCmp(symbol, "inotify_init"));
1427 if (!s_fHaveInotifyLibC)
1428 return NULL;
1429 if (s_fHaveInotifyKernel)
1430 return (void *)(uintptr_t)testInotifyInitGood;
1431 return (void *)(uintptr_t)testInotifyInitBad;
1432}
1433
1434void TestUSBSetInotifyAvailable(bool fHaveInotifyLibC, bool fHaveInotifyKernel)
1435{
1436 s_fHaveInotifyLibC = fHaveInotifyLibC;
1437 s_fHaveInotifyKernel = fHaveInotifyKernel;
1438}
1439# define dlsym testDLSym
1440# define close(a) do {} while (0)
1441#endif
1442
1443/** Is inotify available and working on this system? This is a requirement
1444 * for using USB with sysfs */
1445static bool inotifyAvailable(void)
1446{
1447 int (*inotify_init)(void);
1448
1449 *(void **)(&inotify_init) = dlsym(RTLD_DEFAULT, "inotify_init");
1450 if (!inotify_init)
1451 return false;
1452 int fd = inotify_init();
1453 if (fd == -1)
1454 return false;
1455 close(fd);
1456 return true;
1457}
1458
1459#ifdef UNIT_TEST
1460# undef dlsym
1461# undef close
1462#endif
1463
1464#ifdef UNIT_TEST
1465/** Unit test list of usbfs addresses of connected devices. */
1466static const char **s_pacszUsbfsDeviceAddresses = NULL;
1467
1468static PUSBDEVICE testGetUsbfsDevices(const char *pcszUsbfsRoot, bool testfs)
1469{
1470 const char **pcsz;
1471 PUSBDEVICE pList = NULL, pTail = NULL;
1472 for (pcsz = s_pacszUsbfsDeviceAddresses; pcsz && *pcsz; ++pcsz)
1473 {
1474 PUSBDEVICE pNext = (PUSBDEVICE)RTMemAllocZ(sizeof(USBDEVICE));
1475 if (pNext)
1476 pNext->pszAddress = RTStrDup(*pcsz);
1477 if (!pNext || !pNext->pszAddress)
1478 {
1479 if (pNext)
1480 RTMemFree(pNext);
1481 deviceListFree(&pList);
1482 return NULL;
1483 }
1484 if (pTail)
1485 pTail->pNext = pNext;
1486 else
1487 pList = pNext;
1488 pTail = pNext;
1489 }
1490 return pList;
1491}
1492# define getDevicesFromUsbfs testGetUsbfsDevices
1493
1494/**
1495 * Specify the list of devices that will appear to be available through
1496 * usbfs during unit testing (of USBProxyLinuxGetDevices)
1497 * @param pacszDeviceAddresses NULL terminated array of usbfs device addresses
1498 */
1499void TestUSBSetAvailableUsbfsDevices(const char **pacszDeviceAddresses)
1500{
1501 s_pacszUsbfsDeviceAddresses = pacszDeviceAddresses;
1502}
1503
1504/** Unit test list of files reported as accessible by access(3). We only do
1505 * accessible or not accessible. */
1506static const char **s_pacszAccessibleFiles = NULL;
1507
1508static int testAccess(const char *pcszPath, int mode)
1509{
1510 const char **pcsz;
1511 for (pcsz = s_pacszAccessibleFiles; pcsz && *pcsz; ++pcsz)
1512 if (!RTStrCmp(pcszPath, *pcsz))
1513 return 0;
1514 return -1;
1515}
1516# define access testAccess
1517
1518/**
1519 * Specify the list of files that access will report as accessible (at present
1520 * we only do accessible or not accessible) during unit testing (of
1521 * USBProxyLinuxGetDevices)
1522 * @param pacszAccessibleFiles NULL terminated array of file paths to be
1523 * reported accessible
1524 */
1525void TestUSBSetAccessibleFiles(const char **pacszAccessibleFiles)
1526{
1527 s_pacszAccessibleFiles = pacszAccessibleFiles;
1528}
1529#endif
1530
1531#ifdef UNIT_TEST
1532# ifdef UNIT_TEST
1533 /** The path we pretend the usbfs root is located at, or NULL. */
1534 const char *s_pcszTestUsbfsRoot;
1535 /** Should usbfs be accessible to the current user? */
1536 bool s_fTestUsbfsAccessible;
1537 /** The path we pretend the device node tree root is located at, or NULL. */
1538 const char *s_pcszTestDevicesRoot;
1539 /** Should the device node tree be accessible to the current user? */
1540 bool s_fTestDevicesAccessible;
1541 /** The result of the usbfs/inotify-specific init */
1542 int s_rcTestMethodInitResult;
1543 /** The value of the VBOX_USB environment variable. */
1544 const char *s_pcszTestEnvUsb;
1545 /** The value of the VBOX_USB_ROOT environment variable. */
1546 const char *s_pcszTestEnvUsbRoot;
1547# endif
1548
1549/** Select which access methods will be available to the @a init method
1550 * during unit testing, and (hack!) what return code it will see from
1551 * the access method-specific initialisation. */
1552void TestUSBSetupInit(const char *pcszUsbfsRoot, bool fUsbfsAccessible,
1553 const char *pcszDevicesRoot, bool fDevicesAccessible,
1554 int rcMethodInitResult)
1555{
1556 s_pcszTestUsbfsRoot = pcszUsbfsRoot;
1557 s_fTestUsbfsAccessible = fUsbfsAccessible;
1558 s_pcszTestDevicesRoot = pcszDevicesRoot;
1559 s_fTestDevicesAccessible = fDevicesAccessible;
1560 s_rcTestMethodInitResult = rcMethodInitResult;
1561}
1562
1563/** Specify the environment that the @a init method will see during unit
1564 * testing. */
1565void TestUSBSetEnv(const char *pcszEnvUsb, const char *pcszEnvUsbRoot)
1566{
1567 s_pcszTestEnvUsb = pcszEnvUsb;
1568 s_pcszTestEnvUsbRoot = pcszEnvUsbRoot;
1569}
1570
1571/* For testing we redefine anything that accesses the outside world to
1572 * return test values. */
1573# define RTEnvGet(a) \
1574 ( !RTStrCmp(a, "VBOX_USB") ? s_pcszTestEnvUsb \
1575 : !RTStrCmp(a, "VBOX_USB_ROOT") ? s_pcszTestEnvUsbRoot \
1576 : NULL)
1577# define USBProxyLinuxCheckDeviceRoot(pcszPath, fUseNodes) \
1578 ( ((fUseNodes) && s_fTestDevicesAccessible \
1579 && !RTStrCmp(pcszPath, s_pcszTestDevicesRoot)) \
1580 || (!(fUseNodes) && s_fTestUsbfsAccessible \
1581 && !RTStrCmp(pcszPath, s_pcszTestUsbfsRoot)))
1582# define RTDirExists(pcszDir) \
1583 ( (pcszDir) \
1584 && ( !RTStrCmp(pcszDir, s_pcszTestDevicesRoot) \
1585 || !RTStrCmp(pcszDir, s_pcszTestUsbfsRoot)))
1586# define RTFileExists(pcszFile) \
1587 ( (pcszFile) \
1588 && s_pcszTestUsbfsRoot \
1589 && !RTStrNCmp(pcszFile, s_pcszTestUsbfsRoot, strlen(s_pcszTestUsbfsRoot)) \
1590 && !RTStrCmp(pcszFile + strlen(s_pcszTestUsbfsRoot), "/devices"))
1591#endif
1592
1593/**
1594 * Selects the access method that will be used to access USB devices based on
1595 * what is available on the host and what if anything the user has specified
1596 * in the environment.
1597 * @returns iprt status value
1598 * @param pfUsingUsbfsDevices on success this will be set to true if
1599 * the prefered access method is USBFS-like and to
1600 * false if it is sysfs/device node-like
1601 * @param ppcszDevicesRoot on success the root of the tree of USBFS-like
1602 * device nodes will be stored here
1603 */
1604int USBProxyLinuxChooseMethod(bool *pfUsingUsbfsDevices,
1605 const char **ppcszDevicesRoot)
1606{
1607 /*
1608 * We have two methods available for getting host USB device data - using
1609 * USBFS and using sysfs. The default choice is sysfs; if that is not
1610 * available we fall back to USBFS.
1611 * In the event of both failing, an appropriate error will be returned.
1612 * The user may also specify a method and root using the VBOX_USB and
1613 * VBOX_USB_ROOT environment variables. In this case we don't check
1614 * the root they provide for validity.
1615 */
1616 bool fUsbfsChosen = false, fSysfsChosen = false;
1617 const char *pcszUsbFromEnv = RTEnvGet("VBOX_USB");
1618 const char *pcszUsbRoot = NULL;
1619 if (pcszUsbFromEnv)
1620 {
1621 bool fValidVBoxUSB = true;
1622
1623 pcszUsbRoot = RTEnvGet("VBOX_USB_ROOT");
1624 if (!RTStrICmp(pcszUsbFromEnv, "USBFS"))
1625 {
1626 LogRel(("Default USB access method set to \"usbfs\" from environment\n"));
1627 fUsbfsChosen = true;
1628 }
1629 else if (!RTStrICmp(pcszUsbFromEnv, "SYSFS"))
1630 {
1631 LogRel(("Default USB method set to \"sysfs\" from environment\n"));
1632 fSysfsChosen = true;
1633 }
1634 else
1635 {
1636 LogRel(("Invalid VBOX_USB environment variable setting \"%s\"\n",
1637 pcszUsbFromEnv));
1638 fValidVBoxUSB = false;
1639 pcszUsbFromEnv = NULL;
1640 }
1641 if (!fValidVBoxUSB && pcszUsbRoot)
1642 pcszUsbRoot = NULL;
1643 }
1644 if (!pcszUsbRoot)
1645 {
1646 if ( !fUsbfsChosen
1647 && USBProxyLinuxCheckDeviceRoot("/dev/vboxusb", true))
1648 {
1649 fSysfsChosen = true;
1650 pcszUsbRoot = "/dev/vboxusb";
1651 }
1652 else if ( !fSysfsChosen
1653 && USBProxyLinuxCheckDeviceRoot("/proc/bus/usb", false))
1654 {
1655 fUsbfsChosen = true;
1656 pcszUsbRoot = "/proc/bus/usb";
1657 }
1658 }
1659 else if (!USBProxyLinuxCheckDeviceRoot(pcszUsbRoot, fSysfsChosen))
1660 pcszUsbRoot = NULL;
1661 if (pcszUsbRoot)
1662 {
1663 *pfUsingUsbfsDevices = fUsbfsChosen;
1664 *ppcszDevicesRoot = pcszUsbRoot;
1665 return VINF_SUCCESS;
1666 }
1667 /* else */
1668 return pcszUsbFromEnv ? VERR_NOT_FOUND
1669 : RTDirExists("/dev/vboxusb") ? VERR_VUSB_USB_DEVICE_PERMISSION
1670 : RTFileExists("/proc/bus/usb/devices") ? VERR_VUSB_USBFS_PERMISSION
1671 : VERR_NOT_FOUND;
1672}
1673
1674#ifdef UNIT_TEST
1675# undef RTEnvGet
1676# undef USBProxyLinuxCheckDeviceRoot
1677# undef RTDirExists
1678# undef RTFileExists
1679#endif
1680
1681/**
1682 * Check whether a USB device tree root is usable
1683 * @param pcszRoot the path to the root of the device tree
1684 * @param fIsDeviceNodes whether this is a device node (or usbfs) tree
1685 * @note returns a pointer into a static array so it will stay valid
1686 */
1687bool USBProxyLinuxCheckDeviceRoot(const char *pcszRoot, bool fIsDeviceNodes)
1688{
1689 bool fOK = false;
1690 if (!fIsDeviceNodes) /* usbfs */
1691 {
1692 PUSBDEVICE pDevices;
1693
1694 if (!access(pcszRoot, R_OK | X_OK))
1695 {
1696 fOK = true;
1697 pDevices = getDevicesFromUsbfs(pcszRoot, true);
1698 if (pDevices)
1699 {
1700 PUSBDEVICE pDevice;
1701
1702 for (pDevice = pDevices; pDevice && fOK; pDevice = pDevice->pNext)
1703 if (access(pDevice->pszAddress, R_OK | W_OK))
1704 fOK = false;
1705 deviceListFree(&pDevices);
1706 }
1707 }
1708 }
1709 else /* device nodes */
1710 if (inotifyAvailable() && !access(pcszRoot, R_OK | X_OK))
1711 fOK = true;
1712 return fOK;
1713}
1714
1715#ifdef UNIT_TEST
1716# undef getDevicesFromUsbfs
1717# undef access
1718#endif
1719
1720/**
1721 * Get the list of USB devices supported by the system. Should be freed using
1722 * @a deviceFree or something equivalent.
1723 * @param pcszDevicesRoot the path to the root of the device tree
1724 * @param fUseSysfs whether to use sysfs (or usbfs) for enumeration
1725 */
1726PUSBDEVICE USBProxyLinuxGetDevices(const char *pcszDevicesRoot,
1727 bool fUseSysfs)
1728{
1729 if (!fUseSysfs)
1730 return getDevicesFromUsbfs(pcszDevicesRoot, false);
1731 else
1732 return getDevicesFromSysfs(pcszDevicesRoot, false);
1733}
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette