VirtualBox

source: vbox/trunk/src/VBox/Devices/USB/USBProxyDevice.cpp@ 61648

Last change on this file since 61648 was 61247, checked in by vboxsync, 9 years ago

VUSB: Shorten initial descriptor read timeout.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 39.8 KB
Line 
1/* $Id: USBProxyDevice.cpp 61247 2016-05-27 14:32:05Z vboxsync $ */
2/** @file
3 * USBProxy - USB device proxy.
4 */
5
6/*
7 * Copyright (C) 2006-2015 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#define LOG_GROUP LOG_GROUP_DRV_USBPROXY
23#include <VBox/usb.h>
24#include <VBox/usbfilter.h>
25#include <VBox/vmm/pdm.h>
26#include <VBox/err.h>
27#include <iprt/alloc.h>
28#include <iprt/string.h>
29#include <VBox/log.h>
30#include <iprt/assert.h>
31#include "USBProxyDevice.h"
32#include "VUSBInternal.h"
33#include "VBoxDD.h"
34
35
36/*********************************************************************************************************************************
37* Global Variables *
38*********************************************************************************************************************************/
39/** A dummy name used early during the construction phase to avoid log crashes. */
40static char g_szDummyName[] = "proxy xxxx:yyyy";
41
42/**
43 * Array of supported proxy backends.
44 */
45static PCUSBPROXYBACK g_aUsbProxies[] =
46{
47 &g_USBProxyDeviceHost,
48 &g_USBProxyDeviceVRDP,
49 &g_USBProxyDeviceUsbIp
50};
51
52/* Synchronously obtain a standard USB descriptor for a device, used in order
53 * to grab configuration descriptors when we first add the device
54 */
55static void *GetStdDescSync(PUSBPROXYDEV pProxyDev, uint8_t iDescType, uint8_t iIdx, uint16_t LangId, uint16_t cbHint)
56{
57#define GET_DESC_RETRIES 6
58 int cRetries = 0;
59 uint16_t cbInitialHint = cbHint;
60
61 LogFlow(("GetStdDescSync: pProxyDev=%s\n", pProxyDev->pUsbIns->pszName));
62 for (;;)
63 {
64 /*
65 * Setup a MSG URB, queue and reap it.
66 */
67 int rc = VINF_SUCCESS;
68 VUSBURB Urb;
69 AssertCompile(RT_SIZEOFMEMB(VUSBURB, abData) >= _4K);
70 Urb.u32Magic = VUSBURB_MAGIC;
71 Urb.enmState = VUSBURBSTATE_IN_FLIGHT;
72 Urb.pszDesc = (char*)"URB sync";
73 Urb.pHci = NULL;
74 Urb.paTds = NULL;
75 Urb.Dev.pvPrivate = NULL;
76 Urb.Dev.pNext = NULL;
77 Urb.DstAddress = 0;
78 Urb.EndPt = 0;
79 Urb.enmType = VUSBXFERTYPE_MSG;
80 Urb.enmDir = VUSBDIRECTION_IN;
81 Urb.fShortNotOk = false;
82 Urb.enmStatus = VUSBSTATUS_INVALID;
83 Urb.pVUsb = NULL;
84 cbHint = RT_MIN(cbHint, sizeof(Urb.abData) - sizeof(VUSBSETUP));
85 Urb.cbData = cbHint + sizeof(VUSBSETUP);
86
87 PVUSBSETUP pSetup = (PVUSBSETUP)Urb.abData;
88 pSetup->bmRequestType = VUSB_DIR_TO_HOST | VUSB_REQ_STANDARD | VUSB_TO_DEVICE;
89 pSetup->bRequest = VUSB_REQ_GET_DESCRIPTOR;
90 pSetup->wValue = (iDescType << 8) | iIdx;
91 pSetup->wIndex = LangId;
92 pSetup->wLength = cbHint;
93
94 uint8_t *pbDesc = (uint8_t *)(pSetup + 1);
95 uint32_t cbDesc = 0;
96 PVUSBURB pUrbReaped = NULL;
97
98 rc = pProxyDev->pOps->pfnUrbQueue(pProxyDev, &Urb);
99 if (RT_FAILURE(rc))
100 {
101 Log(("GetStdDescSync: pfnUrbReap failed, rc=%d\n", rc));
102 goto err;
103 }
104
105 /* Don't wait forever, it's just a simple request that should
106 return immediately. Since we're executing in the EMT thread
107 it's important not to get stuck here. (Some of the builtin
108 iMac devices may refuse to respond for instance.) */
109 pUrbReaped = pProxyDev->pOps->pfnUrbReap(pProxyDev, 5000 /* ms */);
110 if (!pUrbReaped)
111 {
112 rc = pProxyDev->pOps->pfnUrbCancel(pProxyDev, &Urb);
113 AssertRC(rc);
114 /** @todo: This breaks the comment above... */
115 pUrbReaped = pProxyDev->pOps->pfnUrbReap(pProxyDev, RT_INDEFINITE_WAIT);
116 }
117 if (pUrbReaped != &Urb)
118 {
119 Log(("GetStdDescSync: pfnUrbReap failed, pUrbReaped=%p\n", pUrbReaped));
120 goto err;
121 }
122
123 if (Urb.enmStatus != VUSBSTATUS_OK)
124 {
125 Log(("GetStdDescSync: Urb.enmStatus=%d\n", Urb.enmStatus));
126 goto err;
127 }
128
129 /*
130 * Check the length, config descriptors have total_length field
131 */
132 if (iDescType == VUSB_DT_CONFIG)
133 {
134 if (Urb.cbData < sizeof(VUSBSETUP) + 4)
135 {
136 Log(("GetStdDescSync: Urb.cbData=%#x (min 4)\n", Urb.cbData));
137 goto err;
138 }
139 cbDesc = RT_LE2H_U16(((uint16_t *)pbDesc)[1]);
140 }
141 else
142 {
143 if (Urb.cbData < sizeof(VUSBSETUP) + 1)
144 {
145 Log(("GetStdDescSync: Urb.cbData=%#x (min 1)\n", Urb.cbData));
146 goto err;
147 }
148 cbDesc = ((uint8_t *)pbDesc)[0];
149 }
150
151 Log(("GetStdDescSync: got Urb.cbData=%u, cbDesc=%u cbHint=%u\n", Urb.cbData, cbDesc, cbHint));
152
153 if ( Urb.cbData == cbHint + sizeof(VUSBSETUP)
154 && cbDesc > Urb.cbData - sizeof(VUSBSETUP))
155 {
156 cbHint = cbDesc;
157 Log(("GetStdDescSync: Part descriptor, Urb.cbData=%u, cbDesc=%u cbHint=%u\n", Urb.cbData, cbDesc, cbHint));
158
159 if (cbHint > sizeof(Urb.abData))
160 Log(("GetStdDescSync: cbHint=%u, Urb.abData=%u\n", cbHint, sizeof(Urb.abData)));
161
162 goto err;
163 }
164
165 if ((cbDesc > (Urb.cbData - sizeof(VUSBSETUP))))
166 {
167 Log(("GetStdDescSync: Descriptor length too short, cbDesc=%u, Urb.cbData=%u\n", cbDesc, Urb.cbData));
168 goto err;
169 }
170
171 if ((cbInitialHint != cbHint) &&
172 ((cbDesc != cbHint) || (Urb.cbData < cbInitialHint)))
173 {
174 Log(("GetStdDescSync: Descriptor length incorrect, cbDesc=%u, Urb.cbData=%u, cbHint=%u\n", cbDesc, Urb.cbData, cbHint));
175 goto err;
176 }
177
178#ifdef LOG_ENABLED
179 vusbUrbTrace(&Urb, "GetStdDescSync", true);
180#endif
181
182 /*
183 * Fine, we got everything return a heap duplicate of the descriptor.
184 */
185 return RTMemDup(pbDesc, cbDesc);
186
187err:
188 cRetries++;
189 if (cRetries < GET_DESC_RETRIES)
190 {
191 Log(("GetStdDescSync: Retrying %u/%u\n", cRetries, GET_DESC_RETRIES));
192 RTThreadSleep(100);
193 continue;
194 }
195 else
196 {
197 Log(("GetStdDescSync: Retries exceeded %u/%u. Giving up.\n", cRetries, GET_DESC_RETRIES));
198 break;
199 }
200 }
201
202 return NULL;
203}
204
205/**
206 * Frees a descriptor returned by GetStdDescSync().
207 */
208static void free_desc(void *pvDesc)
209{
210 RTMemFree(pvDesc);
211}
212
213/**
214 * Get and a device descriptor and byteswap it appropriately.
215 */
216static bool usbProxyGetDeviceDesc(PUSBPROXYDEV pProxyDev, PVUSBDESCDEVICE pOut)
217{
218 /*
219 * Get the descriptor from the device.
220 */
221 PVUSBDESCDEVICE pIn = (PVUSBDESCDEVICE)GetStdDescSync(pProxyDev, VUSB_DT_DEVICE, 0, 0, VUSB_DT_DEVICE_MIN_LEN);
222 if (!pIn)
223 {
224 Log(("usbProxyGetDeviceDesc: pProxyDev=%s: GetStdDescSync failed\n", pProxyDev->pUsbIns->pszName));
225 return false;
226 }
227 if (pIn->bLength < VUSB_DT_DEVICE_MIN_LEN)
228 {
229 Log(("usb-proxy: pProxyDev=%s: Corrupted device descriptor. bLength=%d\n", pProxyDev->pUsbIns->pszName, pIn->bLength));
230 return false;
231 }
232
233 /*
234 * Convert it.
235 */
236 pOut->bLength = VUSB_DT_DEVICE_MIN_LEN;
237 pOut->bDescriptorType = VUSB_DT_DEVICE;
238 pOut->bcdUSB = RT_LE2H_U16(pIn->bcdUSB);
239 pOut->bDeviceClass = pIn->bDeviceClass;
240 pOut->bDeviceSubClass = pIn->bDeviceSubClass;
241 pOut->bDeviceProtocol = pIn->bDeviceProtocol;
242 pOut->bMaxPacketSize0 = pIn->bMaxPacketSize0;
243 pOut->idVendor = RT_LE2H_U16(pIn->idVendor);
244 pOut->idProduct = RT_LE2H_U16(pIn->idProduct);
245 pOut->bcdDevice = RT_LE2H_U16(pIn->bcdDevice);
246 pOut->iManufacturer = pIn->iManufacturer;
247 pOut->iProduct = pIn->iProduct;
248 pOut->iSerialNumber = pIn->iSerialNumber;
249 pOut->bNumConfigurations = pIn->bNumConfigurations;
250
251 free_desc(pIn);
252 return true;
253}
254
255/**
256 * Count the numbers and types of each kind of descriptor that we need to
257 * copy out of the config descriptor
258 */
259struct desc_counts
260{
261 size_t num_ed, num_id, num_if;
262 /** bitmap (128 bits) */
263 uint32_t idmap[4];
264};
265
266static int count_descriptors(struct desc_counts *cnt, uint8_t *buf, size_t len)
267{
268 PVUSBDESCCONFIG cfg;
269 uint8_t *tmp, *end;
270 uint32_t i, x;
271
272 memset(cnt, 0, sizeof(*cnt));
273
274 end = buf + len;
275
276 cfg = (PVUSBDESCCONFIG)buf;
277 if ( cfg->bLength < VUSB_DT_CONFIG_MIN_LEN )
278 return 0;
279 if ( cfg->bLength > len )
280 return 0;
281
282 for (tmp = buf + cfg->bLength; ((tmp + 1) < end) && *tmp; tmp += *tmp)
283 {
284 uint8_t type;
285 uint32_t ifnum;
286 PVUSBDESCINTERFACE id;
287 PVUSBDESCENDPOINT ed;
288
289 type = *(tmp + 1);
290
291 switch ( type ) {
292 case VUSB_DT_INTERFACE:
293 id = (PVUSBDESCINTERFACE)tmp;
294 if ( id->bLength < VUSB_DT_INTERFACE_MIN_LEN )
295 return 0;
296 cnt->num_id++;
297 ifnum = id->bInterfaceNumber;
298 cnt->idmap[ifnum >> 6] |= (1 << (ifnum & 0x1f));
299 break;
300 case VUSB_DT_ENDPOINT:
301 ed = (PVUSBDESCENDPOINT)tmp;
302 if ( ed->bLength < VUSB_DT_ENDPOINT_MIN_LEN )
303 return 0;
304 cnt->num_ed++;
305 break;
306 default:
307 break;
308 }
309 }
310
311 /* count interfaces */
312 for(i=0; i < RT_ELEMENTS(cnt->idmap); i++)
313 for(x=1; x; x<<=1)
314 if ( cnt->idmap[i] & x )
315 cnt->num_if++;
316
317 return 1;
318}
319
320/* Given the pointer to an interface or endpoint descriptor, find any following
321 * non-standard (vendor or class) descriptors.
322 */
323static const void *collect_stray_bits(uint8_t *this_desc, uint8_t *end, uint16_t *cbExtra)
324{
325 uint8_t *tmp, *buf;
326 uint8_t type;
327
328 Assert(*(this_desc + 1) == VUSB_DT_INTERFACE || *(this_desc + 1) == VUSB_DT_ENDPOINT);
329 buf = this_desc;
330
331 /* Skip the current interface/endpoint descriptor. */
332 buf += *(uint8_t *)buf;
333
334 /* Loop until we find another descriptor we understand. */
335 for (tmp = buf; ((tmp + 1) < end) && *tmp; tmp += *tmp)
336 {
337 type = *(tmp + 1);
338 if (type == VUSB_DT_INTERFACE || type == VUSB_DT_ENDPOINT)
339 break;
340 }
341 *cbExtra = tmp - buf;
342 if (*cbExtra)
343 return buf;
344 else
345 return NULL;
346}
347
348/* Setup a vusb_interface structure given some preallocated structures
349 * to use, (we counted them already)
350 */
351static int copy_interface(PVUSBINTERFACE pIf, uint8_t ifnum,
352 PVUSBDESCINTERFACEEX *id, PVUSBDESCENDPOINTEX *ed,
353 uint8_t *buf, size_t len)
354{
355 PVUSBDESCINTERFACEEX cur_if = NULL;
356 uint32_t altmap[4] = {0,};
357 uint8_t *tmp, *end = buf + len;
358 uint8_t *orig_desc = buf;
359 uint8_t alt;
360 int state;
361 size_t num_ep = 0;
362
363 buf += *(uint8_t *)buf;
364
365 pIf->cSettings = 0;
366 pIf->paSettings = NULL;
367
368 for (tmp = buf, state = 0; ((tmp + 1) < end) && *tmp; tmp += *tmp)
369 {
370 uint8_t type;
371 PVUSBDESCINTERFACE ifd;
372 PVUSBDESCENDPOINT epd;
373 PVUSBDESCENDPOINTEX cur_ep;
374
375 type = tmp[1];
376
377 switch ( type ) {
378 case VUSB_DT_INTERFACE:
379 state = 0;
380 ifd = (PVUSBDESCINTERFACE)tmp;
381
382 /* Ignoring this interface */
383 if ( ifd->bInterfaceNumber != ifnum )
384 break;
385
386 /* Check we didn't see this alternate setting already
387 * because that will break stuff
388 */
389 alt = ifd->bAlternateSetting;
390 if ( altmap[alt >> 6] & (1 << (alt & 0x1f)) )
391 return 0;
392 altmap[alt >> 6] |= (1 << (alt & 0x1f));
393
394 cur_if = *id;
395 (*id)++;
396 if ( pIf->cSettings == 0 )
397 pIf->paSettings = cur_if;
398
399 memcpy(cur_if, ifd, sizeof(cur_if->Core));
400
401 /* Point to additional interface descriptor bytes, if any. */
402 AssertCompile(sizeof(cur_if->Core) == VUSB_DT_INTERFACE_MIN_LEN);
403 if (cur_if->Core.bLength - VUSB_DT_INTERFACE_MIN_LEN > 0)
404 cur_if->pvMore = tmp + VUSB_DT_INTERFACE_MIN_LEN;
405 else
406 cur_if->pvMore = NULL;
407
408 cur_if->pvClass = collect_stray_bits(tmp, end, &cur_if->cbClass);
409
410 pIf->cSettings++;
411
412 state = 1;
413 num_ep = 0;
414 break;
415 case VUSB_DT_ENDPOINT:
416 if ( state == 0 )
417 break;
418
419 epd = (PVUSBDESCENDPOINT)tmp;
420
421 cur_ep = *ed;
422 (*ed)++;
423
424 if ( num_ep == 0 )
425 cur_if->paEndpoints = cur_ep;
426
427 if ( num_ep > cur_if->Core.bNumEndpoints )
428 return 0;
429
430 memcpy(cur_ep, epd, sizeof(cur_ep->Core));
431
432 /* Point to additional endpoint descriptor bytes, if any. */
433 AssertCompile(sizeof(cur_ep->Core) == VUSB_DT_ENDPOINT_MIN_LEN);
434 if (cur_ep->Core.bLength - VUSB_DT_ENDPOINT_MIN_LEN > 0)
435 cur_ep->pvMore = tmp + VUSB_DT_ENDPOINT_MIN_LEN;
436 else
437 cur_ep->pvMore = NULL;
438
439 cur_ep->pvClass = collect_stray_bits(tmp, end, &cur_ep->cbClass);
440
441 cur_ep->Core.wMaxPacketSize = RT_LE2H_U16(cur_ep->Core.wMaxPacketSize);
442
443 num_ep++;
444 break;
445 default:
446 /* Skip unknown descriptors. */
447 break;
448 }
449 }
450
451 return 1;
452}
453
454/**
455 * Copy all of a devices config descriptors, this is needed so that the USB
456 * core layer knows all about how to map the different functions on to the
457 * virtual USB bus.
458 */
459static bool copy_config(PUSBPROXYDEV pProxyDev, uint8_t idx, PVUSBDESCCONFIGEX out)
460{
461 PVUSBDESCCONFIG cfg;
462 PVUSBINTERFACE pIf;
463 PVUSBDESCINTERFACEEX ifd;
464 PVUSBDESCENDPOINTEX epd;
465 struct desc_counts cnt;
466 void *descs;
467 size_t tot_len;
468 size_t cbIface;
469 uint32_t i, x;
470
471 descs = GetStdDescSync(pProxyDev, VUSB_DT_CONFIG, idx, 0, VUSB_DT_CONFIG_MIN_LEN);
472 if ( descs == NULL ) {
473 Log(("copy_config: GetStdDescSync failed\n"));
474 return false;
475 }
476
477 cfg = (PVUSBDESCCONFIG)descs;
478 tot_len = RT_LE2H_U16(cfg->wTotalLength);
479
480 if ( !count_descriptors(&cnt, (uint8_t *)descs, tot_len) ) {
481 Log(("copy_config: count_descriptors failed\n"));
482 goto err;
483 }
484
485 if ( cfg->bNumInterfaces != cnt.num_if )
486 Log(("usb-proxy: config%u: bNumInterfaces %u != %u\n",
487 idx, cfg->bNumInterfaces, cnt.num_if));
488
489 Log(("usb-proxy: config%u: %u bytes id=%u ed=%u if=%u\n",
490 idx, tot_len, cnt.num_id, cnt.num_ed, cnt.num_if));
491
492 cbIface = cnt.num_if * sizeof(VUSBINTERFACE)
493 + cnt.num_id * sizeof(VUSBDESCINTERFACEEX)
494 + cnt.num_ed * sizeof(VUSBDESCENDPOINTEX);
495 out->paIfs = (PCVUSBINTERFACE)RTMemAllocZ(cbIface);
496 if ( out->paIfs == NULL ) {
497 free_desc(descs);
498 return false;
499 }
500
501 /* Stash a pointer to the raw config descriptor; we may need bits of it later. */
502 out->pvOriginal = descs;
503
504 pIf = (PVUSBINTERFACE)out->paIfs;
505 ifd = (PVUSBDESCINTERFACEEX)&pIf[cnt.num_if];
506 epd = (PVUSBDESCENDPOINTEX)&ifd[cnt.num_id];
507
508 out->Core.bLength = cfg->bLength;
509 out->Core.bDescriptorType = cfg->bDescriptorType;
510 out->Core.wTotalLength = 0; /* Auto Calculated */
511 out->Core.bNumInterfaces = (uint8_t)cnt.num_if;
512 out->Core.bConfigurationValue = cfg->bConfigurationValue;
513 out->Core.iConfiguration = cfg->iConfiguration;
514 out->Core.bmAttributes = cfg->bmAttributes;
515 out->Core.MaxPower = cfg->MaxPower;
516
517 for(i=0; i < 4; i++)
518 for(x=0; x < 32; x++)
519 if ( cnt.idmap[i] & (1 << x) )
520 if ( !copy_interface(pIf++, (i << 6) | x, &ifd, &epd, (uint8_t *)out->pvOriginal, tot_len) ) {
521 Log(("copy_interface(%d,,) failed\n", pIf - 1));
522 goto err;
523 }
524
525 return true;
526err:
527 Log(("usb-proxy: config%u: Corrupted configuration descriptor\n", idx));
528 free_desc(descs);
529 return false;
530}
531
532
533/**
534 * Edit out masked interface descriptors.
535 *
536 * @param pProxyDev The proxy device
537 */
538static void usbProxyDevEditOutMaskedIfs(PUSBPROXYDEV pProxyDev)
539{
540 unsigned cRemoved = 0;
541
542 PVUSBDESCCONFIGEX paCfgs = pProxyDev->paCfgDescs;
543 for (unsigned iCfg = 0; iCfg < pProxyDev->DevDesc.bNumConfigurations; iCfg++)
544 {
545 PVUSBINTERFACE paIfs = (PVUSBINTERFACE)paCfgs[iCfg].paIfs;
546 for (unsigned iIf = 0; iIf < paCfgs[iCfg].Core.bNumInterfaces; iIf++)
547 for (uint32_t iAlt = 0; iAlt < paIfs[iIf].cSettings; iAlt++)
548 if ( paIfs[iIf].paSettings[iAlt].Core.bInterfaceNumber < 32
549 && ((1 << paIfs[iIf].paSettings[iAlt].Core.bInterfaceNumber) & pProxyDev->fMaskedIfs))
550 {
551 Log(("usb-proxy: removing interface #%d (iIf=%d iAlt=%d) on config #%d (iCfg=%d)\n",
552 paIfs[iIf].paSettings[iAlt].Core.bInterfaceNumber, iIf, iAlt, paCfgs[iCfg].Core.bConfigurationValue, iCfg));
553 cRemoved++;
554
555 paCfgs[iCfg].Core.bNumInterfaces--;
556 unsigned cToCopy = paCfgs[iCfg].Core.bNumInterfaces - iIf;
557 if (cToCopy)
558 memmove(&paIfs[iIf], &paIfs[iIf + 1], sizeof(paIfs[0]) * cToCopy);
559 memset(&paIfs[iIf + cToCopy], '\0', sizeof(paIfs[0]));
560 break;
561 }
562 }
563
564 Log(("usb-proxy: edited out %d interface(s).\n", cRemoved));
565}
566
567
568/**
569 * @copydoc PDMUSBREG::pfnUsbReset
570 *
571 * USB Device Proxy: Call OS specific code to reset the device.
572 */
573static DECLCALLBACK(int) usbProxyDevReset(PPDMUSBINS pUsbIns, bool fResetOnLinux)
574{
575 PUSBPROXYDEV pProxyDev = PDMINS_2_DATA(pUsbIns, PUSBPROXYDEV);
576
577 if (pProxyDev->fMaskedIfs)
578 {
579 Log(("usbProxyDevReset: pProxyDev=%s - ignoring reset request fMaskedIfs=%#x\n", pUsbIns->pszName, pProxyDev->fMaskedIfs));
580 return VINF_SUCCESS;
581 }
582 LogFlow(("usbProxyDevReset: pProxyDev=%s\n", pUsbIns->pszName));
583 return pProxyDev->pOps->pfnReset(pProxyDev, fResetOnLinux);
584}
585
586
587/**
588 * @copydoc PDMUSBREG::pfnUsbGetDescriptorCache
589 */
590static DECLCALLBACK(PCPDMUSBDESCCACHE) usbProxyDevGetDescriptorCache(PPDMUSBINS pUsbIns)
591{
592 PUSBPROXYDEV pThis = PDMINS_2_DATA(pUsbIns, PUSBPROXYDEV);
593 return &pThis->DescCache;
594}
595
596
597/**
598 * @copydoc PDMUSBREG::pfnUsbSetConfiguration
599 *
600 * USB Device Proxy: Release claimed interfaces, tell the OS+device about the config change, claim the new interfaces.
601 */
602static DECLCALLBACK(int) usbProxyDevSetConfiguration(PPDMUSBINS pUsbIns, uint8_t bConfigurationValue,
603 const void *pvOldCfgDesc, const void *pvOldIfState, const void *pvNewCfgDesc)
604{
605 PUSBPROXYDEV pProxyDev = PDMINS_2_DATA(pUsbIns, PUSBPROXYDEV);
606 LogFlow(("usbProxyDevSetConfiguration: pProxyDev=%s iActiveCfg=%d bConfigurationValue=%d\n",
607 pUsbIns->pszName, pProxyDev->iActiveCfg, bConfigurationValue));
608
609 /*
610 * Release the current config.
611 */
612 if (pvOldCfgDesc)
613 {
614 PCVUSBDESCCONFIGEX pOldCfgDesc = (PCVUSBDESCCONFIGEX)pvOldCfgDesc;
615 PCVUSBINTERFACESTATE pOldIfState = (PCVUSBINTERFACESTATE)pvOldIfState;
616 for (unsigned i = 0; i < pOldCfgDesc->Core.bNumInterfaces; i++)
617 if (pOldIfState[i].pCurIfDesc)
618 pProxyDev->pOps->pfnReleaseInterface(pProxyDev, pOldIfState[i].pCurIfDesc->Core.bInterfaceNumber);
619 }
620
621 /*
622 * Do the actual SET_CONFIGURE.
623 * The mess here is because most backends will already have selected a
624 * configuration and there are a bunch of devices which will freak out
625 * if we do SET_CONFIGURE twice with the same value. (PalmOne, TrekStor USB-StickGO, ..)
626 *
627 * After open and reset the backend should use the members iActiveCfg and cIgnoreSetConfigs
628 * to indicate the new configuration state and what to do on the next SET_CONFIGURATION call.
629 */
630 if ( pProxyDev->iActiveCfg != bConfigurationValue
631 || ( bConfigurationValue == 0
632 && pProxyDev->iActiveCfg != -1 /* this test doesn't make sense, we know it's 0 */
633 && pProxyDev->cIgnoreSetConfigs >= 2)
634 || !pProxyDev->cIgnoreSetConfigs)
635 {
636 pProxyDev->cIgnoreSetConfigs = 0;
637 int rc = pProxyDev->pOps->pfnSetConfig(pProxyDev, bConfigurationValue);
638 if (RT_FAILURE(rc))
639 {
640 pProxyDev->iActiveCfg = -1;
641 return rc;
642 }
643 pProxyDev->iActiveCfg = bConfigurationValue;
644 }
645 else if (pProxyDev->cIgnoreSetConfigs > 0)
646 pProxyDev->cIgnoreSetConfigs--;
647
648 /*
649 * Claim the interfaces.
650 */
651 PCVUSBDESCCONFIGEX pNewCfgDesc = (PCVUSBDESCCONFIGEX)pvNewCfgDesc;
652 Assert(pNewCfgDesc->Core.bConfigurationValue == bConfigurationValue);
653 for (unsigned iIf = 0; iIf < pNewCfgDesc->Core.bNumInterfaces; iIf++)
654 {
655 PCVUSBINTERFACE pIf = &pNewCfgDesc->paIfs[iIf];
656 for (uint32_t iAlt = 0; iAlt < pIf->cSettings; iAlt++)
657 {
658 if (pIf->paSettings[iAlt].Core.bAlternateSetting != 0)
659 continue;
660 pProxyDev->pOps->pfnClaimInterface(pProxyDev, pIf->paSettings[iAlt].Core.bInterfaceNumber);
661 /* ignore failures - the backend deals with that and does the necessary logging. */
662 break;
663 }
664 }
665
666 return VINF_SUCCESS;
667}
668
669
670/**
671 * @copydoc PDMUSBREG::pfnUsbSetInterface
672 *
673 * USB Device Proxy: Call OS specific code to select alternate interface settings.
674 */
675static DECLCALLBACK(int) usbProxyDevSetInterface(PPDMUSBINS pUsbIns, uint8_t bInterfaceNumber, uint8_t bAlternateSetting)
676{
677 PUSBPROXYDEV pProxyDev = PDMINS_2_DATA(pUsbIns, PUSBPROXYDEV);
678 LogFlow(("usbProxyDevSetInterface: pProxyDev=%s bInterfaceNumber=%d bAlternateSetting=%d\n",
679 pUsbIns->pszName, bInterfaceNumber, bAlternateSetting));
680
681 return pProxyDev->pOps->pfnSetInterface(pProxyDev, bInterfaceNumber, bAlternateSetting);
682}
683
684
685/**
686 * @copydoc PDMUSBREG::pfnUsbClearHaltedEndpoint
687 *
688 * USB Device Proxy: Call OS specific code to clear the endpoint.
689 */
690static DECLCALLBACK(int) usbProxyDevClearHaltedEndpoint(PPDMUSBINS pUsbIns, unsigned uEndpoint)
691{
692 PUSBPROXYDEV pProxyDev = PDMINS_2_DATA(pUsbIns, PUSBPROXYDEV);
693 LogFlow(("usbProxyDevClearHaltedEndpoint: pProxyDev=%s uEndpoint=%u\n",
694 pUsbIns->pszName, uEndpoint));
695
696 return pProxyDev->pOps->pfnClearHaltedEndpoint(pProxyDev, uEndpoint);
697}
698
699
700/**
701 * @copydoc PDMUSBREG::pfnUrbQueue
702 *
703 * USB Device Proxy: Call OS specific code.
704 */
705static DECLCALLBACK(int) usbProxyDevUrbQueue(PPDMUSBINS pUsbIns, PVUSBURB pUrb)
706{
707 int rc = VINF_SUCCESS;
708 PUSBPROXYDEV pProxyDev = PDMINS_2_DATA(pUsbIns, PUSBPROXYDEV);
709 rc = pProxyDev->pOps->pfnUrbQueue(pProxyDev, pUrb);
710 if (RT_FAILURE(rc))
711 return pProxyDev->fDetached
712 ? VERR_VUSB_DEVICE_NOT_ATTACHED
713 : VERR_VUSB_FAILED_TO_QUEUE_URB;
714 return rc;
715}
716
717
718/**
719 * @copydoc PDMUSBREG::pfnUrbCancel
720 *
721 * USB Device Proxy: Call OS specific code.
722 */
723static DECLCALLBACK(int) usbProxyDevUrbCancel(PPDMUSBINS pUsbIns, PVUSBURB pUrb)
724{
725 PUSBPROXYDEV pProxyDev = PDMINS_2_DATA(pUsbIns, PUSBPROXYDEV);
726 return pProxyDev->pOps->pfnUrbCancel(pProxyDev, pUrb);
727}
728
729
730/**
731 * @copydoc PDMUSBREG::pfnUrbReap
732 *
733 * USB Device Proxy: Call OS specific code.
734 */
735static DECLCALLBACK(PVUSBURB) usbProxyDevUrbReap(PPDMUSBINS pUsbIns, RTMSINTERVAL cMillies)
736{
737 PUSBPROXYDEV pProxyDev = PDMINS_2_DATA(pUsbIns, PUSBPROXYDEV);
738 PVUSBURB pUrb = pProxyDev->pOps->pfnUrbReap(pProxyDev, cMillies);
739 if ( pUrb
740 && pUrb->enmState == VUSBURBSTATE_CANCELLED
741 && pUrb->enmStatus == VUSBSTATUS_OK)
742 pUrb->enmStatus = VUSBSTATUS_DNR;
743 return pUrb;
744}
745
746
747/**
748 * @copydoc PDMUSBREG::pfnWakeup
749 *
750 * USB Device Proxy: Call OS specific code.
751 */
752static DECLCALLBACK(int) usbProxyDevWakeup(PPDMUSBINS pUsbIns)
753{
754 PUSBPROXYDEV pProxyDev = PDMINS_2_DATA(pUsbIns, PUSBPROXYDEV);
755
756 return pProxyDev->pOps->pfnWakeup(pProxyDev);
757}
758
759
760/** @copydoc PDMUSBREG::pfnDestruct */
761static DECLCALLBACK(void) usbProxyDestruct(PPDMUSBINS pUsbIns)
762{
763 PUSBPROXYDEV pThis = PDMINS_2_DATA(pUsbIns, PUSBPROXYDEV);
764 Log(("usbProxyDestruct: destroying pProxyDev=%s\n", pUsbIns->pszName));
765
766 /* close it. */
767 if (pThis->fOpened)
768 {
769 pThis->pOps->pfnClose(pThis);
770 pThis->fOpened = false;
771 }
772
773 /* free the config descriptors. */
774 if (pThis->paCfgDescs)
775 {
776 for (unsigned i = 0; i < pThis->DevDesc.bNumConfigurations; i++)
777 {
778 RTMemFree((void *)pThis->paCfgDescs[i].paIfs);
779 RTMemFree((void *)pThis->paCfgDescs[i].pvOriginal);
780 }
781 RTMemFree(pThis->paCfgDescs);
782 pThis->paCfgDescs = NULL;
783 }
784
785 /* free dev */
786 if (&g_szDummyName[0] != pUsbIns->pszName)
787 RTStrFree(pUsbIns->pszName);
788 pUsbIns->pszName = NULL;
789
790 if (pThis->pvInstanceDataR3)
791 RTMemFree(pThis->pvInstanceDataR3);
792}
793
794
795/**
796 * Helper function used by usbProxyConstruct when
797 * reading a filter from CFG.
798 *
799 * @returns VBox status code.
800 * @param pFilter The filter.
801 * @param enmFieldIdx The filter field indext.
802 * @param pNode The CFGM node.
803 * @param pszExact The exact value name.
804 * @param pszExpr The expression value name.
805 */
806static int usbProxyQueryNum(PUSBFILTER pFilter, USBFILTERIDX enmFieldIdx, PCFGMNODE pNode, const char *pszExact, const char *pszExpr)
807{
808 char szTmp[256];
809
810 /* try exact first */
811 uint16_t u16;
812 int rc = CFGMR3QueryU16(pNode, pszExact, &u16);
813 if (RT_SUCCESS(rc))
814 {
815 rc = USBFilterSetNumExact(pFilter, enmFieldIdx, u16, true);
816 AssertRCReturn(rc, rc);
817
818 /* make sure only the exact attribute is present. */
819 rc = CFGMR3QueryString(pNode, pszExpr, szTmp, sizeof(szTmp));
820 if (RT_UNLIKELY(rc != VERR_CFGM_VALUE_NOT_FOUND))
821 {
822 szTmp[0] = '\0';
823 CFGMR3GetName(pNode, szTmp, sizeof(szTmp));
824 LogRel(("usbProxyConstruct: %s: Both %s and %s are present!\n", szTmp, pszExact, pszExpr));
825 return VERR_INVALID_PARAMETER;
826 }
827 return VINF_SUCCESS;
828 }
829 if (RT_UNLIKELY(rc != VERR_CFGM_VALUE_NOT_FOUND))
830 {
831 szTmp[0] = '\0';
832 CFGMR3GetName(pNode, szTmp, sizeof(szTmp));
833 LogRel(("usbProxyConstruct: %s: %s query failed, rc=%Rrc\n", szTmp, pszExact, rc));
834 return rc;
835 }
836
837 /* expression? */
838 rc = CFGMR3QueryString(pNode, pszExpr, szTmp, sizeof(szTmp));
839 if (RT_SUCCESS(rc))
840 {
841 rc = USBFilterSetNumExpression(pFilter, enmFieldIdx, szTmp, true);
842 AssertRCReturn(rc, rc);
843 return VINF_SUCCESS;
844 }
845 if (RT_UNLIKELY(rc != VERR_CFGM_VALUE_NOT_FOUND))
846 {
847 szTmp[0] = '\0';
848 CFGMR3GetName(pNode, szTmp, sizeof(szTmp));
849 LogRel(("usbProxyConstruct: %s: %s query failed, rc=%Rrc\n", szTmp, pszExpr, rc));
850 return rc;
851 }
852
853 return VINF_SUCCESS;
854}
855
856
857/** @copydoc PDMUSBREG::pfnConstruct */
858static DECLCALLBACK(int) usbProxyConstruct(PPDMUSBINS pUsbIns, int iInstance, PCFGMNODE pCfg, PCFGMNODE pCfgGlobal)
859{
860 PUSBPROXYDEV pThis = PDMINS_2_DATA(pUsbIns, PUSBPROXYDEV);
861 LogFlow(("usbProxyConstruct: pUsbIns=%p iInstance=%d\n", pUsbIns, iInstance));
862
863 /*
864 * Initialize the instance data.
865 */
866 pThis->pUsbIns = pUsbIns;
867 pThis->pUsbIns->pszName = g_szDummyName;
868 pThis->iActiveCfg = -1;
869 pThis->fMaskedIfs = 0;
870 pThis->fOpened = false;
871 pThis->fInited = false;
872
873 /*
874 * Read the basic configuration.
875 */
876 char szAddress[1024];
877 int rc = CFGMR3QueryString(pCfg, "Address", szAddress, sizeof(szAddress));
878 AssertRCReturn(rc, rc);
879
880 char szBackend[64];
881 rc = CFGMR3QueryString(pCfg, "Backend", szBackend, sizeof(szBackend));
882 AssertRCReturn(rc, rc);
883
884 void *pvBackend;
885 rc = CFGMR3QueryPtr(pCfg, "pvBackend", &pvBackend);
886 AssertRCReturn(rc, rc);
887
888 /*
889 * Select backend and open the device.
890 */
891 rc = VERR_NOT_FOUND;
892 for (unsigned i = 0; i < RT_ELEMENTS(g_aUsbProxies); i++)
893 {
894 if (!RTStrICmp(szBackend, g_aUsbProxies[i]->pszName))
895 {
896 pThis->pOps = g_aUsbProxies[i];
897 rc = VINF_SUCCESS;
898 break;
899 }
900 }
901 if (RT_FAILURE(rc))
902 return PDMUSB_SET_ERROR(pUsbIns, rc, N_("USBProxy: Failed to find backend"));
903
904 pThis->pvInstanceDataR3 = RTMemAllocZ(pThis->pOps->cbBackend);
905 if (!pThis->pvInstanceDataR3)
906 return PDMUSB_SET_ERROR(pUsbIns, VERR_NO_MEMORY, N_("USBProxy: can't allocate memory for host backend"));
907
908 rc = pThis->pOps->pfnOpen(pThis, szAddress, pvBackend);
909 if (RT_FAILURE(rc))
910 {
911 LogRel(("usbProxyConstruct: Failed to open '%s', rc=%Rrc\n", szAddress, rc));
912 return rc;
913 }
914 pThis->fOpened = true;
915
916 /*
917 * Get the device descriptor and format the device name (for logging).
918 */
919 if (!usbProxyGetDeviceDesc(pThis, &pThis->DevDesc))
920 {
921 Log(("usbProxyConstruct: usbProxyGetDeviceDesc failed\n"));
922 return VERR_READ_ERROR;
923 }
924
925 RTStrAPrintf(&pUsbIns->pszName, "%p[proxy %04x:%04x]", pThis, pThis->DevDesc.idVendor, pThis->DevDesc.idProduct); /** @todo append the user comment */
926 AssertReturn(pUsbIns->pszName, VERR_NO_MEMORY);
927
928 /*
929 * Get config descriptors.
930 */
931 size_t cbConfigs = pThis->DevDesc.bNumConfigurations * sizeof(pThis->paCfgDescs[0]);
932 pThis->paCfgDescs = (PVUSBDESCCONFIGEX)RTMemAllocZ(cbConfigs);
933 AssertReturn(pThis->paCfgDescs, VERR_NO_MEMORY);
934
935 unsigned i;
936 for (i = 0; i < pThis->DevDesc.bNumConfigurations; i++)
937 if (!copy_config(pThis, i, (PVUSBDESCCONFIGEX)&pThis->paCfgDescs[i]))
938 break;
939 if (i < pThis->DevDesc.bNumConfigurations)
940 {
941 Log(("usbProxyConstruct: copy_config failed, i=%d\n", i));
942 return VERR_READ_ERROR;
943 }
944
945 /*
946 * Pickup best matching global configuration for this device.
947 * The global configuration is organized like this:
948 *
949 * GlobalConfig/Whatever/
950 * |- idVendor = 300
951 * |- idProduct = 300
952 * - Config/
953 *
954 * The first level contains filter attributes which we stuff into a USBFILTER
955 * structure and match against the device info that's available. The highest
956 * ranked match is will be used. If nothing is found, the values will be
957 * queried from the GlobalConfig node (simplifies code and might actually
958 * be useful).
959 */
960 PCFGMNODE pCfgGlobalDev = pCfgGlobal;
961 PCFGMNODE pCur = CFGMR3GetFirstChild(pCfgGlobal);
962 if (pCur)
963 {
964 /*
965 * Create a device filter from the device configuration
966 * descriptor ++. No strings currently.
967 */
968 USBFILTER Device;
969 USBFilterInit(&Device, USBFILTERTYPE_CAPTURE);
970 rc = USBFilterSetNumExact(&Device, USBFILTERIDX_VENDOR_ID, pThis->DevDesc.idVendor, true); AssertRC(rc);
971 rc = USBFilterSetNumExact(&Device, USBFILTERIDX_PRODUCT_ID, pThis->DevDesc.idProduct, true); AssertRC(rc);
972 rc = USBFilterSetNumExact(&Device, USBFILTERIDX_DEVICE_REV, pThis->DevDesc.bcdDevice, true); AssertRC(rc);
973 rc = USBFilterSetNumExact(&Device, USBFILTERIDX_DEVICE_CLASS, pThis->DevDesc.bDeviceClass, true); AssertRC(rc);
974 rc = USBFilterSetNumExact(&Device, USBFILTERIDX_DEVICE_SUB_CLASS, pThis->DevDesc.bDeviceSubClass, true); AssertRC(rc);
975 rc = USBFilterSetNumExact(&Device, USBFILTERIDX_DEVICE_PROTOCOL, pThis->DevDesc.bDeviceProtocol, true); AssertRC(rc);
976 /** @todo manufacturer, product and serial strings */
977
978 int iBestMatchRate = -1;
979 PCFGMNODE pBestMatch = NULL;
980 for (pCur = CFGMR3GetFirstChild(pCfgGlobal); pCur; pCur = CFGMR3GetNextChild(pCur))
981 {
982 /*
983 * Construct a filter from the attributes in the node.
984 */
985 USBFILTER Filter;
986 USBFilterInit(&Filter, USBFILTERTYPE_CAPTURE);
987
988 /* numeric */
989 if ( RT_FAILURE(usbProxyQueryNum(&Filter, USBFILTERIDX_VENDOR_ID, pCur, "idVendor", "idVendorExpr"))
990 || RT_FAILURE(usbProxyQueryNum(&Filter, USBFILTERIDX_PRODUCT_ID, pCur, "idProduct", "idProcutExpr"))
991 || RT_FAILURE(usbProxyQueryNum(&Filter, USBFILTERIDX_DEVICE_REV, pCur, "bcdDevice", "bcdDeviceExpr"))
992 || RT_FAILURE(usbProxyQueryNum(&Filter, USBFILTERIDX_DEVICE_CLASS, pCur, "bDeviceClass", "bDeviceClassExpr"))
993 || RT_FAILURE(usbProxyQueryNum(&Filter, USBFILTERIDX_DEVICE_SUB_CLASS, pCur, "bDeviceSubClass", "bDeviceSubClassExpr"))
994 || RT_FAILURE(usbProxyQueryNum(&Filter, USBFILTERIDX_DEVICE_PROTOCOL, pCur, "bDeviceProtocol", "bDeviceProtocolExpr")))
995 continue; /* skip it */
996
997 /* strings */
998 /** @todo manufacturer, product and serial strings */
999
1000 /* ignore unknown config values, but not without bitching. */
1001 if (!CFGMR3AreValuesValid(pCur,
1002 "idVendor\0idVendorExpr\0"
1003 "idProduct\0idProductExpr\0"
1004 "bcdDevice\0bcdDeviceExpr\0"
1005 "bDeviceClass\0bDeviceClassExpr\0"
1006 "bDeviceSubClass\0bDeviceSubClassExpr\0"
1007 "bDeviceProtocol\0bDeviceProtocolExpr\0"))
1008 LogRel(("usbProxyConstruct: Unknown value(s) in config filter (ignored)!\n"));
1009
1010 /*
1011 * Try match it and on match see if it has is a higher rate hit
1012 * than the previous match. Quit if its a 100% match.
1013 */
1014 int iRate = USBFilterMatchRated(&Filter, &Device);
1015 if (iRate > iBestMatchRate)
1016 {
1017 pBestMatch = pCur;
1018 iBestMatchRate = iRate;
1019 if (iRate >= 100)
1020 break;
1021 }
1022 }
1023 if (pBestMatch)
1024 pCfgGlobalDev = CFGMR3GetChild(pBestMatch, "Config");
1025 if (pCfgGlobalDev)
1026 pCfgGlobalDev = pCfgGlobal;
1027 }
1028
1029 /*
1030 * Query the rest of the configuration using the global as fallback.
1031 */
1032 rc = CFGMR3QueryU32(pCfg, "MaskedIfs", &pThis->fMaskedIfs);
1033 if (rc == VERR_CFGM_VALUE_NOT_FOUND)
1034 rc = CFGMR3QueryU32(pCfgGlobalDev, "MaskedIfs", &pThis->fMaskedIfs);
1035 if (rc == VERR_CFGM_VALUE_NOT_FOUND)
1036 pThis->fMaskedIfs = 0;
1037 else
1038 AssertRCReturn(rc, rc);
1039
1040 bool fForce11Device;
1041 rc = CFGMR3QueryBool(pCfg, "Force11Device", &fForce11Device);
1042 if (rc == VERR_CFGM_VALUE_NOT_FOUND)
1043 rc = CFGMR3QueryBool(pCfgGlobalDev, "Force11Device", &fForce11Device);
1044 if (rc == VERR_CFGM_VALUE_NOT_FOUND)
1045 fForce11Device = false;
1046 else
1047 AssertRCReturn(rc, rc);
1048
1049 bool fForce11PacketSize;
1050 rc = CFGMR3QueryBool(pCfg, "Force11PacketSize", &fForce11PacketSize);
1051 if (rc == VERR_CFGM_VALUE_NOT_FOUND)
1052 rc = CFGMR3QueryBool(pCfgGlobalDev, "Force11PacketSize", &fForce11PacketSize);
1053 if (rc == VERR_CFGM_VALUE_NOT_FOUND)
1054 fForce11PacketSize = false;
1055 else
1056 AssertRCReturn(rc, rc);
1057
1058 /*
1059 * If we're masking interfaces, edit the descriptors.
1060 */
1061 bool fEdited = pThis->fMaskedIfs != 0;
1062 if (pThis->fMaskedIfs)
1063 usbProxyDevEditOutMaskedIfs(pThis);
1064
1065 /*
1066 * Do 2.0 -> 1.1 device edits if requested to do so.
1067 */
1068 if ( fForce11PacketSize
1069 && pThis->DevDesc.bcdUSB >= 0x0200)
1070 {
1071 PVUSBDESCCONFIGEX paCfgs = pThis->paCfgDescs;
1072 for (unsigned iCfg = 0; iCfg < pThis->DevDesc.bNumConfigurations; iCfg++)
1073 {
1074 PVUSBINTERFACE paIfs = (PVUSBINTERFACE)paCfgs[iCfg].paIfs;
1075 for (unsigned iIf = 0; iIf < paCfgs[iCfg].Core.bNumInterfaces; iIf++)
1076 for (uint32_t iAlt = 0; iAlt < paIfs[iIf].cSettings; iAlt++)
1077 {
1078 /*
1079 * USB 1.1 defines the max for control, interrupt and bulk to be 64 bytes.
1080 * While isochronous has a max of 1023 bytes.
1081 */
1082 PVUSBDESCENDPOINTEX paEps = (PVUSBDESCENDPOINTEX)paIfs[iIf].paSettings[iAlt].paEndpoints;
1083 for (unsigned iEp = 0; iEp < paIfs[iIf].paSettings[iAlt].Core.bNumEndpoints; iEp++)
1084 {
1085 const uint16_t cbMax = (paEps[iEp].Core.bmAttributes & 3) == 1 /* isoc */
1086 ? 1023
1087 : 64;
1088 if (paEps[iEp].Core.wMaxPacketSize > cbMax)
1089 {
1090 Log(("usb-proxy: pProxyDev=%s correcting wMaxPacketSize from %#x to %#x (mainly for vista)\n",
1091 pUsbIns->pszName, paEps[iEp].Core.wMaxPacketSize, cbMax));
1092 paEps[iEp].Core.wMaxPacketSize = cbMax;
1093 fEdited = true;
1094 }
1095 }
1096 }
1097 }
1098 }
1099
1100 if ( fForce11Device
1101 && pThis->DevDesc.bcdUSB == 0x0200)
1102 {
1103 /*
1104 * Discourages windows from helping you find a 2.0 port.
1105 */
1106 Log(("usb-proxy: %s correcting USB version 2.0 to 1.1 (to avoid Windows warning)\n", pUsbIns->pszName));
1107 pThis->DevDesc.bcdUSB = 0x110;
1108 fEdited = true;
1109 }
1110
1111
1112 /*
1113 * Init the PDM/VUSB descriptor cache.
1114 */
1115 pThis->DescCache.pDevice = &pThis->DevDesc;
1116 pThis->DescCache.paConfigs = pThis->paCfgDescs;
1117 pThis->DescCache.paLanguages = NULL;
1118 pThis->DescCache.cLanguages = 0;
1119 pThis->DescCache.fUseCachedDescriptors = fEdited;
1120 pThis->DescCache.fUseCachedStringsDescriptors = false;
1121
1122 /*
1123 * Call the backend if it wishes to do some more initializing
1124 * after we've read the config and descriptors.
1125 */
1126 if (pThis->pOps->pfnInit)
1127 {
1128 rc = pThis->pOps->pfnInit(pThis);
1129 if (RT_FAILURE(rc))
1130 return rc;
1131 }
1132 pThis->fInited = true;
1133
1134 /*
1135 * We're good!
1136 */
1137 Log(("usb-proxy: created pProxyDev=%s address '%s' fMaskedIfs=%#x (rc=%Rrc)\n",
1138 pUsbIns->pszName, szAddress, pThis->fMaskedIfs, rc));
1139 return VINF_SUCCESS;
1140}
1141
1142
1143/**
1144 * The USB proxy device registration record.
1145 */
1146const PDMUSBREG g_UsbDevProxy =
1147{
1148 /* u32Version */
1149 PDM_USBREG_VERSION,
1150 /* szName */
1151 "USBProxy",
1152 /* pszDescription */
1153 "USB Proxy Device.",
1154 /* fFlags */
1155 0,
1156 /* cMaxInstances */
1157 ~0U,
1158 /* cbInstance */
1159 sizeof(USBPROXYDEV),
1160 /* pfnConstruct */
1161 usbProxyConstruct,
1162 /* pfnDestruct */
1163 usbProxyDestruct,
1164 /* pfnVMInitComplete */
1165 NULL,
1166 /* pfnVMPowerOn */
1167 NULL,
1168 /* pfnVMReset */
1169 NULL,
1170 /* pfnVMSuspend */
1171 NULL,
1172 /* pfnVMResume */
1173 NULL,
1174 /* pfnVMPowerOff */
1175 NULL,
1176 /* pfnHotPlugged */
1177 NULL,
1178 /* pfnHotUnplugged */
1179 NULL,
1180 /* pfnDriverAttach */
1181 NULL,
1182 /* pfnDriverDetach */
1183 NULL,
1184 /* pfnQueryInterface */
1185 NULL,
1186 /* pfnUsbReset */
1187 usbProxyDevReset,
1188 /* pfnUsbGetDescriptorCache */
1189 usbProxyDevGetDescriptorCache,
1190 /* pfnUsbSetConfiguration */
1191 usbProxyDevSetConfiguration,
1192 /* pfnUsbSetInterface */
1193 usbProxyDevSetInterface,
1194 /* pfnUsbClearHaltedEndpoint */
1195 usbProxyDevClearHaltedEndpoint,
1196 /* pfnUrbNew */
1197 NULL,
1198 /* pfnUrbQueue */
1199 usbProxyDevUrbQueue,
1200 /* pfnUrbCancel */
1201 usbProxyDevUrbCancel,
1202 /* pfnUrbReap */
1203 usbProxyDevUrbReap,
1204 /* pfnWakeup */
1205 usbProxyDevWakeup,
1206
1207 /* u32TheEnd */
1208 PDM_USBREG_VERSION
1209};
1210
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