VirtualBox

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

Last change on this file since 65919 was 62992, checked in by vboxsync, 8 years ago

Devices: warnings

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 40.1 KB
Line 
1/* $Id: USBProxyDevice.cpp 62992 2016-08-04 15:08:21Z 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 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 * @interface_method_impl{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 * @interface_method_impl{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 * @interface_method_impl{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 * @interface_method_impl{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 * @interface_method_impl{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 * @interface_method_impl{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 * @interface_method_impl{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 * @interface_method_impl{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 * @interface_method_impl{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/** @interface_method_impl{PDMUSBREG,pfnDestruct} */
761static DECLCALLBACK(void) usbProxyDestruct(PPDMUSBINS pUsbIns)
762{
763 PDMUSB_CHECK_VERSIONS_RETURN_VOID(pUsbIns);
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/** @interface_method_impl{PDMUSBREG,pfnConstruct} */
859static DECLCALLBACK(int) usbProxyConstruct(PPDMUSBINS pUsbIns, int iInstance, PCFGMNODE pCfg, PCFGMNODE pCfgGlobal)
860{
861 PDMUSB_CHECK_VERSIONS_RETURN(pUsbIns);
862 RT_NOREF(iInstance);
863 PUSBPROXYDEV pThis = PDMINS_2_DATA(pUsbIns, PUSBPROXYDEV);
864 LogFlow(("usbProxyConstruct: pUsbIns=%p iInstance=%d\n", pUsbIns, iInstance));
865
866 /*
867 * Initialize the instance data.
868 */
869 pThis->pUsbIns = pUsbIns;
870 pThis->pUsbIns->pszName = g_szDummyName;
871 pThis->iActiveCfg = -1;
872 pThis->fMaskedIfs = 0;
873 pThis->fOpened = false;
874 pThis->fInited = false;
875
876 /*
877 * Read the basic configuration.
878 */
879 char szAddress[1024];
880 int rc = CFGMR3QueryString(pCfg, "Address", szAddress, sizeof(szAddress));
881 AssertRCReturn(rc, rc);
882
883 char szBackend[64];
884 rc = CFGMR3QueryString(pCfg, "Backend", szBackend, sizeof(szBackend));
885 AssertRCReturn(rc, rc);
886
887 void *pvBackend;
888 rc = CFGMR3QueryPtr(pCfg, "pvBackend", &pvBackend);
889 AssertRCReturn(rc, rc);
890
891 /*
892 * Select backend and open the device.
893 */
894 rc = VERR_NOT_FOUND;
895 for (unsigned i = 0; i < RT_ELEMENTS(g_aUsbProxies); i++)
896 {
897 if (!RTStrICmp(szBackend, g_aUsbProxies[i]->pszName))
898 {
899 pThis->pOps = g_aUsbProxies[i];
900 rc = VINF_SUCCESS;
901 break;
902 }
903 }
904 if (RT_FAILURE(rc))
905 return PDMUSB_SET_ERROR(pUsbIns, rc, N_("USBProxy: Failed to find backend"));
906
907 pThis->pvInstanceDataR3 = RTMemAllocZ(pThis->pOps->cbBackend);
908 if (!pThis->pvInstanceDataR3)
909 return PDMUSB_SET_ERROR(pUsbIns, VERR_NO_MEMORY, N_("USBProxy: can't allocate memory for host backend"));
910
911 rc = pThis->pOps->pfnOpen(pThis, szAddress, pvBackend);
912 if (RT_FAILURE(rc))
913 {
914 LogRel(("usbProxyConstruct: Failed to open '%s', rc=%Rrc\n", szAddress, rc));
915 return rc;
916 }
917 pThis->fOpened = true;
918
919 /*
920 * Get the device descriptor and format the device name (for logging).
921 */
922 if (!usbProxyGetDeviceDesc(pThis, &pThis->DevDesc))
923 {
924 Log(("usbProxyConstruct: usbProxyGetDeviceDesc failed\n"));
925 return VERR_READ_ERROR;
926 }
927
928 RTStrAPrintf(&pUsbIns->pszName, "%p[proxy %04x:%04x]", pThis, pThis->DevDesc.idVendor, pThis->DevDesc.idProduct); /** @todo append the user comment */
929 AssertReturn(pUsbIns->pszName, VERR_NO_MEMORY);
930
931 /*
932 * Get config descriptors.
933 */
934 size_t cbConfigs = pThis->DevDesc.bNumConfigurations * sizeof(pThis->paCfgDescs[0]);
935 pThis->paCfgDescs = (PVUSBDESCCONFIGEX)RTMemAllocZ(cbConfigs);
936 AssertReturn(pThis->paCfgDescs, VERR_NO_MEMORY);
937
938 unsigned i;
939 for (i = 0; i < pThis->DevDesc.bNumConfigurations; i++)
940 if (!copy_config(pThis, i, (PVUSBDESCCONFIGEX)&pThis->paCfgDescs[i]))
941 break;
942 if (i < pThis->DevDesc.bNumConfigurations)
943 {
944 Log(("usbProxyConstruct: copy_config failed, i=%d\n", i));
945 return VERR_READ_ERROR;
946 }
947
948 /*
949 * Pickup best matching global configuration for this device.
950 * The global configuration is organized like this:
951 *
952 * GlobalConfig/Whatever/
953 * |- idVendor = 300
954 * |- idProduct = 300
955 * - Config/
956 *
957 * The first level contains filter attributes which we stuff into a USBFILTER
958 * structure and match against the device info that's available. The highest
959 * ranked match is will be used. If nothing is found, the values will be
960 * queried from the GlobalConfig node (simplifies code and might actually
961 * be useful).
962 */
963 PCFGMNODE pCfgGlobalDev = pCfgGlobal;
964 PCFGMNODE pCur = CFGMR3GetFirstChild(pCfgGlobal);
965 if (pCur)
966 {
967 /*
968 * Create a device filter from the device configuration
969 * descriptor ++. No strings currently.
970 */
971 USBFILTER Device;
972 USBFilterInit(&Device, USBFILTERTYPE_CAPTURE);
973 rc = USBFilterSetNumExact(&Device, USBFILTERIDX_VENDOR_ID, pThis->DevDesc.idVendor, true); AssertRC(rc);
974 rc = USBFilterSetNumExact(&Device, USBFILTERIDX_PRODUCT_ID, pThis->DevDesc.idProduct, true); AssertRC(rc);
975 rc = USBFilterSetNumExact(&Device, USBFILTERIDX_DEVICE_REV, pThis->DevDesc.bcdDevice, true); AssertRC(rc);
976 rc = USBFilterSetNumExact(&Device, USBFILTERIDX_DEVICE_CLASS, pThis->DevDesc.bDeviceClass, true); AssertRC(rc);
977 rc = USBFilterSetNumExact(&Device, USBFILTERIDX_DEVICE_SUB_CLASS, pThis->DevDesc.bDeviceSubClass, true); AssertRC(rc);
978 rc = USBFilterSetNumExact(&Device, USBFILTERIDX_DEVICE_PROTOCOL, pThis->DevDesc.bDeviceProtocol, true); AssertRC(rc);
979 /** @todo manufacturer, product and serial strings */
980
981 int iBestMatchRate = -1;
982 PCFGMNODE pBestMatch = NULL;
983 for (pCur = CFGMR3GetFirstChild(pCfgGlobal); pCur; pCur = CFGMR3GetNextChild(pCur))
984 {
985 /*
986 * Construct a filter from the attributes in the node.
987 */
988 USBFILTER Filter;
989 USBFilterInit(&Filter, USBFILTERTYPE_CAPTURE);
990
991 /* numeric */
992 if ( RT_FAILURE(usbProxyQueryNum(&Filter, USBFILTERIDX_VENDOR_ID, pCur, "idVendor", "idVendorExpr"))
993 || RT_FAILURE(usbProxyQueryNum(&Filter, USBFILTERIDX_PRODUCT_ID, pCur, "idProduct", "idProcutExpr"))
994 || RT_FAILURE(usbProxyQueryNum(&Filter, USBFILTERIDX_DEVICE_REV, pCur, "bcdDevice", "bcdDeviceExpr"))
995 || RT_FAILURE(usbProxyQueryNum(&Filter, USBFILTERIDX_DEVICE_CLASS, pCur, "bDeviceClass", "bDeviceClassExpr"))
996 || RT_FAILURE(usbProxyQueryNum(&Filter, USBFILTERIDX_DEVICE_SUB_CLASS, pCur, "bDeviceSubClass", "bDeviceSubClassExpr"))
997 || RT_FAILURE(usbProxyQueryNum(&Filter, USBFILTERIDX_DEVICE_PROTOCOL, pCur, "bDeviceProtocol", "bDeviceProtocolExpr")))
998 continue; /* skip it */
999
1000 /* strings */
1001 /** @todo manufacturer, product and serial strings */
1002
1003 /* ignore unknown config values, but not without bitching. */
1004 if (!CFGMR3AreValuesValid(pCur,
1005 "idVendor\0idVendorExpr\0"
1006 "idProduct\0idProductExpr\0"
1007 "bcdDevice\0bcdDeviceExpr\0"
1008 "bDeviceClass\0bDeviceClassExpr\0"
1009 "bDeviceSubClass\0bDeviceSubClassExpr\0"
1010 "bDeviceProtocol\0bDeviceProtocolExpr\0"))
1011 LogRel(("usbProxyConstruct: Unknown value(s) in config filter (ignored)!\n"));
1012
1013 /*
1014 * Try match it and on match see if it has is a higher rate hit
1015 * than the previous match. Quit if its a 100% match.
1016 */
1017 int iRate = USBFilterMatchRated(&Filter, &Device);
1018 if (iRate > iBestMatchRate)
1019 {
1020 pBestMatch = pCur;
1021 iBestMatchRate = iRate;
1022 if (iRate >= 100)
1023 break;
1024 }
1025 }
1026 if (pBestMatch)
1027 pCfgGlobalDev = CFGMR3GetChild(pBestMatch, "Config");
1028 if (pCfgGlobalDev)
1029 pCfgGlobalDev = pCfgGlobal;
1030 }
1031
1032 /*
1033 * Query the rest of the configuration using the global as fallback.
1034 */
1035 rc = CFGMR3QueryU32(pCfg, "MaskedIfs", &pThis->fMaskedIfs);
1036 if (rc == VERR_CFGM_VALUE_NOT_FOUND)
1037 rc = CFGMR3QueryU32(pCfgGlobalDev, "MaskedIfs", &pThis->fMaskedIfs);
1038 if (rc == VERR_CFGM_VALUE_NOT_FOUND)
1039 pThis->fMaskedIfs = 0;
1040 else
1041 AssertRCReturn(rc, rc);
1042
1043 bool fForce11Device;
1044 rc = CFGMR3QueryBool(pCfg, "Force11Device", &fForce11Device);
1045 if (rc == VERR_CFGM_VALUE_NOT_FOUND)
1046 rc = CFGMR3QueryBool(pCfgGlobalDev, "Force11Device", &fForce11Device);
1047 if (rc == VERR_CFGM_VALUE_NOT_FOUND)
1048 fForce11Device = false;
1049 else
1050 AssertRCReturn(rc, rc);
1051
1052 bool fForce11PacketSize;
1053 rc = CFGMR3QueryBool(pCfg, "Force11PacketSize", &fForce11PacketSize);
1054 if (rc == VERR_CFGM_VALUE_NOT_FOUND)
1055 rc = CFGMR3QueryBool(pCfgGlobalDev, "Force11PacketSize", &fForce11PacketSize);
1056 if (rc == VERR_CFGM_VALUE_NOT_FOUND)
1057 fForce11PacketSize = false;
1058 else
1059 AssertRCReturn(rc, rc);
1060
1061 /*
1062 * If we're masking interfaces, edit the descriptors.
1063 */
1064 bool fEdited = pThis->fMaskedIfs != 0;
1065 if (pThis->fMaskedIfs)
1066 usbProxyDevEditOutMaskedIfs(pThis);
1067
1068 /*
1069 * Do 2.0 -> 1.1 device edits if requested to do so.
1070 */
1071 if ( fForce11PacketSize
1072 && pThis->DevDesc.bcdUSB >= 0x0200)
1073 {
1074 PVUSBDESCCONFIGEX paCfgs = pThis->paCfgDescs;
1075 for (unsigned iCfg = 0; iCfg < pThis->DevDesc.bNumConfigurations; iCfg++)
1076 {
1077 PVUSBINTERFACE paIfs = (PVUSBINTERFACE)paCfgs[iCfg].paIfs;
1078 for (unsigned iIf = 0; iIf < paCfgs[iCfg].Core.bNumInterfaces; iIf++)
1079 for (uint32_t iAlt = 0; iAlt < paIfs[iIf].cSettings; iAlt++)
1080 {
1081 /*
1082 * USB 1.1 defines the max for control, interrupt and bulk to be 64 bytes.
1083 * While isochronous has a max of 1023 bytes.
1084 */
1085 PVUSBDESCENDPOINTEX paEps = (PVUSBDESCENDPOINTEX)paIfs[iIf].paSettings[iAlt].paEndpoints;
1086 for (unsigned iEp = 0; iEp < paIfs[iIf].paSettings[iAlt].Core.bNumEndpoints; iEp++)
1087 {
1088 const uint16_t cbMax = (paEps[iEp].Core.bmAttributes & 3) == 1 /* isoc */
1089 ? 1023
1090 : 64;
1091 if (paEps[iEp].Core.wMaxPacketSize > cbMax)
1092 {
1093 Log(("usb-proxy: pProxyDev=%s correcting wMaxPacketSize from %#x to %#x (mainly for vista)\n",
1094 pUsbIns->pszName, paEps[iEp].Core.wMaxPacketSize, cbMax));
1095 paEps[iEp].Core.wMaxPacketSize = cbMax;
1096 fEdited = true;
1097 }
1098 }
1099 }
1100 }
1101 }
1102
1103 if ( fForce11Device
1104 && pThis->DevDesc.bcdUSB == 0x0200)
1105 {
1106 /*
1107 * Discourages windows from helping you find a 2.0 port.
1108 */
1109 Log(("usb-proxy: %s correcting USB version 2.0 to 1.1 (to avoid Windows warning)\n", pUsbIns->pszName));
1110 pThis->DevDesc.bcdUSB = 0x110;
1111 fEdited = true;
1112 }
1113
1114
1115 /*
1116 * Init the PDM/VUSB descriptor cache.
1117 */
1118 pThis->DescCache.pDevice = &pThis->DevDesc;
1119 pThis->DescCache.paConfigs = pThis->paCfgDescs;
1120 pThis->DescCache.paLanguages = NULL;
1121 pThis->DescCache.cLanguages = 0;
1122 pThis->DescCache.fUseCachedDescriptors = fEdited;
1123 pThis->DescCache.fUseCachedStringsDescriptors = false;
1124
1125 /*
1126 * Call the backend if it wishes to do some more initializing
1127 * after we've read the config and descriptors.
1128 */
1129 if (pThis->pOps->pfnInit)
1130 {
1131 rc = pThis->pOps->pfnInit(pThis);
1132 if (RT_FAILURE(rc))
1133 return rc;
1134 }
1135 pThis->fInited = true;
1136
1137 /*
1138 * We're good!
1139 */
1140 Log(("usb-proxy: created pProxyDev=%s address '%s' fMaskedIfs=%#x (rc=%Rrc)\n",
1141 pUsbIns->pszName, szAddress, pThis->fMaskedIfs, rc));
1142 return VINF_SUCCESS;
1143}
1144
1145
1146/**
1147 * The USB proxy device registration record.
1148 */
1149const PDMUSBREG g_UsbDevProxy =
1150{
1151 /* u32Version */
1152 PDM_USBREG_VERSION,
1153 /* szName */
1154 "USBProxy",
1155 /* pszDescription */
1156 "USB Proxy Device.",
1157 /* fFlags */
1158 0,
1159 /* cMaxInstances */
1160 ~0U,
1161 /* cbInstance */
1162 sizeof(USBPROXYDEV),
1163 /* pfnConstruct */
1164 usbProxyConstruct,
1165 /* pfnDestruct */
1166 usbProxyDestruct,
1167 /* pfnVMInitComplete */
1168 NULL,
1169 /* pfnVMPowerOn */
1170 NULL,
1171 /* pfnVMReset */
1172 NULL,
1173 /* pfnVMSuspend */
1174 NULL,
1175 /* pfnVMResume */
1176 NULL,
1177 /* pfnVMPowerOff */
1178 NULL,
1179 /* pfnHotPlugged */
1180 NULL,
1181 /* pfnHotUnplugged */
1182 NULL,
1183 /* pfnDriverAttach */
1184 NULL,
1185 /* pfnDriverDetach */
1186 NULL,
1187 /* pfnQueryInterface */
1188 NULL,
1189 /* pfnUsbReset */
1190 usbProxyDevReset,
1191 /* pfnUsbGetDescriptorCache */
1192 usbProxyDevGetDescriptorCache,
1193 /* pfnUsbSetConfiguration */
1194 usbProxyDevSetConfiguration,
1195 /* pfnUsbSetInterface */
1196 usbProxyDevSetInterface,
1197 /* pfnUsbClearHaltedEndpoint */
1198 usbProxyDevClearHaltedEndpoint,
1199 /* pfnUrbNew */
1200 NULL,
1201 /* pfnUrbQueue */
1202 usbProxyDevUrbQueue,
1203 /* pfnUrbCancel */
1204 usbProxyDevUrbCancel,
1205 /* pfnUrbReap */
1206 usbProxyDevUrbReap,
1207 /* pfnWakeup */
1208 usbProxyDevWakeup,
1209
1210 /* u32TheEnd */
1211 PDM_USBREG_VERSION
1212};
1213
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