VirtualBox

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

Last change on this file since 62513 was 62502, checked in by vboxsync, 9 years ago

(C) 2016

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