VirtualBox

source: vbox/trunk/src/VBox/HostDrivers/VBoxUSB/win/lib/VBoxUsbLib-win.cpp@ 44528

Last change on this file since 44528 was 44528, checked in by vboxsync, 12 years ago

header (C) fixes

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 54.8 KB
Line 
1/* $Id: VBoxUsbLib-win.cpp 44528 2013-02-04 14:27:54Z vboxsync $ */
2/** @file
3 * VBox USB ring-3 Driver Interface library, Windows.
4 */
5
6/*
7 * Copyright (C) 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* Header Files *
20*******************************************************************************/
21#define LOG_GROUP LOG_GROUP_DRV_USBPROXY
22#include <windows.h>
23
24#include <VBox/sup.h>
25#include <VBox/types.h>
26#include <VBox/err.h>
27#include <VBox/param.h>
28#include <iprt/path.h>
29#include <iprt/assert.h>
30#include <iprt/alloc.h>
31#include <iprt/string.h>
32#include <iprt/thread.h>
33#include <VBox/log.h>
34#include <VBox/usblib.h>
35#include <VBox/usblib-win.h>
36#include <VBox/usb.h>
37#include <VBox/VBoxDrvCfg-win.h>
38#include <stdio.h>
39#pragma warning (disable:4200) /* shuts up the empty array member warnings */
40#include <setupapi.h>
41#include <usbdi.h>
42#include <hidsdi.h>
43
44#define VBOX_USB_USE_DEVICE_NOTIFICATION
45
46#ifdef VBOX_USB_USE_DEVICE_NOTIFICATION
47# include <Dbt.h>
48#endif
49
50/*******************************************************************************
51* Structures and Typedefs *
52*******************************************************************************/
53typedef struct _USB_INTERFACE_DESCRIPTOR2
54{
55 UCHAR bLength;
56 UCHAR bDescriptorType;
57 UCHAR bInterfaceNumber;
58 UCHAR bAlternateSetting;
59 UCHAR bNumEndpoints;
60 UCHAR bInterfaceClass;
61 UCHAR bInterfaceSubClass;
62 UCHAR bInterfaceProtocol;
63 UCHAR iInterface;
64 USHORT wNumClasses;
65} USB_INTERFACE_DESCRIPTOR2, *PUSB_INTERFACE_DESCRIPTOR2;
66
67typedef struct VBOXUSBGLOBALSTATE
68{
69 HANDLE hMonitor;
70 HANDLE hNotifyEvent;
71 HANDLE hInterruptEvent;
72#ifdef VBOX_USB_USE_DEVICE_NOTIFICATION
73 HANDLE hThread;
74 HWND hWnd;
75 HANDLE hTimerQueue;
76 HANDLE hTimer;
77#endif
78} VBOXUSBGLOBALSTATE, *PVBOXUSBGLOBALSTATE;
79
80typedef struct VBOXUSB_STRING_DR_ENTRY
81{
82 struct VBOXUSB_STRING_DR_ENTRY *pNext;
83 UCHAR iDr;
84 USHORT idLang;
85 USB_STRING_DESCRIPTOR StrDr;
86} VBOXUSB_STRING_DR_ENTRY, *PVBOXUSB_STRING_DR_ENTRY;
87
88/**
89 * This represents VBoxUsb device instance
90 */
91typedef struct VBOXUSB_DEV
92{
93 struct VBOXUSB_DEV *pNext;
94 char szName[512];
95 char szDriverRegName[512];
96} VBOXUSB_DEV, *PVBOXUSB_DEV;
97
98
99/*******************************************************************************
100* Global Variables *
101*******************************************************************************/
102static VBOXUSBGLOBALSTATE g_VBoxUsbGlobal;
103
104
105int usbLibVuDeviceValidate(PVBOXUSB_DEV pVuDev)
106{
107 HANDLE hOut = INVALID_HANDLE_VALUE;
108
109 hOut = CreateFile(pVuDev->szName, GENERIC_READ | GENERIC_WRITE, FILE_SHARE_WRITE | FILE_SHARE_READ, NULL,
110 OPEN_EXISTING, FILE_ATTRIBUTE_SYSTEM, NULL);
111
112 if (hOut == INVALID_HANDLE_VALUE)
113 {
114 DWORD winEr = GetLastError();
115 AssertMsgFailed(("CreateFile FAILED to open %s, winEr (%d)\n", pVuDev->szName, winEr));
116 return VERR_GENERAL_FAILURE;
117 }
118
119 USBSUP_VERSION version = {0};
120 DWORD cbReturned = 0;
121 int rc = VERR_VERSION_MISMATCH;
122
123 do
124 {
125 if (!DeviceIoControl(hOut, SUPUSB_IOCTL_GET_VERSION, NULL, 0,&version, sizeof(version), &cbReturned, NULL))
126 {
127 AssertMsgFailed(("DeviceIoControl SUPUSB_IOCTL_GET_VERSION failed with LastError=%Rwa\n", GetLastError()));
128 break;
129 }
130
131 if (version.u32Major != USBDRV_MAJOR_VERSION
132 || version.u32Minor < USBDRV_MINOR_VERSION)
133 {
134 AssertMsgFailed(("Invalid version %d:%d vs %d:%d\n", version.u32Major, version.u32Minor, USBDRV_MAJOR_VERSION, USBDRV_MINOR_VERSION));
135 break;
136 }
137
138 if (!DeviceIoControl(hOut, SUPUSB_IOCTL_IS_OPERATIONAL, NULL, 0, NULL, NULL, &cbReturned, NULL))
139 {
140 AssertMsgFailed(("DeviceIoControl SUPUSB_IOCTL_IS_OPERATIONAL failed with LastError=%Rwa\n", GetLastError()));
141 break;
142 }
143
144 rc = VINF_SUCCESS;
145 } while (0);
146
147 CloseHandle(hOut);
148 return rc;
149}
150
151static int usbLibVuDevicePopulate(PVBOXUSB_DEV pVuDev, HDEVINFO hDevInfo, PSP_DEVICE_INTERFACE_DATA pIfData)
152{
153 DWORD cbIfDetailData;
154 int rc = VINF_SUCCESS;
155
156 SetupDiGetDeviceInterfaceDetail(hDevInfo, pIfData,
157 NULL, /* OUT PSP_DEVICE_INTERFACE_DETAIL_DATA DeviceInterfaceDetailData */
158 0, /* IN DWORD DeviceInterfaceDetailDataSize */
159 &cbIfDetailData,
160 NULL
161 );
162 Assert(GetLastError() == ERROR_INSUFFICIENT_BUFFER);
163
164 PSP_DEVICE_INTERFACE_DETAIL_DATA pIfDetailData = (PSP_DEVICE_INTERFACE_DETAIL_DATA)RTMemAllocZ(cbIfDetailData);
165 if (!pIfDetailData)
166 {
167 AssertMsgFailed(("RTMemAllocZ failed\n"));
168 return VERR_OUT_OF_RESOURCES;
169 }
170
171 DWORD cbDbgRequired;
172 SP_DEVINFO_DATA DevInfoData;
173 DevInfoData.cbSize = sizeof (DevInfoData);
174 /* the cbSize should contain the sizeof a fixed-size part according to the docs */
175 pIfDetailData->cbSize = sizeof (*pIfDetailData);
176 do
177 {
178 if (!SetupDiGetDeviceInterfaceDetail(hDevInfo, pIfData,
179 pIfDetailData,
180 cbIfDetailData,
181 &cbDbgRequired,
182 &DevInfoData))
183 {
184 DWORD winEr = GetLastError();
185 AssertMsgFailed(("SetupDiGetDeviceInterfaceDetail, cbRequired (%d), was (%d), winEr (%d)\n", cbDbgRequired, cbIfDetailData, winEr));
186 rc = VERR_GENERAL_FAILURE;
187 break;
188 }
189
190 strncpy(pVuDev->szName, pIfDetailData->DevicePath, sizeof (pVuDev->szName));
191
192 if (!SetupDiGetDeviceRegistryPropertyA(hDevInfo, &DevInfoData, SPDRP_DRIVER,
193 NULL, /* OUT PDWORD PropertyRegDataType */
194 (PBYTE)pVuDev->szDriverRegName,
195 sizeof (pVuDev->szDriverRegName),
196 &cbDbgRequired))
197 {
198 DWORD winEr = GetLastError();
199 AssertMsgFailed(("SetupDiGetDeviceRegistryPropertyA, cbRequired (%d), was (%d), winEr (%d)\n", cbDbgRequired, sizeof (pVuDev->szDriverRegName), winEr));
200 rc = VERR_GENERAL_FAILURE;
201 break;
202 }
203
204 rc = usbLibVuDeviceValidate(pVuDev);
205 AssertRC(rc);
206 } while (0);
207
208 RTMemFree(pIfDetailData);
209 return rc;
210}
211
212static void usbLibVuFreeDevices(PVBOXUSB_DEV pDevInfos)
213{
214 while (pDevInfos)
215 {
216 PVBOXUSB_DEV pNext = pDevInfos->pNext;
217 RTMemFree(pDevInfos);
218 pDevInfos = pNext;
219 }
220}
221
222static int usbLibVuGetDevices(PVBOXUSB_DEV *ppVuDevs, uint32_t *pcVuDevs)
223{
224 *ppVuDevs = NULL;
225 *pcVuDevs = 0;
226
227 HDEVINFO hDevInfo = SetupDiGetClassDevs(&GUID_CLASS_VBOXUSB,
228 NULL, /* IN PCTSTR Enumerator */
229 NULL, /* IN HWND hwndParent */
230 (DIGCF_PRESENT | DIGCF_DEVICEINTERFACE) /* IN DWORD Flags */
231 );
232 if (hDevInfo == INVALID_HANDLE_VALUE)
233 {
234 DWORD winEr = GetLastError();
235 AssertMsgFailed(("SetupDiGetClassDevs, winEr (%d)\n", winEr));
236 return VERR_GENERAL_FAILURE;
237 }
238
239 for (int i = 0; ; ++i)
240 {
241 SP_DEVICE_INTERFACE_DATA IfData;
242 IfData.cbSize = sizeof (IfData);
243 if (!SetupDiEnumDeviceInterfaces(hDevInfo,
244 NULL, /* IN PSP_DEVINFO_DATA DeviceInfoData */
245 &GUID_CLASS_VBOXUSB, /* IN LPGUID InterfaceClassGuid */
246 i,
247 &IfData))
248 {
249 DWORD winEr = GetLastError();
250 if (winEr == ERROR_NO_MORE_ITEMS)
251 break;
252
253 AssertMsgFailed(("SetupDiEnumDeviceInterfaces, winEr (%d), resuming\n", winEr));
254 continue;
255 }
256
257 /* we've now got the IfData */
258 PVBOXUSB_DEV pVuDev = (PVBOXUSB_DEV)RTMemAllocZ(sizeof (*pVuDev));
259 if (!pVuDev)
260 {
261 AssertMsgFailed(("RTMemAllocZ failed, resuming\n"));
262 continue;
263 }
264
265 int rc = usbLibVuDevicePopulate(pVuDev, hDevInfo, &IfData);
266 if (!RT_SUCCESS(rc))
267 {
268 AssertMsgFailed(("usbLibVuDevicePopulate failed, rc (%d), resuming\n", rc));
269 continue;
270 }
271
272 pVuDev->pNext = *ppVuDevs;
273 *ppVuDevs = pVuDev;
274 ++*pcVuDevs;
275 }
276
277 SetupDiDestroyDeviceInfoList(hDevInfo);
278
279 return VINF_SUCCESS;
280}
281
282static void usbLibDevFree(PUSBDEVICE pDevice)
283{
284 RTStrFree((char*)pDevice->pszAddress);
285 RTStrFree((char*)pDevice->pszHubName);
286 if (pDevice->pszManufacturer)
287 RTStrFree((char*)pDevice->pszManufacturer);
288 if (pDevice->pszProduct)
289 RTStrFree((char*)pDevice->pszProduct);
290 if (pDevice->pszSerialNumber)
291 RTStrFree((char*)pDevice->pszSerialNumber);
292 RTMemFree(pDevice);
293}
294
295static void usbLibDevFreeList(PUSBDEVICE pDevice)
296{
297 while (pDevice)
298 {
299 PUSBDEVICE pNext = pDevice->pNext;
300 usbLibDevFree(pDevice);
301 pDevice = pNext;
302 }
303}
304
305static int usbLibDevPopulate(PUSBDEVICE pDev, PUSB_NODE_CONNECTION_INFORMATION_EX pConInfo, ULONG iPort, LPCSTR lpszDrvKeyName, LPCSTR lpszHubName, PVBOXUSB_STRING_DR_ENTRY pDrList)
306{
307 pDev->bcdUSB = pConInfo->DeviceDescriptor.bcdUSB;
308 pDev->bDeviceClass = pConInfo->DeviceDescriptor.bDeviceClass;
309 pDev->bDeviceSubClass = pConInfo->DeviceDescriptor.bDeviceSubClass;
310 pDev->bDeviceProtocol = pConInfo->DeviceDescriptor.bDeviceProtocol;
311 pDev->idVendor = pConInfo->DeviceDescriptor.idVendor;
312 pDev->idProduct = pConInfo->DeviceDescriptor.idProduct;
313 pDev->bcdDevice = pConInfo->DeviceDescriptor.bcdDevice;
314 pDev->bBus = 0; /** @todo figure out bBus on windows... */
315 pDev->bPort = iPort;
316 /** @todo check which devices are used for primary input (keyboard & mouse) */
317 if (!lpszDrvKeyName || *lpszDrvKeyName == 0)
318 pDev->enmState = USBDEVICESTATE_UNUSED;
319 else
320 pDev->enmState = USBDEVICESTATE_USED_BY_HOST_CAPTURABLE;
321 pDev->enmSpeed = USBDEVICESPEED_UNKNOWN;
322 pDev->pszAddress = RTStrDup(lpszDrvKeyName);
323 pDev->pszHubName = RTStrDup(lpszHubName);
324 pDev->bNumConfigurations = 0;
325 pDev->u64SerialHash = 0;
326
327 for (; pDrList; pDrList = pDrList->pNext)
328 {
329 char ** lppszString = NULL;
330 if (pConInfo->DeviceDescriptor.iManufacturer && pDrList->iDr == pConInfo->DeviceDescriptor.iManufacturer)
331 {
332 lppszString = (char**)&pDev->pszManufacturer;
333 }
334 else if (pConInfo->DeviceDescriptor.iProduct && pDrList->iDr == pConInfo->DeviceDescriptor.iProduct)
335 {
336 lppszString = (char**)&pDev->pszProduct;
337 }
338 else if (pConInfo->DeviceDescriptor.iSerialNumber && pDrList->iDr == pConInfo->DeviceDescriptor.iSerialNumber)
339 {
340 lppszString = (char**)&pDev->pszSerialNumber;
341 }
342
343 if (lppszString)
344 {
345/** @todo r=bird: This code is making bad asumptions that strings are sane and
346 * that stuff succeeds:
347 * http://vbox.innotek.de/pipermail/vbox-dev/2011-August/004516.html
348 *
349 * */
350 int rc = RTUtf16ToUtf8((PCRTUTF16)pDrList->StrDr.bString, lppszString);
351 if (RT_FAILURE(rc))
352 {
353 AssertMsgFailed(("RTUtf16ToUtf8 failed, rc (%d), resuming\n", rc));
354 continue;
355 }
356
357 Assert(lppszString);
358 if (pDrList->iDr == pConInfo->DeviceDescriptor.iSerialNumber)
359 {
360 pDev->u64SerialHash = USBLibHashSerial(pDev->pszSerialNumber);
361 }
362 }
363 }
364
365 return VINF_SUCCESS;
366}
367
368static void usbLibDevStrFree(LPSTR lpszName)
369{
370 RTStrFree(lpszName);
371}
372
373static int usbLibDevStrDriverKeyGet(HANDLE hHub, ULONG iPort, LPSTR* plpszName)
374{
375 USB_NODE_CONNECTION_DRIVERKEY_NAME Name;
376 DWORD cbReturned = 0;
377 Name.ConnectionIndex = iPort;
378 *plpszName = NULL;
379 if (!DeviceIoControl(hHub, IOCTL_USB_GET_NODE_CONNECTION_DRIVERKEY_NAME, &Name, sizeof (Name), &Name, sizeof (Name), &cbReturned, NULL))
380 {
381#ifdef VBOX_WITH_ANNOYING_USB_ASSERTIONS
382 DWORD winEr = GetLastError();
383 AssertMsgFailed(("DeviceIoControl 1 fail winEr (%d)\n", winEr));
384#endif
385 return VERR_GENERAL_FAILURE;
386 }
387
388 if (Name.ActualLength < sizeof (Name))
389 {
390 AssertFailed();
391 return VERR_OUT_OF_RESOURCES;
392 }
393
394 PUSB_NODE_CONNECTION_DRIVERKEY_NAME pName = (PUSB_NODE_CONNECTION_DRIVERKEY_NAME)RTMemAllocZ(Name.ActualLength);
395 if (!pName)
396 {
397 AssertFailed();
398 return VERR_OUT_OF_RESOURCES;
399 }
400
401 int rc = VINF_SUCCESS;
402 pName->ConnectionIndex = iPort;
403 if (DeviceIoControl(hHub, IOCTL_USB_GET_NODE_CONNECTION_DRIVERKEY_NAME, pName, Name.ActualLength, pName, Name.ActualLength, &cbReturned, NULL))
404 {
405 rc = RTUtf16ToUtf8Ex((PCRTUTF16)pName->DriverKeyName, pName->ActualLength / sizeof (WCHAR), plpszName, 0, NULL);
406 AssertRC(rc);
407 if (RT_SUCCESS(rc))
408 rc = VINF_SUCCESS;
409 }
410 else
411 {
412 DWORD winEr = GetLastError();
413 AssertMsgFailed(("DeviceIoControl 2 fail winEr (%d)\n", winEr));
414 rc = VERR_GENERAL_FAILURE;
415 }
416 RTMemFree(pName);
417 return rc;
418}
419
420static int usbLibDevStrHubNameGet(HANDLE hHub, ULONG iPort, LPSTR* plpszName)
421{
422 USB_NODE_CONNECTION_NAME Name;
423 DWORD cbReturned = 0;
424 Name.ConnectionIndex = iPort;
425 *plpszName = NULL;
426 if (!DeviceIoControl(hHub, IOCTL_USB_GET_NODE_CONNECTION_NAME, &Name, sizeof (Name), &Name, sizeof (Name), &cbReturned, NULL))
427 {
428 AssertFailed();
429 return VERR_GENERAL_FAILURE;
430 }
431
432 if (Name.ActualLength < sizeof (Name))
433 {
434 AssertFailed();
435 return VERR_OUT_OF_RESOURCES;
436 }
437
438 PUSB_NODE_CONNECTION_NAME pName = (PUSB_NODE_CONNECTION_NAME)RTMemAllocZ(Name.ActualLength);
439 if (!pName)
440 {
441 AssertFailed();
442 return VERR_OUT_OF_RESOURCES;
443 }
444
445 int rc = VINF_SUCCESS;
446 pName->ConnectionIndex = iPort;
447 if (DeviceIoControl(hHub, IOCTL_USB_GET_NODE_CONNECTION_NAME, pName, Name.ActualLength, pName, Name.ActualLength, &cbReturned, NULL))
448 {
449 rc = RTUtf16ToUtf8Ex((PCRTUTF16)pName->NodeName, pName->ActualLength / sizeof (WCHAR), plpszName, 0, NULL);
450 AssertRC(rc);
451 if (RT_SUCCESS(rc))
452 rc = VINF_SUCCESS;
453 }
454 else
455 {
456 AssertFailed();
457 rc = VERR_GENERAL_FAILURE;
458 }
459 RTMemFree(pName);
460 return rc;
461}
462
463static int usbLibDevStrRootHubNameGet(HANDLE hCtl, LPSTR* plpszName)
464{
465 USB_ROOT_HUB_NAME HubName;
466 DWORD cbReturned = 0;
467 *plpszName = NULL;
468 if (!DeviceIoControl(hCtl, IOCTL_USB_GET_ROOT_HUB_NAME, NULL, 0, &HubName, sizeof (HubName), &cbReturned, NULL))
469 {
470 return VERR_GENERAL_FAILURE;
471 }
472 PUSB_ROOT_HUB_NAME pHubName = (PUSB_ROOT_HUB_NAME)RTMemAllocZ(HubName.ActualLength);
473 if (!pHubName)
474 return VERR_OUT_OF_RESOURCES;
475
476 int rc = VINF_SUCCESS;
477 if (DeviceIoControl(hCtl, IOCTL_USB_GET_ROOT_HUB_NAME, NULL, 0, pHubName, HubName.ActualLength, &cbReturned, NULL))
478 {
479 rc = RTUtf16ToUtf8Ex((PCRTUTF16)pHubName->RootHubName, pHubName->ActualLength / sizeof (WCHAR), plpszName, 0, NULL);
480 AssertRC(rc);
481 if (RT_SUCCESS(rc))
482 rc = VINF_SUCCESS;
483 }
484 else
485 {
486 rc = VERR_GENERAL_FAILURE;
487 }
488 RTMemFree(pHubName);
489 return rc;
490}
491
492static int usbLibDevCfgDrGet(HANDLE hHub, ULONG iPort, ULONG iDr, PUSB_CONFIGURATION_DESCRIPTOR *ppDr)
493{
494 *ppDr = NULL;
495
496 char Buf[sizeof (USB_DESCRIPTOR_REQUEST) + sizeof (USB_CONFIGURATION_DESCRIPTOR)];
497 memset(&Buf, 0, sizeof (Buf));
498
499 PUSB_DESCRIPTOR_REQUEST pCfgDrRq = (PUSB_DESCRIPTOR_REQUEST)Buf;
500 PUSB_CONFIGURATION_DESCRIPTOR pCfgDr = (PUSB_CONFIGURATION_DESCRIPTOR)(Buf + sizeof (*pCfgDrRq));
501
502 pCfgDrRq->ConnectionIndex = iPort;
503 pCfgDrRq->SetupPacket.wValue = (USB_CONFIGURATION_DESCRIPTOR_TYPE << 8) | iDr;
504 pCfgDrRq->SetupPacket.wLength = (USHORT)(sizeof (USB_CONFIGURATION_DESCRIPTOR));
505 DWORD cbReturned = 0;
506 if (!DeviceIoControl(hHub, IOCTL_USB_GET_DESCRIPTOR_FROM_NODE_CONNECTION, pCfgDrRq, sizeof (Buf),
507 pCfgDrRq, sizeof (Buf),
508 &cbReturned, NULL))
509 {
510 DWORD winEr = GetLastError();
511 LogRel((__FUNCTION__": DeviceIoControl 1 fail winEr (%d)\n", winEr));
512#ifdef VBOX_WITH_ANNOYING_USB_ASSERTIONS
513 AssertFailed();
514#endif
515 return VERR_GENERAL_FAILURE;
516 }
517
518 if (sizeof (Buf) != cbReturned)
519 {
520 AssertFailed();
521 return VERR_GENERAL_FAILURE;
522 }
523
524 if (pCfgDr->wTotalLength < sizeof (USB_CONFIGURATION_DESCRIPTOR))
525 {
526 AssertFailed();
527 return VERR_GENERAL_FAILURE;
528 }
529
530 DWORD cbRq = sizeof (USB_DESCRIPTOR_REQUEST) + pCfgDr->wTotalLength;
531 PUSB_DESCRIPTOR_REQUEST pRq = (PUSB_DESCRIPTOR_REQUEST)RTMemAllocZ(cbRq);
532 Assert(pRq);
533 if (!pRq)
534 return VERR_OUT_OF_RESOURCES;
535
536 int rc = VERR_GENERAL_FAILURE;
537 do
538 {
539 PUSB_CONFIGURATION_DESCRIPTOR pDr = (PUSB_CONFIGURATION_DESCRIPTOR)(pRq + 1);
540 pRq->ConnectionIndex = iPort;
541 pRq->SetupPacket.wValue = (USB_CONFIGURATION_DESCRIPTOR_TYPE << 8) | iDr;
542 pRq->SetupPacket.wLength = (USHORT)(cbRq - sizeof (USB_DESCRIPTOR_REQUEST));
543 if (!DeviceIoControl(hHub, IOCTL_USB_GET_DESCRIPTOR_FROM_NODE_CONNECTION, pRq, cbRq,
544 pRq, cbRq,
545 &cbReturned, NULL))
546 {
547 DWORD winEr = GetLastError();
548 LogRel((__FUNCTION__": DeviceIoControl 2 fail winEr (%d)\n", winEr));
549#ifdef VBOX_WITH_ANNOYING_USB_ASSERTIONS
550 AssertFailed();
551#endif
552 break;
553 }
554
555 if (cbRq != cbReturned)
556 {
557 AssertFailed();
558 break;
559 }
560
561 if (pDr->wTotalLength != cbRq - sizeof (USB_DESCRIPTOR_REQUEST))
562 {
563 AssertFailed();
564 break;
565 }
566
567 *ppDr = pDr;
568 return VINF_SUCCESS;
569 } while (0);
570
571 RTMemFree(pRq);
572 return rc;
573}
574
575static void usbLibDevCfgDrFree(PUSB_CONFIGURATION_DESCRIPTOR pDr)
576{
577 Assert(pDr);
578 PUSB_DESCRIPTOR_REQUEST pRq = ((PUSB_DESCRIPTOR_REQUEST)pDr)-1;
579 RTMemFree(pRq);
580}
581
582static int usbLibDevStrDrEntryGet(HANDLE hHub, ULONG iPort, ULONG iDr, USHORT idLang, PVBOXUSB_STRING_DR_ENTRY *ppList)
583{
584 char Buf[sizeof (USB_DESCRIPTOR_REQUEST) + MAXIMUM_USB_STRING_LENGTH];
585 PUSB_DESCRIPTOR_REQUEST pRq = (PUSB_DESCRIPTOR_REQUEST)Buf;
586 PUSB_STRING_DESCRIPTOR pDr = (PUSB_STRING_DESCRIPTOR)(Buf + sizeof (*pRq));
587 memset (&Buf, 0, sizeof (Buf));
588 pRq->ConnectionIndex = iPort;
589 pRq->SetupPacket.wValue = (USB_STRING_DESCRIPTOR_TYPE << 8) | iDr;
590 pRq->SetupPacket.wIndex = idLang;
591 pRq->SetupPacket.wLength = sizeof (Buf) - sizeof (*pRq);
592 DWORD cbReturned = 0;
593 if (!DeviceIoControl(hHub, IOCTL_USB_GET_DESCRIPTOR_FROM_NODE_CONNECTION, pRq, sizeof (Buf),
594 pRq, sizeof (Buf),
595 &cbReturned, NULL))
596 {
597 DWORD winEr = GetLastError();
598 LogRel((__FUNCTION__": DeviceIoControl 1 fail winEr (%d)\n", winEr));
599#ifdef VBOX_WITH_ANNOYING_USB_ASSERTIONS
600 AssertFailed();
601#endif
602 return VERR_GENERAL_FAILURE;
603 }
604
605 if (cbReturned < sizeof (*pDr) + 2)
606 {
607 AssertFailed();
608 return VERR_GENERAL_FAILURE;
609 }
610
611 if (!!(pDr->bLength % 2))
612 {
613 AssertFailed();
614 return VERR_GENERAL_FAILURE;
615 }
616
617 if (pDr->bLength != cbReturned - sizeof (*pRq))
618 {
619 AssertFailed();
620 return VERR_GENERAL_FAILURE;
621 }
622
623 if (pDr->bDescriptorType != USB_STRING_DESCRIPTOR_TYPE)
624 {
625 AssertFailed();
626 return VERR_GENERAL_FAILURE;
627 }
628
629 PVBOXUSB_STRING_DR_ENTRY pEntry = (PVBOXUSB_STRING_DR_ENTRY)RTMemAllocZ(sizeof (*pEntry) + pDr->bLength + 2);
630 Assert(pEntry);
631 if (!pEntry)
632 {
633 return VERR_OUT_OF_RESOURCES;
634 }
635
636 pEntry->pNext = *ppList;
637 pEntry->iDr = iDr;
638 pEntry->idLang = idLang;
639 memcpy(&pEntry->StrDr, pDr, pDr->bLength);
640 *ppList = pEntry;
641 return VINF_SUCCESS;
642}
643
644static void usbLibDevStrDrEntryFree(PVBOXUSB_STRING_DR_ENTRY pDr)
645{
646 RTMemFree(pDr);
647}
648
649static void usbLibDevStrDrEntryFreeList(PVBOXUSB_STRING_DR_ENTRY pDr)
650{
651 while (pDr)
652 {
653 PVBOXUSB_STRING_DR_ENTRY pNext = pDr->pNext;
654 usbLibDevStrDrEntryFree(pDr);
655 pDr = pNext;
656 }
657}
658
659static int usbLibDevStrDrEntryGetForLangs(HANDLE hHub, ULONG iPort, ULONG iDr, ULONG cIdLang, const USHORT *pIdLang, PVBOXUSB_STRING_DR_ENTRY *ppList)
660{
661 for (ULONG i = 0; i < cIdLang; ++i)
662 {
663 usbLibDevStrDrEntryGet(hHub, iPort, iDr, pIdLang[i], ppList);
664 }
665 return VINF_SUCCESS;
666}
667
668static int usbLibDevStrDrEntryGetAll(HANDLE hHub, ULONG iPort, PUSB_DEVICE_DESCRIPTOR pDevDr, PUSB_CONFIGURATION_DESCRIPTOR pCfgDr, PVBOXUSB_STRING_DR_ENTRY *ppList)
669{
670 int rc = usbLibDevStrDrEntryGet(hHub, iPort, 0, 0, ppList);
671#ifdef VBOX_WITH_ANNOYING_USB_ASSERTIONS
672 AssertRC(rc);
673#endif
674 if (RT_FAILURE(rc))
675 return rc;
676
677 PUSB_STRING_DESCRIPTOR pLandStrDr = &(*ppList)->StrDr;
678 USHORT *pIdLang = pLandStrDr->bString;
679 ULONG cIdLang = (pLandStrDr->bLength - RT_OFFSETOF(USB_STRING_DESCRIPTOR, bString)) / sizeof (*pIdLang);
680
681 if (pDevDr->iManufacturer)
682 {
683 rc = usbLibDevStrDrEntryGetForLangs(hHub, iPort, pDevDr->iManufacturer, cIdLang, pIdLang, ppList);
684 AssertRC(rc);
685 }
686
687 if (pDevDr->iProduct)
688 {
689 rc = usbLibDevStrDrEntryGetForLangs(hHub, iPort, pDevDr->iProduct, cIdLang, pIdLang, ppList);
690 AssertRC(rc);
691 }
692
693 if (pDevDr->iSerialNumber)
694 {
695 rc = usbLibDevStrDrEntryGetForLangs(hHub, iPort, pDevDr->iSerialNumber, cIdLang, pIdLang, ppList);
696 AssertRC(rc);
697 }
698
699 PUCHAR pCur = (PUCHAR)pCfgDr;
700 PUCHAR pEnd = pCur + pCfgDr->wTotalLength;
701 while (pCur + sizeof (USB_COMMON_DESCRIPTOR) <= pEnd)
702 {
703 PUSB_COMMON_DESCRIPTOR pCmnDr = (PUSB_COMMON_DESCRIPTOR)pCur;
704 if (pCur + pCmnDr->bLength > pEnd)
705 {
706 AssertFailed();
707 break;
708 }
709
710 switch (pCmnDr->bDescriptorType)
711 {
712 case USB_CONFIGURATION_DESCRIPTOR_TYPE:
713 {
714 if (pCmnDr->bLength != sizeof (USB_CONFIGURATION_DESCRIPTOR))
715 {
716 AssertFailed();
717 break;
718 }
719 PUSB_CONFIGURATION_DESCRIPTOR pCurCfgDr = (PUSB_CONFIGURATION_DESCRIPTOR)pCmnDr;
720 if (!pCurCfgDr->iConfiguration)
721 break;
722 rc = usbLibDevStrDrEntryGetForLangs(hHub, iPort, pCurCfgDr->iConfiguration, cIdLang, pIdLang, ppList);
723 AssertRC(rc);
724 break;
725 }
726 case USB_INTERFACE_DESCRIPTOR_TYPE:
727 {
728 if (pCmnDr->bLength != sizeof (USB_INTERFACE_DESCRIPTOR) && pCmnDr->bLength != sizeof (USB_INTERFACE_DESCRIPTOR2))
729 {
730 AssertFailed();
731 break;
732 }
733 PUSB_INTERFACE_DESCRIPTOR pCurIfDr = (PUSB_INTERFACE_DESCRIPTOR)pCmnDr;
734 if (!pCurIfDr->iInterface)
735 break;
736 rc = usbLibDevStrDrEntryGetForLangs(hHub, iPort, pCurIfDr->iInterface, cIdLang, pIdLang, ppList);
737 AssertRC(rc);
738 break;
739 }
740 default:
741 break;
742 }
743
744 pCur = pCur + pCmnDr->bLength;
745 }
746
747 return VINF_SUCCESS;
748}
749
750static int usbLibDevGetHubDevices(LPCSTR lpszName, PUSBDEVICE *ppDevs, uint32_t *pcDevs);
751
752static int usbLibDevGetHubPortDevices(HANDLE hHub, LPCSTR lpcszHubName, ULONG iPort, PUSBDEVICE *ppDevs, uint32_t *pcDevs)
753{
754 int rc = VINF_SUCCESS;
755 char Buf[sizeof (USB_NODE_CONNECTION_INFORMATION_EX) + (sizeof (USB_PIPE_INFO) * 20)];
756 PUSB_NODE_CONNECTION_INFORMATION_EX pConInfo = (PUSB_NODE_CONNECTION_INFORMATION_EX)Buf;
757 PUSB_PIPE_INFO paPipeInfo = (PUSB_PIPE_INFO)(Buf + sizeof (PUSB_NODE_CONNECTION_INFORMATION_EX));
758 DWORD cbReturned = 0;
759 memset(&Buf, 0, sizeof (Buf));
760 pConInfo->ConnectionIndex = iPort;
761 if (!DeviceIoControl(hHub, IOCTL_USB_GET_NODE_CONNECTION_INFORMATION_EX,
762 pConInfo, sizeof (Buf),
763 pConInfo, sizeof (Buf),
764 &cbReturned, NULL))
765 {
766 DWORD winEr = GetLastError();
767 AssertMsg(winEr == ERROR_DEVICE_NOT_CONNECTED, (__FUNCTION__": DeviceIoControl failed winEr (%d)\n", winEr));
768 return VERR_GENERAL_FAILURE;
769 }
770
771 if (pConInfo->ConnectionStatus != DeviceConnected)
772 {
773 /* just ignore & return success */
774 return VWRN_INVALID_HANDLE;
775 }
776
777 if (pConInfo->DeviceIsHub)
778 {
779 LPSTR lpszHubName = NULL;
780 rc = usbLibDevStrHubNameGet(hHub, iPort, &lpszHubName);
781 AssertRC(rc);
782 if (RT_SUCCESS(rc))
783 {
784 rc = usbLibDevGetHubDevices(lpszHubName, ppDevs, pcDevs);
785 usbLibDevStrFree(lpszHubName);
786 AssertRC(rc);
787 return rc;
788 }
789 /* ignore this err */
790 return VINF_SUCCESS;
791 }
792
793 bool fFreeNameBuf = true;
794 char nameEmptyBuf = '\0';
795 LPSTR lpszName = NULL;
796 rc = usbLibDevStrDriverKeyGet(hHub, iPort, &lpszName);
797 Assert(!!lpszName == !!RT_SUCCESS(rc));
798 if (!lpszName)
799 {
800 lpszName = &nameEmptyBuf;
801 fFreeNameBuf = false;
802 }
803
804 PUSB_CONFIGURATION_DESCRIPTOR pCfgDr = NULL;
805 PVBOXUSB_STRING_DR_ENTRY pList = NULL;
806 rc = usbLibDevCfgDrGet(hHub, iPort, 0, &pCfgDr);
807 if (pCfgDr)
808 {
809 rc = usbLibDevStrDrEntryGetAll(hHub, iPort, &pConInfo->DeviceDescriptor, pCfgDr, &pList);
810#ifdef VBOX_WITH_ANNOYING_USB_ASSERTIONS
811 AssertRC(rc);
812#endif
813 }
814
815 PUSBDEVICE pDev = (PUSBDEVICE)RTMemAllocZ(sizeof (*pDev));
816 rc = usbLibDevPopulate(pDev, pConInfo, iPort, lpszName, lpcszHubName, pList);
817 AssertRC(rc);
818 if (RT_SUCCESS(rc))
819 {
820 pDev->pNext = *ppDevs;
821 *ppDevs = pDev;
822 ++*pcDevs;
823 }
824
825 if (pCfgDr)
826 usbLibDevCfgDrFree(pCfgDr);
827 if (fFreeNameBuf)
828 {
829 Assert(lpszName);
830 usbLibDevStrFree(lpszName);
831 }
832 if (pList)
833 usbLibDevStrDrEntryFreeList(pList);
834
835 return VINF_SUCCESS;
836}
837
838static int usbLibDevGetHubDevices(LPCSTR lpszName, PUSBDEVICE *ppDevs, uint32_t *pcDevs)
839{
840 LPSTR lpszDevName = (LPSTR)RTMemAllocZ(strlen(lpszName) + sizeof("\\\\.\\"));
841 HANDLE hDev = INVALID_HANDLE_VALUE;
842 Assert(lpszDevName);
843 if (!lpszDevName)
844 {
845 AssertFailed();
846 return VERR_OUT_OF_RESOURCES;
847 }
848
849 int rc = VINF_SUCCESS;
850 strcpy(lpszDevName, "\\\\.\\");
851 strcpy(lpszDevName + sizeof("\\\\.\\") - sizeof (lpszDevName[0]), lpszName);
852 do
853 {
854 DWORD cbReturned = 0;
855 hDev = CreateFile(lpszDevName, GENERIC_WRITE, FILE_SHARE_WRITE, NULL, OPEN_EXISTING, 0, NULL);
856 if (hDev == INVALID_HANDLE_VALUE)
857 {
858 AssertFailed();
859 break;
860 }
861
862 USB_NODE_INFORMATION NodeInfo;
863 memset(&NodeInfo, 0, sizeof (NodeInfo));
864 if (!DeviceIoControl(hDev, IOCTL_USB_GET_NODE_INFORMATION,
865 &NodeInfo, sizeof (NodeInfo),
866 &NodeInfo, sizeof (NodeInfo),
867 &cbReturned, NULL))
868 {
869 AssertFailed();
870 break;
871 }
872
873 for (ULONG i = 1; i <= NodeInfo.u.HubInformation.HubDescriptor.bNumberOfPorts; ++i)
874 {
875 usbLibDevGetHubPortDevices(hDev, lpszName, i, ppDevs, pcDevs);
876 }
877 } while (0);
878
879 if (hDev != INVALID_HANDLE_VALUE)
880 CloseHandle(hDev);
881
882 RTMemFree(lpszDevName);
883
884 return rc;
885}
886
887static int usbLibDevGetDevices(PUSBDEVICE *ppDevs, uint32_t *pcDevs)
888{
889 char CtlName[16];
890 int rc = VINF_SUCCESS;
891
892 for (int i = 0; i < 10; ++i)
893 {
894 sprintf(CtlName, "\\\\.\\HCD%d", i);
895 HANDLE hCtl = CreateFile(CtlName, GENERIC_WRITE, FILE_SHARE_WRITE, NULL, OPEN_EXISTING, 0, NULL);
896 if (hCtl != INVALID_HANDLE_VALUE)
897 {
898 char* lpszName;
899 rc = usbLibDevStrRootHubNameGet(hCtl, &lpszName);
900 AssertRC(rc);
901 if (RT_SUCCESS(rc))
902 {
903 rc = usbLibDevGetHubDevices(lpszName, ppDevs, pcDevs);
904 AssertRC(rc);
905 usbLibDevStrFree(lpszName);
906 }
907 CloseHandle(hCtl);
908 if (RT_FAILURE(rc))
909 break;
910 }
911 }
912 return VINF_SUCCESS;
913}
914
915static PUSBSUP_GET_DEVICES usbLibMonGetDevRqAlloc(uint32_t cDevs, PDWORD pcbRq)
916{
917 DWORD cbRq = RT_OFFSETOF(USBSUP_GET_DEVICES, aDevices[cDevs]);
918 PUSBSUP_GET_DEVICES pRq = (PUSBSUP_GET_DEVICES)RTMemAllocZ(cbRq);
919 Assert(pRq);
920 if (!pRq)
921 return NULL;
922 pRq->cDevices = cDevs;
923 *pcbRq = cbRq;
924 return pRq;
925}
926
927static int usbLibMonDevicesCmp(PUSBDEVICE pDev, PVBOXUSB_DEV pDevInfo)
928{
929 int iDiff;
930 iDiff = strcmp(pDev->pszAddress, pDevInfo->szDriverRegName);
931 return iDiff;
932}
933
934static int usbLibMonDevicesUpdate(PVBOXUSBGLOBALSTATE pGlobal, PUSBDEVICE pDevs, uint32_t cDevs, PVBOXUSB_DEV pDevInfos, uint32_t cDevInfos)
935{
936 PUSBDEVICE pDevsHead = pDevs;
937 for (; pDevInfos; pDevInfos = pDevInfos->pNext)
938 {
939 for (pDevs = pDevsHead; pDevs; pDevs = pDevs->pNext)
940 {
941 if (usbLibMonDevicesCmp(pDevs, pDevInfos))
942 continue;
943
944 if (!pDevInfos->szDriverRegName[0])
945 {
946 AssertFailed();
947 break;
948 }
949
950 USBSUP_GETDEV Dev = {0};
951 HANDLE hDev = CreateFile(pDevInfos->szName, GENERIC_READ | GENERIC_WRITE, FILE_SHARE_WRITE | FILE_SHARE_READ, NULL,
952 OPEN_EXISTING, FILE_ATTRIBUTE_SYSTEM, NULL);
953 if (hDev == INVALID_HANDLE_VALUE)
954 {
955 AssertFailed();
956 break;
957 }
958
959 DWORD cbReturned = 0;
960 if (!DeviceIoControl(hDev, SUPUSB_IOCTL_GET_DEVICE, &Dev, sizeof (Dev), &Dev, sizeof (Dev), &cbReturned, NULL))
961 {
962 DWORD winEr = GetLastError();
963 /* ERROR_DEVICE_NOT_CONNECTED -> device was removed just now */
964#ifdef VBOX_WITH_ANNOYING_USB_ASSERTIONS
965 AssertMsg(winEr == ERROR_DEVICE_NOT_CONNECTED, (__FUNCTION__": DeviceIoControl failed winEr (%d)\n", winEr));
966#endif
967 Log(("SUPUSB_IOCTL_GET_DEVICE: DeviceIoControl no longer connected\n"));
968 CloseHandle(hDev);
969 break;
970 }
971
972 /* we must not close the handle until we request for the device state from the monitor to ensure
973 * the device handle returned by the device driver does not disappear */
974 Assert(Dev.hDevice);
975 USBSUP_GETDEV_MON MonInfo;
976 HVBOXUSBDEVUSR hDevice = Dev.hDevice;
977 if (!DeviceIoControl(pGlobal->hMonitor, SUPUSBFLT_IOCTL_GET_DEVICE, &hDevice, sizeof (hDevice), &MonInfo, sizeof (MonInfo), &cbReturned, NULL))
978 {
979 DWORD winEr = GetLastError();
980 /* ERROR_DEVICE_NOT_CONNECTED -> device was removed just now */
981 AssertMsgFailed(("Monitor DeviceIoControl failed winEr (%d)\n", winEr));
982 Log(("SUPUSBFLT_IOCTL_GET_DEVICE: DeviceIoControl no longer connected\n"));
983 CloseHandle(hDev);
984 break;
985 }
986
987 CloseHandle(hDev);
988
989 /* success!! update device info */
990 /* ensure the state returned is valid */
991 Assert( MonInfo.enmState == USBDEVICESTATE_USED_BY_HOST
992 || MonInfo.enmState == USBDEVICESTATE_USED_BY_HOST_CAPTURABLE
993 || MonInfo.enmState == USBDEVICESTATE_UNUSED
994 || MonInfo.enmState == USBDEVICESTATE_HELD_BY_PROXY
995 || MonInfo.enmState == USBDEVICESTATE_USED_BY_GUEST);
996 pDevs->enmState = MonInfo.enmState;
997 /* The following is not 100% accurate but we only care about high-speed vs. non-high-speed */
998 pDevs->enmSpeed = Dev.fHiSpeed ? USBDEVICESPEED_HIGH : USBDEVICESPEED_FULL;
999 if (pDevs->enmState != USBDEVICESTATE_USED_BY_HOST)
1000 {
1001 /* only set the interface name if device can be grabbed */
1002 RTStrFree(pDevs->pszAltAddress);
1003 pDevs->pszAltAddress = (char*)pDevs->pszAddress;
1004 pDevs->pszAddress = RTStrDup(pDevInfos->szName);
1005 }
1006#ifdef VBOX_WITH_ANNOYING_USB_ASSERTIONS
1007 else
1008 {
1009 /* dbg breakpoint */
1010 Assert(0);
1011 }
1012#endif
1013
1014 /* we've found the device, break in any way */
1015 break;
1016 }
1017 }
1018
1019 return VINF_SUCCESS;
1020}
1021
1022static int usbLibGetDevices(PVBOXUSBGLOBALSTATE pGlobal, PUSBDEVICE *ppDevs, uint32_t *pcDevs)
1023{
1024 *ppDevs = NULL;
1025 *pcDevs = 0;
1026
1027 int rc = usbLibDevGetDevices(ppDevs, pcDevs);
1028 AssertRC(rc);
1029 if (RT_SUCCESS(rc))
1030 {
1031 PVBOXUSB_DEV pDevInfos = NULL;
1032 uint32_t cDevInfos = 0;
1033 rc = usbLibVuGetDevices(&pDevInfos, &cDevInfos);
1034 AssertRC(rc);
1035 if (RT_SUCCESS(rc))
1036 {
1037 rc = usbLibMonDevicesUpdate(pGlobal, *ppDevs, *pcDevs, pDevInfos, cDevInfos);
1038 AssertRC(rc);
1039 usbLibVuFreeDevices(pDevInfos);
1040 }
1041
1042 return VINF_SUCCESS;
1043 }
1044 return rc;
1045}
1046
1047AssertCompile(INFINITE == RT_INDEFINITE_WAIT);
1048static int usbLibStateWaitChange(PVBOXUSBGLOBALSTATE pGlobal, RTMSINTERVAL cMillies)
1049{
1050 HANDLE ahEvents[] = {pGlobal->hNotifyEvent, pGlobal->hInterruptEvent};
1051 DWORD dwResult = WaitForMultipleObjects(RT_ELEMENTS(ahEvents), ahEvents,
1052 FALSE, /* BOOL bWaitAll */
1053 cMillies);
1054
1055 switch (dwResult)
1056 {
1057 case WAIT_OBJECT_0:
1058 return VINF_SUCCESS;
1059 case WAIT_OBJECT_0 + 1:
1060 return VERR_INTERRUPTED;
1061 case WAIT_TIMEOUT:
1062 return VERR_TIMEOUT;
1063 default:
1064 {
1065 DWORD winEr = GetLastError();
1066 AssertMsgFailed(("WaitForMultipleObjects failed, winEr (%d)\n", winEr));
1067 return VERR_GENERAL_FAILURE;
1068 }
1069 }
1070}
1071
1072AssertCompile(RT_INDEFINITE_WAIT == INFINITE);
1073AssertCompile(sizeof (RTMSINTERVAL) == sizeof (DWORD));
1074USBLIB_DECL(int) USBLibWaitChange(RTMSINTERVAL msWaitTimeout)
1075{
1076 return usbLibStateWaitChange(&g_VBoxUsbGlobal, msWaitTimeout);
1077}
1078
1079static int usbLibInterruptWaitChange(PVBOXUSBGLOBALSTATE pGlobal)
1080{
1081 BOOL bRc = SetEvent(pGlobal->hInterruptEvent);
1082 if (!bRc)
1083 {
1084 DWORD winEr = GetLastError();
1085 AssertMsgFailed(("SetEvent failed, winEr (%d)\n", winEr));
1086 return VERR_GENERAL_FAILURE;
1087 }
1088 return VINF_SUCCESS;
1089}
1090
1091USBLIB_DECL(int) USBLibInterruptWaitChange()
1092{
1093 return usbLibInterruptWaitChange(&g_VBoxUsbGlobal);
1094}
1095
1096/*
1097USBLIB_DECL(bool) USBLibHasPendingDeviceChanges(void)
1098{
1099 int rc = USBLibWaitChange(0);
1100 return rc == VINF_SUCCESS;
1101}
1102*/
1103
1104USBLIB_DECL(int) USBLibGetDevices(PUSBDEVICE *ppDevices, uint32_t *pcbNumDevices)
1105{
1106 Assert(g_VBoxUsbGlobal.hMonitor != INVALID_HANDLE_VALUE);
1107 return usbLibGetDevices(&g_VBoxUsbGlobal, ppDevices, pcbNumDevices);
1108}
1109
1110USBLIB_DECL(void *) USBLibAddFilter(PCUSBFILTER pFilter)
1111{
1112 USBSUP_FLTADDOUT FltAddRc;
1113 DWORD cbReturned = 0;
1114
1115 if (g_VBoxUsbGlobal.hMonitor == INVALID_HANDLE_VALUE)
1116 {
1117#ifdef VBOX_WITH_ANNOYING_USB_ASSERTIONS
1118 AssertFailed();
1119#endif
1120 return NULL;
1121 }
1122
1123 Log(("usblibInsertFilter: Manufacturer=%s Product=%s Serial=%s\n",
1124 USBFilterGetString(pFilter, USBFILTERIDX_MANUFACTURER_STR) ? USBFilterGetString(pFilter, USBFILTERIDX_MANUFACTURER_STR) : "<null>",
1125 USBFilterGetString(pFilter, USBFILTERIDX_PRODUCT_STR) ? USBFilterGetString(pFilter, USBFILTERIDX_PRODUCT_STR) : "<null>",
1126 USBFilterGetString(pFilter, USBFILTERIDX_SERIAL_NUMBER_STR) ? USBFilterGetString(pFilter, USBFILTERIDX_SERIAL_NUMBER_STR) : "<null>"));
1127
1128 if (!DeviceIoControl(g_VBoxUsbGlobal.hMonitor, SUPUSBFLT_IOCTL_ADD_FILTER,
1129 (LPVOID)pFilter, sizeof(*pFilter),
1130 &FltAddRc, sizeof(FltAddRc),
1131 &cbReturned, NULL))
1132 {
1133 DWORD winEr = GetLastError();
1134 AssertMsgFailed(("DeviceIoControl failed with winEr (%d(\n", winEr));
1135 return NULL;
1136 }
1137
1138 if (RT_FAILURE(FltAddRc.rc))
1139 {
1140 AssertMsgFailed(("Adding filter failed with %d\n", FltAddRc.rc));
1141 return NULL;
1142 }
1143 return (void *)FltAddRc.uId;
1144}
1145
1146
1147USBLIB_DECL(void) USBLibRemoveFilter(void *pvId)
1148{
1149 uintptr_t uId;
1150 DWORD cbReturned = 0;
1151
1152 if (g_VBoxUsbGlobal.hMonitor == INVALID_HANDLE_VALUE)
1153 {
1154#ifdef VBOX_WITH_ANNOYING_USB_ASSERTIONS
1155 AssertFailed();
1156#endif
1157 return;
1158 }
1159
1160 Log(("usblibRemoveFilter %p\n", pvId));
1161
1162 uId = (uintptr_t)pvId;
1163 if (!DeviceIoControl(g_VBoxUsbGlobal.hMonitor, SUPUSBFLT_IOCTL_REMOVE_FILTER, &uId, sizeof(uId), NULL, 0,&cbReturned, NULL))
1164 AssertMsgFailed(("DeviceIoControl failed with LastError=%Rwa\n", GetLastError()));
1165}
1166
1167USBLIB_DECL(int) USBLibRunFilters()
1168{
1169 DWORD cbReturned = 0;
1170
1171 Assert(g_VBoxUsbGlobal.hMonitor != INVALID_HANDLE_VALUE);
1172
1173 if (!DeviceIoControl(g_VBoxUsbGlobal.hMonitor, SUPUSBFLT_IOCTL_RUN_FILTERS,
1174 NULL, 0,
1175 NULL, 0,
1176 &cbReturned, NULL))
1177 {
1178 DWORD winEr = GetLastError();
1179 AssertMsgFailed(("DeviceIoControl failed with winEr (%d(\n", winEr));
1180 return RTErrConvertFromWin32(winEr);
1181 }
1182
1183 return VINF_SUCCESS;
1184}
1185
1186
1187#ifdef VBOX_USB_USE_DEVICE_NOTIFICATION
1188
1189static VOID CALLBACK usbLibTimerCallback(__in PVOID lpParameter, __in BOOLEAN TimerOrWaitFired)
1190{
1191 SetEvent(g_VBoxUsbGlobal.hNotifyEvent);
1192}
1193
1194static void usbLibOnDeviceChange(void)
1195{
1196 /* we're getting series of events like that especially on device re-attach
1197 * (i.e. first for device detach and then for device attach)
1198 * unfortunately the event does not tell us what actually happened.
1199 * To avoid extra notifications, we delay the SetEvent via a timer
1200 * and update the timer if additional notification comes before the timer fires
1201 * */
1202 if (g_VBoxUsbGlobal.hTimer)
1203 {
1204 if (!DeleteTimerQueueTimer(g_VBoxUsbGlobal.hTimerQueue, g_VBoxUsbGlobal.hTimer, NULL))
1205 {
1206 DWORD winEr = GetLastError();
1207 AssertMsg(winEr == ERROR_IO_PENDING, ("DeleteTimerQueueTimer failed, winEr (%d)\n", winEr));
1208 }
1209 }
1210
1211 if (!CreateTimerQueueTimer(&g_VBoxUsbGlobal.hTimer, g_VBoxUsbGlobal.hTimerQueue,
1212 usbLibTimerCallback,
1213 NULL,
1214 500, /* ms*/
1215 0,
1216 WT_EXECUTEONLYONCE))
1217 {
1218 DWORD winEr = GetLastError();
1219 AssertMsgFailed(("CreateTimerQueueTimer failed, winEr (%d)\n", winEr));
1220
1221 /* call it directly */
1222 usbLibTimerCallback(NULL, FALSE);
1223 }
1224}
1225
1226static LRESULT CALLBACK usbLibWndProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
1227{
1228 switch (uMsg)
1229 {
1230 case WM_DEVICECHANGE:
1231 if (wParam == DBT_DEVNODES_CHANGED)
1232 {
1233 /* we notify change any device arivals/removals on the system
1234 * and let the client decide whether the usb change actually happened
1235 * so far this is more clean than reporting events from the Monitor
1236 * because monitor sees only PDO arrivals/removals,
1237 * and by the time PDO is created, device can not
1238 * be yet started and fully functional,
1239 * so usblib won't be able to pick it up
1240 * */
1241
1242 usbLibOnDeviceChange();
1243 }
1244 break;
1245 case WM_DESTROY:
1246 return 0;
1247 }
1248 return DefWindowProc (hwnd, uMsg, wParam, lParam);
1249}
1250
1251/** @todo r=bird: Use an IPRT thread? */
1252static DWORD WINAPI usbLibMsgThreadProc(__in LPVOID lpParameter)
1253{
1254 static LPCSTR s_szVBoxUsbWndClassName = "VBoxUsbLibClass";
1255 const HINSTANCE hInstance = (HINSTANCE)GetModuleHandle(NULL);
1256
1257 Assert(g_VBoxUsbGlobal.hWnd == NULL);
1258 g_VBoxUsbGlobal.hWnd = NULL;
1259
1260 /*
1261 * Register the Window Class and the hitten window create.
1262 */
1263 WNDCLASS wc;
1264 wc.style = 0;
1265 wc.lpfnWndProc = usbLibWndProc;
1266 wc.cbClsExtra = 0;
1267 wc.cbWndExtra = sizeof(void *);
1268 wc.hInstance = hInstance;
1269 wc.hIcon = NULL;
1270 wc.hCursor = NULL;
1271 wc.hbrBackground = (HBRUSH)(COLOR_BACKGROUND + 1);
1272 wc.lpszMenuName = NULL;
1273 wc.lpszClassName = s_szVBoxUsbWndClassName;
1274 ATOM atomWindowClass = RegisterClass(&wc);
1275 if (atomWindowClass != 0)
1276 g_VBoxUsbGlobal.hWnd = CreateWindowEx(WS_EX_TOOLWINDOW | WS_EX_TRANSPARENT | WS_EX_TOPMOST,
1277 s_szVBoxUsbWndClassName, s_szVBoxUsbWndClassName,
1278 WS_POPUPWINDOW,
1279 -200, -200, 100, 100, NULL, NULL, hInstance, NULL);
1280 else
1281 AssertMsgFailed(("RegisterClass failed, last error %u\n", GetLastError()));
1282
1283 /*
1284 * Signal the creator thread.
1285 */
1286 ASMCompilerBarrier();
1287 SetEvent(g_VBoxUsbGlobal.hNotifyEvent);
1288
1289 if (g_VBoxUsbGlobal.hWnd)
1290 {
1291 /* Make sure it's really hidden. */
1292 SetWindowPos(g_VBoxUsbGlobal.hWnd, HWND_TOPMOST, -200, -200, 0, 0,
1293 SWP_NOACTIVATE | SWP_HIDEWINDOW | SWP_NOCOPYBITS | SWP_NOREDRAW | SWP_NOSIZE);
1294
1295 /*
1296 * The message pump.
1297 */
1298 MSG msg;
1299 while (GetMessage(&msg, NULL, 0, 0))
1300 {
1301 TranslateMessage(&msg);
1302 DispatchMessage(&msg);
1303 }
1304
1305 /*
1306 * Clean up.
1307 */
1308 DestroyWindow(g_VBoxUsbGlobal.hWnd);
1309 }
1310
1311 if (atomWindowClass != NULL)
1312 UnregisterClass(s_szVBoxUsbWndClassName, hInstance);
1313
1314 return 0;
1315}
1316
1317#endif /* VBOX_USB_USE_DEVICE_NOTIFICATION */
1318
1319/**
1320 * Initialize the USB library
1321 *
1322 * @returns VBox status code.
1323 */
1324USBLIB_DECL(int) USBLibInit(void)
1325{
1326 int rc = VERR_GENERAL_FAILURE;
1327
1328 Log(("usbproxy: usbLibInit\n"));
1329
1330 RT_ZERO(g_VBoxUsbGlobal);
1331 g_VBoxUsbGlobal.hMonitor = INVALID_HANDLE_VALUE;
1332
1333 /*
1334 * Create the notification and interrupt event before opening the device.
1335 */
1336 g_VBoxUsbGlobal.hNotifyEvent = CreateEvent(NULL, /* LPSECURITY_ATTRIBUTES lpEventAttributes */
1337 FALSE, /* BOOL bManualReset */
1338#ifndef VBOX_USB_USE_DEVICE_NOTIFICATION
1339 TRUE, /* BOOL bInitialState */
1340#else
1341 FALSE, /* set to false since it will be initially used for notification thread startup sync */
1342#endif
1343 NULL /* LPCTSTR lpName */);
1344 if (g_VBoxUsbGlobal.hNotifyEvent)
1345 {
1346 g_VBoxUsbGlobal.hInterruptEvent = CreateEvent(NULL, /* LPSECURITY_ATTRIBUTES lpEventAttributes */
1347 FALSE, /* BOOL bManualReset */
1348 FALSE, /* BOOL bInitialState */
1349 NULL /* LPCTSTR lpName */);
1350 if (g_VBoxUsbGlobal.hInterruptEvent)
1351 {
1352 /*
1353 * Open the USB monitor device, starting if needed.
1354 */
1355 g_VBoxUsbGlobal.hMonitor = CreateFile(USBMON_DEVICE_NAME,
1356 GENERIC_READ | GENERIC_WRITE,
1357 FILE_SHARE_READ | FILE_SHARE_WRITE,
1358 NULL,
1359 OPEN_EXISTING,
1360 FILE_ATTRIBUTE_SYSTEM,
1361 NULL);
1362
1363 if (g_VBoxUsbGlobal.hMonitor == INVALID_HANDLE_VALUE)
1364 {
1365 HRESULT hr = VBoxDrvCfgSvcStart(USBMON_SERVICE_NAME_W);
1366 if (hr == S_OK)
1367 {
1368 g_VBoxUsbGlobal.hMonitor = CreateFile(USBMON_DEVICE_NAME,
1369 GENERIC_READ | GENERIC_WRITE,
1370 FILE_SHARE_READ | FILE_SHARE_WRITE,
1371 NULL,
1372 OPEN_EXISTING,
1373 FILE_ATTRIBUTE_SYSTEM,
1374 NULL);
1375 if (g_VBoxUsbGlobal.hMonitor == INVALID_HANDLE_VALUE)
1376 {
1377 DWORD winEr = GetLastError();
1378 LogRel((__FUNCTION__": CreateFile failed winEr(%d)\n", winEr));
1379 rc = VERR_FILE_NOT_FOUND;
1380 }
1381 }
1382 }
1383
1384 if (g_VBoxUsbGlobal.hMonitor != INVALID_HANDLE_VALUE)
1385 {
1386 /*
1387 * Check the USB monitor version.
1388 *
1389 * Drivers are backwards compatible within the same major
1390 * number. We consider the minor version number this library
1391 * is compiled with to be the minimum required by the driver.
1392 * This is by reasoning that the library uses the full feature
1393 * set of the driver it's written for.
1394 */
1395 USBSUP_VERSION Version = {0};
1396 DWORD cbReturned = 0;
1397 if (DeviceIoControl(g_VBoxUsbGlobal.hMonitor, SUPUSBFLT_IOCTL_GET_VERSION,
1398 NULL, 0,
1399 &Version, sizeof (Version),
1400 &cbReturned, NULL))
1401 {
1402 if ( Version.u32Major == USBMON_MAJOR_VERSION
1403 && Version.u32Minor >= USBMON_MINOR_VERSION)
1404 {
1405#ifndef VBOX_USB_USE_DEVICE_NOTIFICATION
1406 /*
1407 * Tell the monitor driver which event object to use
1408 * for notifications.
1409 */
1410 USBSUP_SET_NOTIFY_EVENT SetEvent = {0};
1411 Assert(g_VBoxUsbGlobal.hNotifyEvent);
1412 SetEvent.u.hEvent = g_VBoxUsbGlobal.hNotifyEvent;
1413 if (DeviceIoControl(g_VBoxUsbGlobal.hMonitor, SUPUSBFLT_IOCTL_SET_NOTIFY_EVENT,
1414 &SetEvent, sizeof(SetEvent),
1415 &SetEvent, sizeof(SetEvent),
1416 &cbReturned, NULL))
1417 {
1418 rc = SetEvent.u.rc;
1419 if (RT_SUCCESS(rc))
1420 {
1421 /*
1422 * We're DONE!
1423 */
1424 return VINF_SUCCESS;
1425 }
1426
1427 AssertMsgFailed(("SetEvent failed, %Rrc (%d)\n", rc, rc));
1428 }
1429 else
1430 {
1431 DWORD winEr = GetLastError();
1432 AssertMsgFailed(("SetEvent Ioctl failed, winEr (%d)\n", winEr));
1433 rc = VERR_VERSION_MISMATCH;
1434 }
1435#else
1436 /*
1437 * We can not use USB Mon for reliable device add/remove tracking
1438 * since once USB Mon is notified about PDO creation and/or IRP_MN_START_DEVICE,
1439 * the function device driver may still do some initialization, which might result in
1440 * notifying too early.
1441 * Instead we use WM_DEVICECHANGE + DBT_DEVNODES_CHANGED to make Windows notify us about
1442 * device arivals/removals.
1443 * Since WM_DEVICECHANGE is a window message, create a dedicated thread to be used for WndProc and stuff.
1444 * The thread would create a window, track windows messages and call usbLibOnDeviceChange on WM_DEVICECHANGE arrival.
1445 * See comments in usbLibOnDeviceChange function for detail about using the timer queue.
1446 */
1447 g_VBoxUsbGlobal.hTimerQueue = CreateTimerQueue();
1448 if (g_VBoxUsbGlobal.hTimerQueue)
1449 {
1450 g_VBoxUsbGlobal.hThread = CreateThread(
1451 NULL, /*__in_opt LPSECURITY_ATTRIBUTES lpThreadAttributes, */
1452 0, /*__in SIZE_T dwStackSize, */
1453 usbLibMsgThreadProc, /*__in LPTHREAD_START_ROUTINE lpStartAddress,*/
1454 NULL, /*__in_opt LPVOID lpParameter,*/
1455 0, /*__in DWORD dwCreationFlags,*/
1456 NULL /*__out_opt LPDWORD lpThreadId*/
1457 );
1458 if (g_VBoxUsbGlobal.hThread)
1459 {
1460 DWORD dwResult = WaitForSingleObject(g_VBoxUsbGlobal.hNotifyEvent, INFINITE);
1461 Assert(dwResult == WAIT_OBJECT_0);
1462 if (g_VBoxUsbGlobal.hWnd)
1463 {
1464 /*
1465 * We're DONE!
1466 *
1467 * Juse ensure that the event is set so the
1468 * first "wait change" request is processed.
1469 */
1470 SetEvent(g_VBoxUsbGlobal.hNotifyEvent);
1471 return VINF_SUCCESS;
1472 }
1473
1474 dwResult = WaitForSingleObject(g_VBoxUsbGlobal.hThread, INFINITE);
1475 Assert(dwResult == WAIT_OBJECT_0);
1476 BOOL bRc = CloseHandle(g_VBoxUsbGlobal.hThread);
1477 AssertMsg(bRc, ("CloseHandle for hThread failed winEr(%d)\n", GetLastError()));
1478 g_VBoxUsbGlobal.hThread = INVALID_HANDLE_VALUE;
1479 }
1480 else
1481 {
1482 DWORD winEr = GetLastError();
1483 AssertMsgFailed(("CreateThread failed, winEr (%d)\n", winEr));
1484 rc = VERR_GENERAL_FAILURE;
1485 }
1486
1487 DeleteTimerQueueEx(g_VBoxUsbGlobal.hTimerQueue, INVALID_HANDLE_VALUE /* see term */);
1488 g_VBoxUsbGlobal.hTimerQueue = NULL;
1489 }
1490 else
1491 {
1492 DWORD winEr = GetLastError();
1493 AssertMsgFailed(("CreateTimerQueue failed winEr(%d)\n", winEr));
1494 }
1495#endif
1496 }
1497 else
1498 {
1499 LogRel((__FUNCTION__": USB Monitor driver version mismatch! driver=%u.%u library=%u.%u\n",
1500 Version.u32Major, Version.u32Minor, USBMON_MAJOR_VERSION, USBMON_MINOR_VERSION));
1501#ifdef VBOX_WITH_ANNOYING_USB_ASSERTIONS
1502 AssertFailed();
1503#endif
1504 rc = VERR_VERSION_MISMATCH;
1505 }
1506 }
1507 else
1508 {
1509 DWORD winEr = GetLastError();
1510 AssertMsgFailed(("DeviceIoControl failed winEr(%d)\n", winEr));
1511 rc = VERR_VERSION_MISMATCH;
1512 }
1513
1514 CloseHandle(g_VBoxUsbGlobal.hMonitor);
1515 g_VBoxUsbGlobal.hMonitor = INVALID_HANDLE_VALUE;
1516 }
1517 else
1518 {
1519 LogRel((__FUNCTION__": USB Service not found\n"));
1520#ifdef VBOX_WITH_ANNOYING_USB_ASSERTIONS
1521 AssertFailed();
1522#endif
1523 rc = VERR_FILE_NOT_FOUND;
1524 }
1525
1526 CloseHandle(g_VBoxUsbGlobal.hInterruptEvent);
1527 g_VBoxUsbGlobal.hInterruptEvent = NULL;
1528 }
1529 else
1530 {
1531 DWORD winEr = GetLastError();
1532 AssertMsgFailed(("CreateEvent for InterruptEvent failed winEr(%d)\n", winEr));
1533 rc = VERR_GENERAL_FAILURE;
1534 }
1535
1536 CloseHandle(g_VBoxUsbGlobal.hNotifyEvent);
1537 g_VBoxUsbGlobal.hNotifyEvent = NULL;
1538 }
1539 else
1540 {
1541 DWORD winEr = GetLastError();
1542 AssertMsgFailed(("CreateEvent for NotifyEvent failed winEr(%d)\n", winEr));
1543 rc = VERR_GENERAL_FAILURE;
1544 }
1545
1546 /* since main calls us even if USBLibInit fails,
1547 * we use hMonitor == INVALID_HANDLE_VALUE as a marker to indicate whether the lib is inited */
1548
1549 Assert(RT_FAILURE(rc));
1550 return rc;
1551}
1552
1553
1554/**
1555 * Terminate the USB library
1556 *
1557 * @returns VBox status code.
1558 */
1559USBLIB_DECL(int) USBLibTerm(void)
1560{
1561 if (g_VBoxUsbGlobal.hMonitor == INVALID_HANDLE_VALUE)
1562 {
1563 Assert(g_VBoxUsbGlobal.hInterruptEvent == NULL);
1564 Assert(g_VBoxUsbGlobal.hNotifyEvent == NULL);
1565 return VINF_SUCCESS;
1566 }
1567
1568 BOOL bRc;
1569#ifdef VBOX_USB_USE_DEVICE_NOTIFICATION
1570 bRc = PostMessage(g_VBoxUsbGlobal.hWnd, WM_QUIT, 0, 0);
1571 AssertMsg(bRc, ("PostMessage for hWnd failed winEr(%d)\n", GetLastError()));
1572
1573 if (g_VBoxUsbGlobal.hThread != NULL)
1574 {
1575 DWORD dwResult = WaitForSingleObject(g_VBoxUsbGlobal.hThread, INFINITE);
1576 Assert(dwResult == WAIT_OBJECT_0);
1577 bRc = CloseHandle(g_VBoxUsbGlobal.hThread);
1578 AssertMsg(bRc, ("CloseHandle for hThread failed winEr(%d)\n", GetLastError()));
1579 }
1580
1581 if (g_VBoxUsbGlobal.hTimer)
1582 {
1583 bRc = DeleteTimerQueueTimer(g_VBoxUsbGlobal.hTimerQueue, g_VBoxUsbGlobal.hTimer,
1584 INVALID_HANDLE_VALUE); /* <-- to block until the timer is completed */
1585 AssertMsg(bRc, ("DeleteTimerQueueTimer failed winEr(%d)\n", GetLastError()));
1586 }
1587
1588 if (g_VBoxUsbGlobal.hTimerQueue)
1589 {
1590 bRc = DeleteTimerQueueEx(g_VBoxUsbGlobal.hTimerQueue,
1591 INVALID_HANDLE_VALUE); /* <-- to block until all timers are completed */
1592 AssertMsg(bRc, ("DeleteTimerQueueEx failed winEr(%d)\n", GetLastError()));
1593 }
1594#endif /* VBOX_USB_USE_DEVICE_NOTIFICATION */
1595
1596 bRc = CloseHandle(g_VBoxUsbGlobal.hMonitor);
1597 AssertMsg(bRc, ("CloseHandle for hMonitor failed winEr(%d)\n", GetLastError()));
1598 g_VBoxUsbGlobal.hMonitor = INVALID_HANDLE_VALUE;
1599
1600 bRc = CloseHandle(g_VBoxUsbGlobal.hInterruptEvent);
1601 AssertMsg(bRc, ("CloseHandle for hInterruptEvent failed lasterr=%u\n", GetLastError()));
1602 g_VBoxUsbGlobal.hInterruptEvent = NULL;
1603
1604 bRc = CloseHandle(g_VBoxUsbGlobal.hNotifyEvent);
1605 AssertMsg(bRc, ("CloseHandle for hNotifyEvent failed winEr(%d)\n", GetLastError()));
1606 g_VBoxUsbGlobal.hNotifyEvent = NULL;
1607
1608 return VINF_SUCCESS;
1609}
1610
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