VirtualBox

source: vbox/trunk/src/VBox/Devices/USB/VUSBDevice.cpp@ 59767

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

VUSB: Fix possible crash

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 59.9 KB
Line 
1/* $Id: VUSBDevice.cpp 59767 2016-02-22 10:35:42Z vboxsync $ */
2/** @file
3 * Virtual USB - Device.
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_VUSB
23#include <VBox/vmm/pdm.h>
24#include <VBox/vmm/vmapi.h>
25#include <VBox/err.h>
26#include <VBox/log.h>
27#include <iprt/alloc.h>
28#include <iprt/time.h>
29#include <iprt/thread.h>
30#include <iprt/semaphore.h>
31#include <iprt/string.h>
32#include <iprt/assert.h>
33#include <iprt/asm.h>
34#include "VUSBInternal.h"
35
36#include "VUSBSniffer.h"
37
38
39/*********************************************************************************************************************************
40* Structures and Typedefs *
41*********************************************************************************************************************************/
42/**
43 * Argument package of vusbDevResetThread().
44 */
45typedef struct vusb_reset_args
46{
47 /** Pointer to the device which is being reset. */
48 PVUSBDEV pDev;
49 /** The reset return code. */
50 int rc;
51 /** Pointer to the completion callback. */
52 PFNVUSBRESETDONE pfnDone;
53 /** User argument to pfnDone. */
54 void *pvUser;
55} VUSBRESETARGS, *PVUSBRESETARGS;
56
57
58/*********************************************************************************************************************************
59* Global Variables *
60*********************************************************************************************************************************/
61/** Default message pipe. */
62const VUSBDESCENDPOINTEX g_Endpoint0 =
63{
64 {
65 /* .bLength = */ VUSB_DT_ENDPOINT_MIN_LEN,
66 /* .bDescriptorType = */ VUSB_DT_ENDPOINT,
67 /* .bEndpointAddress = */ 0,
68 /* .bmAttributes = */ 0,
69 /* .wMaxPacketSize = */ 64,
70 /* .bInterval = */ 0
71 },
72 NULL
73};
74
75/** Default configuration. */
76const VUSBDESCCONFIGEX g_Config0 =
77{
78 {
79 /* .bLength = */ VUSB_DT_CONFIG_MIN_LEN,
80 /* .bDescriptorType = */ VUSB_DT_CONFIG,
81 /* .WTotalLength = */ 0, /* (auto-calculated) */
82 /* .bNumInterfaces = */ 0,
83 /* .bConfigurationValue =*/ 0,
84 /* .iConfiguration = */ 0,
85 /* .bmAttributes = */ 0x80,
86 /* .MaxPower = */ 14
87 },
88 NULL,
89 NULL
90};
91
92
93
94static PCVUSBDESCCONFIGEX vusbDevFindCfgDesc(PVUSBDEV pDev, int iCfg)
95{
96 if (iCfg == 0)
97 return &g_Config0;
98
99 for (unsigned i = 0; i < pDev->pDescCache->pDevice->bNumConfigurations; i++)
100 if (pDev->pDescCache->paConfigs[i].Core.bConfigurationValue == iCfg)
101 return &pDev->pDescCache->paConfigs[i];
102 return NULL;
103}
104
105static PVUSBINTERFACESTATE vusbDevFindIfState(PVUSBDEV pDev, int iIf)
106{
107 for (unsigned i = 0; i < pDev->pCurCfgDesc->Core.bNumInterfaces; i++)
108 if (pDev->paIfStates[i].pIf->paSettings[0].Core.bInterfaceNumber == iIf)
109 return &pDev->paIfStates[i];
110 return NULL;
111}
112
113static PCVUSBDESCINTERFACEEX vusbDevFindAltIfDesc(PVUSBDEV pDev, PCVUSBINTERFACESTATE pIfState, int iAlt)
114{
115 for (uint32_t i = 0; i < pIfState->pIf->cSettings; i++)
116 if (pIfState->pIf->paSettings[i].Core.bAlternateSetting == iAlt)
117 return &pIfState->pIf->paSettings[i];
118 return NULL;
119}
120
121void vusbDevMapEndpoint(PVUSBDEV pDev, PCVUSBDESCENDPOINTEX pEndPtDesc)
122{
123 uint8_t i8Addr = pEndPtDesc->Core.bEndpointAddress & 0xF;
124 PVUSBPIPE pPipe = &pDev->aPipes[i8Addr];
125 LogFlow(("vusbDevMapEndpoint: pDev=%p[%s] pEndPtDesc=%p{.bEndpointAddress=%#x, .bmAttributes=%#x} p=%p stage %s->SETUP\n",
126 pDev, pDev->pUsbIns->pszName, pEndPtDesc, pEndPtDesc->Core.bEndpointAddress, pEndPtDesc->Core.bmAttributes,
127 pPipe, g_apszCtlStates[pPipe->pCtrl ? pPipe->pCtrl->enmStage : 3]));
128
129 if ((pEndPtDesc->Core.bmAttributes & 0x3) == 0)
130 {
131 Log(("vusb: map message pipe on address %u\n", i8Addr));
132 pPipe->in = pEndPtDesc;
133 pPipe->out = pEndPtDesc;
134 }
135 else if (pEndPtDesc->Core.bEndpointAddress & 0x80)
136 {
137 Log(("vusb: map input pipe on address %u\n", i8Addr));
138 pPipe->in = pEndPtDesc;
139
140 ///@todo: This is currently utterly broken and causes untold damage.
141#if 0 //defined(RT_OS_LINUX) || defined(RT_OS_SOLARIS) || defined(RT_OS_DARWIN)
142 /*
143 * For high-speed isochronous input endpoints, spin off a read-ahead buffering thread.
144 */
145 if ((pEndPtDesc->Core.bmAttributes & 0x03) == 1)
146 pPipe->hReadAhead = vusbReadAheadStart(pDev, pPipe);
147#endif
148 }
149 else
150 {
151 Log(("vusb: map output pipe on address %u\n", i8Addr));
152 pPipe->out = pEndPtDesc;
153 }
154
155 if (pPipe->pCtrl)
156 {
157 vusbMsgFreeExtraData(pPipe->pCtrl);
158 pPipe->pCtrl = NULL;
159 }
160}
161
162static void unmap_endpoint(PVUSBDEV pDev, PCVUSBDESCENDPOINTEX pEndPtDesc)
163{
164 uint8_t EndPt = pEndPtDesc->Core.bEndpointAddress & 0xF;
165 PVUSBPIPE pPipe = &pDev->aPipes[EndPt];
166 LogFlow(("unmap_endpoint: pDev=%p[%s] pEndPtDesc=%p{.bEndpointAddress=%#x, .bmAttributes=%#x} p=%p stage %s->SETUP\n",
167 pDev, pDev->pUsbIns->pszName, pEndPtDesc, pEndPtDesc->Core.bEndpointAddress, pEndPtDesc->Core.bmAttributes,
168 pPipe, g_apszCtlStates[pPipe->pCtrl ? pPipe->pCtrl->enmStage : 3]));
169
170 if ((pEndPtDesc->Core.bmAttributes & 0x3) == 0)
171 {
172 Log(("vusb: unmap MSG pipe from address %u (%#x)\n", EndPt, pEndPtDesc->Core.bEndpointAddress));
173 pPipe->in = NULL;
174 pPipe->out = NULL;
175 }
176 else if (pEndPtDesc->Core.bEndpointAddress & 0x80)
177 {
178 Log(("vusb: unmap IN pipe from address %u (%#x)\n", EndPt, pEndPtDesc->Core.bEndpointAddress));
179 pPipe->in = NULL;
180
181 /* If there was a read-ahead thread associated with this endpoint, tell it to go away. */
182 if (pPipe->hReadAhead)
183 {
184 Log(("vusb: and tell read-ahead thread for the endpoint to terminate\n"));
185 vusbReadAheadStop(pPipe->hReadAhead);
186 pPipe->hReadAhead = NULL;
187 }
188 }
189 else
190 {
191 Log(("vusb: unmap OUT pipe from address %u (%#x)\n", EndPt, pEndPtDesc->Core.bEndpointAddress));
192 pPipe->out = NULL;
193 }
194
195 if (pPipe->pCtrl)
196 {
197 vusbMsgFreeExtraData(pPipe->pCtrl);
198 pPipe->pCtrl = NULL;
199 }
200}
201
202static void map_interface(PVUSBDEV pDev, PCVUSBDESCINTERFACEEX pIfDesc)
203{
204 LogFlow(("map_interface: pDev=%p[%s] pIfDesc=%p:{.iInterface=%d, .bAlternateSetting=%d}\n",
205 pDev, pDev->pUsbIns->pszName, pIfDesc, pIfDesc->Core.iInterface, pIfDesc->Core.bAlternateSetting));
206
207 for (unsigned i = 0; i < pIfDesc->Core.bNumEndpoints; i++)
208 {
209 if ((pIfDesc->paEndpoints[i].Core.bEndpointAddress & 0xF) == VUSB_PIPE_DEFAULT)
210 Log(("vusb: Endpoint 0x%x on interface %u.%u tried to override the default message pipe!!!\n",
211 pIfDesc->paEndpoints[i].Core.bEndpointAddress, pIfDesc->Core.bInterfaceNumber, pIfDesc->Core.bAlternateSetting));
212 else
213 vusbDevMapEndpoint(pDev, &pIfDesc->paEndpoints[i]);
214 }
215}
216
217
218/**
219 * Worker that resets the pipe data on select config and detach.
220 *
221 * This leaves the critical section unmolested
222 *
223 * @param pPipe The pipe which data should be reset.
224 */
225static void vusbDevResetPipeData(PVUSBPIPE pPipe)
226{
227 vusbMsgFreeExtraData(pPipe->pCtrl);
228 pPipe->pCtrl = NULL;
229
230 if (pPipe->hReadAhead)
231 {
232 vusbReadAheadStop(pPipe->hReadAhead);
233 pPipe->hReadAhead = NULL;
234 }
235
236 RT_ZERO(pPipe->in);
237 RT_ZERO(pPipe->out);
238 pPipe->async = 0;
239}
240
241
242bool vusbDevDoSelectConfig(PVUSBDEV pDev, PCVUSBDESCCONFIGEX pCfgDesc)
243{
244 LogFlow(("vusbDevDoSelectConfig: pDev=%p[%s] pCfgDesc=%p:{.iConfiguration=%d}\n",
245 pDev, pDev->pUsbIns->pszName, pCfgDesc, pCfgDesc->Core.iConfiguration));
246
247 /*
248 * Clean up all pipes and interfaces.
249 */
250 unsigned i;
251 for (i = 0; i < VUSB_PIPE_MAX; i++)
252 if (i != VUSB_PIPE_DEFAULT)
253 vusbDevResetPipeData(&pDev->aPipes[i]);
254 memset(pDev->paIfStates, 0, pCfgDesc->Core.bNumInterfaces * sizeof(pDev->paIfStates[0]));
255
256 /*
257 * Map in the default setting for every interface.
258 */
259 for (i = 0; i < pCfgDesc->Core.bNumInterfaces; i++)
260 {
261 PCVUSBINTERFACE pIf;
262 struct vusb_interface_state *pIfState;
263
264 pIf = &pCfgDesc->paIfs[i];
265 pIfState = &pDev->paIfStates[i];
266 pIfState->pIf = pIf;
267
268 /*
269 * Find the 0 setting, if it is not present we just use
270 * the lowest numbered one.
271 */
272 for (uint32_t j = 0; j < pIf->cSettings; j++)
273 {
274 if ( !pIfState->pCurIfDesc
275 || pIf->paSettings[j].Core.bAlternateSetting < pIfState->pCurIfDesc->Core.bAlternateSetting)
276 pIfState->pCurIfDesc = &pIf->paSettings[j];
277 if (pIfState->pCurIfDesc->Core.bAlternateSetting == 0)
278 break;
279 }
280
281 if (pIfState->pCurIfDesc)
282 map_interface(pDev, pIfState->pCurIfDesc);
283 }
284
285 pDev->pCurCfgDesc = pCfgDesc;
286
287 if (pCfgDesc->Core.bmAttributes & 0x40)
288 pDev->u16Status |= (1 << VUSB_DEV_SELF_POWERED);
289 else
290 pDev->u16Status &= ~(1 << VUSB_DEV_SELF_POWERED);
291
292 return true;
293}
294
295/**
296 * Standard device request: SET_CONFIGURATION
297 * @returns success indicator.
298 */
299static bool vusbDevStdReqSetConfig(PVUSBDEV pDev, int EndPt, PVUSBSETUP pSetup, uint8_t *pbBuf, uint32_t *pcbBuf)
300{
301 unsigned iCfg = pSetup->wValue & 0xff;
302
303 if ((pSetup->bmRequestType & VUSB_RECIP_MASK) != VUSB_TO_DEVICE)
304 {
305 Log(("vusb: error: %s: SET_CONFIGURATION - invalid request (dir) !!!\n", pDev->pUsbIns->pszName));
306 return false;
307 }
308
309 /*
310 * Check that the device is in a valid state.
311 * (The caller has already checked that it's not being reset.)
312 */
313 const VUSBDEVICESTATE enmState = vusbDevGetState(pDev);
314 if (enmState == VUSB_DEVICE_STATE_DEFAULT)
315 {
316 LogFlow(("vusbDevStdReqSetConfig: %s: default dev state !!?\n", pDev->pUsbIns->pszName));
317 return false;
318 }
319
320 PCVUSBDESCCONFIGEX pNewCfgDesc = vusbDevFindCfgDesc(pDev, iCfg);
321 if (!pNewCfgDesc)
322 {
323 Log(("vusb: error: %s: config %i not found !!!\n", pDev->pUsbIns->pszName, iCfg));
324 return false;
325 }
326
327 if (iCfg == 0)
328 vusbDevSetState(pDev, VUSB_DEVICE_STATE_ADDRESS);
329 else
330 vusbDevSetState(pDev, VUSB_DEVICE_STATE_CONFIGURED);
331 if (pDev->pUsbIns->pReg->pfnUsbSetConfiguration)
332 {
333 int rc = vusbDevIoThreadExecSync(pDev, (PFNRT)pDev->pUsbIns->pReg->pfnUsbSetConfiguration, 5,
334 pDev->pUsbIns, pNewCfgDesc->Core.bConfigurationValue,
335 pDev->pCurCfgDesc, pDev->paIfStates, pNewCfgDesc);
336 if (RT_FAILURE(rc))
337 {
338 Log(("vusb: error: %s: failed to set config %i (%Rrc) !!!\n", pDev->pUsbIns->pszName, iCfg, rc));
339 return false;
340 }
341 }
342 Log(("vusb: %p[%s]: SET_CONFIGURATION: Selected config %u\n", pDev, pDev->pUsbIns->pszName, iCfg));
343 return vusbDevDoSelectConfig(pDev, pNewCfgDesc);
344}
345
346
347/**
348 * Standard device request: GET_CONFIGURATION
349 * @returns success indicator.
350 */
351static bool vusbDevStdReqGetConfig(PVUSBDEV pDev, int EndPt, PVUSBSETUP pSetup, uint8_t *pbBuf, uint32_t *pcbBuf)
352{
353 if ((pSetup->bmRequestType & VUSB_RECIP_MASK) != VUSB_TO_DEVICE)
354 {
355 Log(("vusb: error: %s: GET_CONFIGURATION - invalid request (dir) !!!\n", pDev->pUsbIns->pszName));
356 return false;
357 }
358
359 /*
360 * Check that the device is in a valid state.
361 * (The caller has already checked that it's not being reset.)
362 */
363 const VUSBDEVICESTATE enmState = vusbDevGetState(pDev);
364 if ( enmState != VUSB_DEVICE_STATE_CONFIGURED
365 && enmState != VUSB_DEVICE_STATE_ADDRESS)
366 {
367 LogFlow(("vusbDevStdReqGetConfig: error: %s: invalid device state %d!!!\n", pDev->pUsbIns->pszName, enmState));
368 return false;
369 }
370
371 if (*pcbBuf < 1)
372 {
373 LogFlow(("vusbDevStdReqGetConfig: %s: no space for data!\n", pDev->pUsbIns->pszName));
374 return true;
375 }
376
377 uint8_t iCfg;
378 if (enmState == VUSB_DEVICE_STATE_ADDRESS)
379 iCfg = 0;
380 else
381 iCfg = pDev->pCurCfgDesc->Core.bConfigurationValue;
382
383 *pbBuf = iCfg;
384 *pcbBuf = 1;
385 LogFlow(("vusbDevStdReqGetConfig: %s: returns iCfg=%d\n", pDev->pUsbIns->pszName, iCfg));
386 return true;
387}
388
389/**
390 * Standard device request: GET_INTERFACE
391 * @returns success indicator.
392 */
393static bool vusbDevStdReqGetInterface(PVUSBDEV pDev, int EndPt, PVUSBSETUP pSetup, uint8_t *pbBuf, uint32_t *pcbBuf)
394{
395 if ((pSetup->bmRequestType & VUSB_RECIP_MASK) != VUSB_TO_INTERFACE)
396 {
397 Log(("vusb: error: %s: GET_INTERFACE - invalid request (dir) !!!\n", pDev->pUsbIns->pszName));
398 return false;
399 }
400
401 /*
402 * Check that the device is in a valid state.
403 * (The caller has already checked that it's not being reset.)
404 */
405 const VUSBDEVICESTATE enmState = vusbDevGetState(pDev);
406 if (enmState != VUSB_DEVICE_STATE_CONFIGURED)
407 {
408 LogFlow(("vusbDevStdReqGetInterface: error: %s: invalid device state %d!!!\n", pDev->pUsbIns->pszName, enmState));
409 return false;
410 }
411
412 if (*pcbBuf < 1)
413 {
414 LogFlow(("vusbDevStdReqGetInterface: %s: no space for data!\n", pDev->pUsbIns->pszName));
415 return true;
416 }
417
418 for (unsigned i = 0; i < pDev->pCurCfgDesc->Core.bNumInterfaces; i++)
419 {
420 PCVUSBDESCINTERFACEEX pIfDesc = pDev->paIfStates[i].pCurIfDesc;
421 if ( pIfDesc
422 && pSetup->wIndex == pIfDesc->Core.bInterfaceNumber)
423 {
424 *pbBuf = pIfDesc->Core.bAlternateSetting;
425 *pcbBuf = 1;
426 Log(("vusb: %s: GET_INTERFACE: %u.%u\n", pDev->pUsbIns->pszName, pIfDesc->Core.bInterfaceNumber, *pbBuf));
427 return true;
428 }
429 }
430
431 Log(("vusb: error: %s: GET_INTERFACE - unknown iface %u !!!\n", pDev->pUsbIns->pszName, pSetup->wIndex));
432 return false;
433}
434
435/**
436 * Standard device request: SET_INTERFACE
437 * @returns success indicator.
438 */
439static bool vusbDevStdReqSetInterface(PVUSBDEV pDev, int EndPt, PVUSBSETUP pSetup, uint8_t *pbBuf, uint32_t *pcbBuf)
440{
441 if ((pSetup->bmRequestType & VUSB_RECIP_MASK) != VUSB_TO_INTERFACE)
442 {
443 Log(("vusb: error: %s: SET_INTERFACE - invalid request (dir) !!!\n", pDev->pUsbIns->pszName));
444 return false;
445 }
446
447 /*
448 * Check that the device is in a valid state.
449 * (The caller has already checked that it's not being reset.)
450 */
451 const VUSBDEVICESTATE enmState = vusbDevGetState(pDev);
452 if (enmState != VUSB_DEVICE_STATE_CONFIGURED)
453 {
454 LogFlow(("vusbDevStdReqSetInterface: error: %s: invalid device state %d !!!\n", pDev->pUsbIns->pszName, enmState));
455 return false;
456 }
457
458 /*
459 * Find the interface.
460 */
461 uint8_t iIf = pSetup->wIndex;
462 PVUSBINTERFACESTATE pIfState = vusbDevFindIfState(pDev, iIf);
463 if (!pIfState)
464 {
465 LogFlow(("vusbDevStdReqSetInterface: error: %s: couldn't find interface %u !!!\n", pDev->pUsbIns->pszName, iIf));
466 return false;
467 }
468 uint8_t iAlt = pSetup->wValue;
469 PCVUSBDESCINTERFACEEX pIfDesc = vusbDevFindAltIfDesc(pDev, pIfState, iAlt);
470 if (!pIfDesc)
471 {
472 LogFlow(("vusbDevStdReqSetInterface: error: %s: couldn't find alt interface %u.%u !!!\n", pDev->pUsbIns->pszName, iIf, iAlt));
473 return false;
474 }
475
476 if (pDev->pUsbIns->pReg->pfnUsbSetInterface)
477 {
478 int rc = vusbDevIoThreadExecSync(pDev, (PFNRT)pDev->pUsbIns->pReg->pfnUsbSetInterface, 3, pDev->pUsbIns, iIf, iAlt);
479 if (RT_FAILURE(rc))
480 {
481 LogFlow(("vusbDevStdReqSetInterface: error: %s: couldn't find alt interface %u.%u (%Rrc)\n", pDev->pUsbIns->pszName, iIf, iAlt, rc));
482 return false;
483 }
484 }
485
486 for (unsigned i = 0; i < pIfState->pCurIfDesc->Core.bNumEndpoints; i++)
487 unmap_endpoint(pDev, &pIfState->pCurIfDesc->paEndpoints[i]);
488
489 Log(("vusb: SET_INTERFACE: Selected %u.%u\n", iIf, iAlt));
490
491 map_interface(pDev, pIfDesc);
492 pIfState->pCurIfDesc = pIfDesc;
493
494 return true;
495}
496
497/**
498 * Standard device request: SET_ADDRESS
499 * @returns success indicator.
500 */
501static bool vusbDevStdReqSetAddress(PVUSBDEV pDev, int EndPt, PVUSBSETUP pSetup, uint8_t *pbBuf, uint32_t *pcbBuf)
502{
503 if ((pSetup->bmRequestType & VUSB_RECIP_MASK) != VUSB_TO_DEVICE)
504 {
505 Log(("vusb: error: %s: SET_ADDRESS - invalid request (dir) !!!\n", pDev->pUsbIns->pszName));
506 return false;
507 }
508
509 /*
510 * Check that the device is in a valid state.
511 * (The caller has already checked that it's not being reset.)
512 */
513 const VUSBDEVICESTATE enmState = vusbDevGetState(pDev);
514 if ( enmState != VUSB_DEVICE_STATE_DEFAULT
515 && enmState != VUSB_DEVICE_STATE_ADDRESS)
516 {
517 LogFlow(("vusbDevStdReqSetAddress: error: %s: invalid device state %d !!!\n", pDev->pUsbIns->pszName, enmState));
518 return false;
519 }
520
521 pDev->u8NewAddress = pSetup->wValue;
522 return true;
523}
524
525/**
526 * Standard device request: CLEAR_FEATURE
527 * @returns success indicator.
528 *
529 * @remark This is only called for VUSB_TO_ENDPOINT && ep == 0 && wValue == ENDPOINT_HALT.
530 * All other cases of CLEAR_FEATURE is handled in the normal async/sync manner.
531 */
532static bool vusbDevStdReqClearFeature(PVUSBDEV pDev, int EndPt, PVUSBSETUP pSetup, uint8_t *pbBuf, uint32_t *pcbBuf)
533{
534 switch (pSetup->bmRequestType & VUSB_RECIP_MASK)
535 {
536 case VUSB_TO_DEVICE:
537 Log(("vusb: ClearFeature: dev(%u): selector=%u\n", pSetup->wIndex, pSetup->wValue));
538 break;
539 case VUSB_TO_INTERFACE:
540 Log(("vusb: ClearFeature: iface(%u): selector=%u\n", pSetup->wIndex, pSetup->wValue));
541 break;
542 case VUSB_TO_ENDPOINT:
543 Log(("vusb: ClearFeature: ep(%u): selector=%u\n", pSetup->wIndex, pSetup->wValue));
544 if ( !EndPt /* Default control pipe only */
545 && pSetup->wValue == 0 /* ENDPOINT_HALT */
546 && pDev->pUsbIns->pReg->pfnUsbClearHaltedEndpoint)
547 {
548 int rc = vusbDevIoThreadExecSync(pDev, (PFNRT)pDev->pUsbIns->pReg->pfnUsbClearHaltedEndpoint,
549 2, pDev->pUsbIns, pSetup->wIndex);
550 return RT_SUCCESS(rc);
551 }
552 break;
553 default:
554 AssertMsgFailed(("VUSB_TO_OTHER!\n"));
555 break;
556 }
557
558 AssertMsgFailed(("Invalid safe check !!!\n"));
559 return false;
560}
561
562/**
563 * Standard device request: SET_FEATURE
564 * @returns success indicator.
565 */
566static bool vusbDevStdReqSetFeature(PVUSBDEV pDev, int EndPt, PVUSBSETUP pSetup, uint8_t *pbBuf, uint32_t *pcbBuf)
567{
568 switch (pSetup->bmRequestType & VUSB_RECIP_MASK)
569 {
570 case VUSB_TO_DEVICE:
571 Log(("vusb: SetFeature: dev(%u): selector=%u\n",
572 pSetup->wIndex, pSetup->wValue));
573 break;
574 case VUSB_TO_INTERFACE:
575 Log(("vusb: SetFeature: if(%u): selector=%u\n",
576 pSetup->wIndex, pSetup->wValue));
577 break;
578 case VUSB_TO_ENDPOINT:
579 Log(("vusb: SetFeature: ep(%u): selector=%u\n",
580 pSetup->wIndex, pSetup->wValue));
581 break;
582 default:
583 AssertMsgFailed(("VUSB_TO_OTHER!\n"));
584 return false;
585 }
586 AssertMsgFailed(("This stuff is bogus\n"));
587 return false;
588}
589
590static bool vusbDevStdReqGetStatus(PVUSBDEV pDev, int EndPt, PVUSBSETUP pSetup, uint8_t *pbBuf, uint32_t *pcbBuf)
591{
592 if (*pcbBuf != 2)
593 {
594 LogFlow(("vusbDevStdReqGetStatus: %s: buffer is too small! (%d)\n", pDev->pUsbIns->pszName, *pcbBuf));
595 return false;
596 }
597
598 uint16_t u16Status;
599 switch (pSetup->bmRequestType & VUSB_RECIP_MASK)
600 {
601 case VUSB_TO_DEVICE:
602 u16Status = pDev->u16Status;
603 LogFlow(("vusbDevStdReqGetStatus: %s: device status %#x (%d)\n", pDev->pUsbIns->pszName, u16Status, u16Status));
604 break;
605 case VUSB_TO_INTERFACE:
606 u16Status = 0;
607 LogFlow(("vusbDevStdReqGetStatus: %s: bogus interface status request!!\n", pDev->pUsbIns->pszName));
608 break;
609 case VUSB_TO_ENDPOINT:
610 u16Status = 0;
611 LogFlow(("vusbDevStdReqGetStatus: %s: bogus endpoint status request!!\n", pDev->pUsbIns->pszName));
612 break;
613 default:
614 AssertMsgFailed(("VUSB_TO_OTHER!\n"));
615 return false;
616 }
617
618 *(uint16_t *)pbBuf = u16Status;
619 return true;
620}
621
622
623/**
624 * Finds a cached string.
625 *
626 * @returns Pointer to the cached string if found. NULL if not.
627 * @param paLanguages The languages to search.
628 * @param cLanguages The number of languages in the table.
629 * @param idLang The language ID.
630 * @param iString The string index.
631 */
632static PCPDMUSBDESCCACHESTRING FindCachedString(PCPDMUSBDESCCACHELANG paLanguages, unsigned cLanguages,
633 uint16_t idLang, uint8_t iString)
634{
635 /** @todo binary lookups! */
636 unsigned iCurLang = cLanguages;
637 while (iCurLang-- > 0)
638 if (paLanguages[iCurLang].idLang == idLang)
639 {
640 PCPDMUSBDESCCACHESTRING paStrings = paLanguages[iCurLang].paStrings;
641 unsigned iCurStr = paLanguages[iCurLang].cStrings;
642 while (iCurStr-- > 0)
643 if (paStrings[iCurStr].idx == iString)
644 return &paStrings[iCurStr];
645 break;
646 }
647 return NULL;
648}
649
650
651/** Macro for copying descriptor data. */
652#define COPY_DATA(pbDst, cbLeft, pvSrc, cbSrc) \
653 do { \
654 uint32_t cbSrc_ = cbSrc; \
655 uint32_t cbCopy = RT_MIN(cbLeft, cbSrc_); \
656 memcpy(pbBuf, pvSrc, cbCopy); \
657 cbLeft -= cbCopy; \
658 if (!cbLeft) \
659 return; \
660 pbBuf += cbCopy; \
661 } while (0)
662
663/**
664 * Internal function for reading the language IDs.
665 */
666static void ReadCachedStringDesc(PCPDMUSBDESCCACHESTRING pString, uint8_t *pbBuf, uint32_t *pcbBuf)
667{
668 uint32_t cbLeft = *pcbBuf;
669
670 RTUTF16 wsz[128]; /* 128-1 => bLength=0xff */
671 PRTUTF16 pwsz = wsz;
672 size_t cwc;
673 int rc = RTStrToUtf16Ex(pString->psz, RT_ELEMENTS(wsz) - 1, &pwsz, RT_ELEMENTS(wsz), &cwc);
674 if (RT_FAILURE(rc))
675 {
676 AssertRC(rc);
677 wsz[0] = 'e';
678 wsz[1] = 'r';
679 wsz[2] = 'r';
680 cwc = 3;
681 }
682
683 VUSBDESCSTRING StringDesc;
684 StringDesc.bLength = (uint8_t)(sizeof(StringDesc) + cwc * sizeof(RTUTF16));
685 StringDesc.bDescriptorType = VUSB_DT_STRING;
686 COPY_DATA(pbBuf, cbLeft, &StringDesc, sizeof(StringDesc));
687 COPY_DATA(pbBuf, cbLeft, wsz, (uint32_t)cwc * sizeof(RTUTF16));
688
689 /* updated the size of the output buffer. */
690 *pcbBuf -= cbLeft;
691}
692
693
694/**
695 * Internal function for reading the language IDs.
696 */
697static void ReadCachedLangIdDesc(PCPDMUSBDESCCACHELANG paLanguages, unsigned cLanguages,
698 uint8_t *pbBuf, uint32_t *pcbBuf)
699{
700 uint32_t cbLeft = *pcbBuf;
701
702 VUSBDESCLANGID LangIdDesc;
703 size_t cbDesc = sizeof(LangIdDesc) + cLanguages * sizeof(paLanguages[0].idLang);
704 LangIdDesc.bLength = (uint8_t)RT_MIN(0xff, cbDesc);
705 LangIdDesc.bDescriptorType = VUSB_DT_STRING;
706 COPY_DATA(pbBuf, cbLeft, &LangIdDesc, sizeof(LangIdDesc));
707
708 unsigned iLanguage = cLanguages;
709 while (iLanguage-- > 0)
710 COPY_DATA(pbBuf, cbLeft, &paLanguages[iLanguage].idLang, sizeof(paLanguages[iLanguage].idLang));
711
712 /* updated the size of the output buffer. */
713 *pcbBuf -= cbLeft;
714}
715
716
717/**
718 * Internal function which performs a descriptor read on the cached descriptors.
719 */
720static void ReadCachedConfigDesc(PCVUSBDESCCONFIGEX pCfgDesc, uint8_t *pbBuf, uint32_t *pcbBuf)
721{
722 uint32_t cbLeft = *pcbBuf;
723
724/** @todo See @bugref{2693} */
725 /*
726 * Make a copy of the config descriptor and calculate the wTotalLength field.
727 */
728 VUSBDESCCONFIG CfgDesc;
729 memcpy(&CfgDesc, pCfgDesc, VUSB_DT_CONFIG_MIN_LEN);
730 uint32_t cbTotal = pCfgDesc->Core.bLength;
731 for (unsigned i = 0; i < pCfgDesc->Core.bNumInterfaces; i++)
732 {
733 PCVUSBINTERFACE pIf = &pCfgDesc->paIfs[i];
734 for (uint32_t j = 0; j < pIf->cSettings; j++)
735 {
736 cbTotal += pIf->paSettings[j].cbIAD;
737 cbTotal += pIf->paSettings[j].Core.bLength;
738 cbTotal += pIf->paSettings[j].cbClass;
739 for (unsigned k = 0; k < pIf->paSettings[j].Core.bNumEndpoints; k++)
740 {
741 cbTotal += pIf->paSettings[j].paEndpoints[k].Core.bLength;
742 cbTotal += pIf->paSettings[j].paEndpoints[k].cbSsepc;
743 cbTotal += pIf->paSettings[j].paEndpoints[k].cbClass;
744 }
745 }
746 }
747 CfgDesc.wTotalLength = RT_H2LE_U16(cbTotal);
748
749 /*
750 * Copy the config descriptor
751 */
752 COPY_DATA(pbBuf, cbLeft, &CfgDesc, VUSB_DT_CONFIG_MIN_LEN);
753 COPY_DATA(pbBuf, cbLeft, pCfgDesc->pvMore, pCfgDesc->Core.bLength - VUSB_DT_CONFIG_MIN_LEN);
754
755 /*
756 * Copy out all the interfaces for this configuration
757 */
758 for (unsigned i = 0; i < pCfgDesc->Core.bNumInterfaces; i++)
759 {
760 PCVUSBINTERFACE pIf = &pCfgDesc->paIfs[i];
761 for (uint32_t j = 0; j < pIf->cSettings; j++)
762 {
763 PCVUSBDESCINTERFACEEX pIfDesc = &pIf->paSettings[j];
764
765 COPY_DATA(pbBuf, cbLeft, pIfDesc->pIAD, pIfDesc->cbIAD);
766 COPY_DATA(pbBuf, cbLeft, pIfDesc, VUSB_DT_INTERFACE_MIN_LEN);
767 COPY_DATA(pbBuf, cbLeft, pIfDesc->pvMore, pIfDesc->Core.bLength - VUSB_DT_INTERFACE_MIN_LEN);
768 COPY_DATA(pbBuf, cbLeft, pIfDesc->pvClass, pIfDesc->cbClass);
769
770 /*
771 * Copy out all the endpoints for this interface
772 */
773 for (unsigned k = 0; k < pIfDesc->Core.bNumEndpoints; k++)
774 {
775 VUSBDESCENDPOINT EndPtDesc;
776 memcpy(&EndPtDesc, &pIfDesc->paEndpoints[k], VUSB_DT_ENDPOINT_MIN_LEN);
777 EndPtDesc.wMaxPacketSize = RT_H2LE_U16(EndPtDesc.wMaxPacketSize);
778
779 COPY_DATA(pbBuf, cbLeft, &EndPtDesc, VUSB_DT_ENDPOINT_MIN_LEN);
780 COPY_DATA(pbBuf, cbLeft, pIfDesc->paEndpoints[k].pvMore, EndPtDesc.bLength - VUSB_DT_ENDPOINT_MIN_LEN);
781 COPY_DATA(pbBuf, cbLeft, pIfDesc->paEndpoints[k].pvSsepc, pIfDesc->paEndpoints[k].cbSsepc);
782 COPY_DATA(pbBuf, cbLeft, pIfDesc->paEndpoints[k].pvClass, pIfDesc->paEndpoints[k].cbClass);
783 }
784 }
785 }
786
787 /* updated the size of the output buffer. */
788 *pcbBuf -= cbLeft;
789}
790
791/**
792 * Internal function which performs a descriptor read on the cached descriptors.
793 */
794static void ReadCachedDeviceDesc(PCVUSBDESCDEVICE pDevDesc, uint8_t *pbBuf, uint32_t *pcbBuf)
795{
796 uint32_t cbLeft = *pcbBuf;
797
798 /*
799 * Duplicate the device description and update some fields we keep in cpu type.
800 */
801 Assert(sizeof(VUSBDESCDEVICE) == 18);
802 VUSBDESCDEVICE DevDesc = *pDevDesc;
803 DevDesc.bcdUSB = RT_H2LE_U16(DevDesc.bcdUSB);
804 DevDesc.idVendor = RT_H2LE_U16(DevDesc.idVendor);
805 DevDesc.idProduct = RT_H2LE_U16(DevDesc.idProduct);
806 DevDesc.bcdDevice = RT_H2LE_U16(DevDesc.bcdDevice);
807
808 COPY_DATA(pbBuf, cbLeft, &DevDesc, sizeof(DevDesc));
809 COPY_DATA(pbBuf, cbLeft, pDevDesc + 1, pDevDesc->bLength - sizeof(DevDesc));
810
811 /* updated the size of the output buffer. */
812 *pcbBuf -= cbLeft;
813}
814
815#undef COPY_DATA
816
817/**
818 * Standard device request: GET_DESCRIPTOR
819 * @returns success indicator.
820 * @remark not really used yet as we consider GET_DESCRIPTOR 'safe'.
821 */
822static bool vusbDevStdReqGetDescriptor(PVUSBDEV pDev, int EndPt, PVUSBSETUP pSetup, uint8_t *pbBuf, uint32_t *pcbBuf)
823{
824 if ((pSetup->bmRequestType & VUSB_RECIP_MASK) == VUSB_TO_DEVICE)
825 {
826 switch (pSetup->wValue >> 8)
827 {
828 case VUSB_DT_DEVICE:
829 ReadCachedDeviceDesc(pDev->pDescCache->pDevice, pbBuf, pcbBuf);
830 LogFlow(("vusbDevStdReqGetDescriptor: %s: %u bytes of device descriptors\n", pDev->pUsbIns->pszName, *pcbBuf));
831 return true;
832
833 case VUSB_DT_CONFIG:
834 {
835 unsigned int iIndex = (pSetup->wValue & 0xff);
836 if (iIndex >= pDev->pDescCache->pDevice->bNumConfigurations)
837 {
838 LogFlow(("vusbDevStdReqGetDescriptor: %s: iIndex=%p >= bNumConfigurations=%d !!!\n",
839 pDev->pUsbIns->pszName, iIndex, pDev->pDescCache->pDevice->bNumConfigurations));
840 return false;
841 }
842 ReadCachedConfigDesc(&pDev->pDescCache->paConfigs[iIndex], pbBuf, pcbBuf);
843 LogFlow(("vusbDevStdReqGetDescriptor: %s: %u bytes of config descriptors\n", pDev->pUsbIns->pszName, *pcbBuf));
844 return true;
845 }
846
847 case VUSB_DT_STRING:
848 {
849 if (pSetup->wIndex == 0)
850 {
851 ReadCachedLangIdDesc(pDev->pDescCache->paLanguages, pDev->pDescCache->cLanguages, pbBuf, pcbBuf);
852 LogFlow(("vusbDevStdReqGetDescriptor: %s: %u bytes of language ID (string) descriptors\n", pDev->pUsbIns->pszName, *pcbBuf));
853 return true;
854 }
855 PCPDMUSBDESCCACHESTRING pString;
856 pString = FindCachedString(pDev->pDescCache->paLanguages, pDev->pDescCache->cLanguages,
857 pSetup->wIndex, pSetup->wValue & 0xff);
858 if (pString)
859 {
860 ReadCachedStringDesc(pString, pbBuf, pcbBuf);
861 LogFlow(("vusbDevStdReqGetDescriptor: %s: %u bytes of string descriptors \"%s\"\n",
862 pDev->pUsbIns->pszName, *pcbBuf, pString->psz));
863 return true;
864 }
865 break;
866 }
867
868 default:
869 break;
870 }
871 }
872 Log(("vusb: %s: warning: unknown descriptor: type=%u descidx=%u lang=%u len=%u!!!\n",
873 pDev->pUsbIns->pszName, pSetup->wValue >> 8, pSetup->wValue & 0xff, pSetup->wIndex, pSetup->wLength));
874 return false;
875}
876
877
878/**
879 * Service the standard USB requests.
880 *
881 * Devices may call this from controlmsg() if you want vusb core to handle your standard
882 * request, it's not necessary - you could handle them manually
883 *
884 * @param pDev The device.
885 * @param EndPoint The endpoint.
886 * @param pSetup Pointer to the setup request structure.
887 * @param pvBuf Buffer?
888 * @param pcbBuf ?
889 */
890bool vusbDevStandardRequest(PVUSBDEV pDev, int EndPoint, PVUSBSETUP pSetup, void *pvBuf, uint32_t *pcbBuf)
891{
892 static bool (* const s_apfnStdReq[VUSB_REQ_MAX])(PVUSBDEV, int, PVUSBSETUP, uint8_t *, uint32_t *) =
893 {
894 vusbDevStdReqGetStatus,
895 vusbDevStdReqClearFeature,
896 NULL,
897 vusbDevStdReqSetFeature,
898 NULL,
899 vusbDevStdReqSetAddress,
900 vusbDevStdReqGetDescriptor,
901 NULL,
902 vusbDevStdReqGetConfig,
903 vusbDevStdReqSetConfig,
904 vusbDevStdReqGetInterface,
905 vusbDevStdReqSetInterface,
906 NULL /* for iso */
907 };
908
909 /*
910 * Check that the device is in a valid state.
911 */
912 const VUSBDEVICESTATE enmState = vusbDevGetState(pDev);
913 if (enmState == VUSB_DEVICE_STATE_RESET)
914 {
915 LogRel(("VUSB: %s: standard control message ignored, the device is resetting\n", pDev->pUsbIns->pszName));
916 return false;
917 }
918
919 /*
920 * Do the request if it's one we want to deal with.
921 */
922 if ( pSetup->bRequest >= VUSB_REQ_MAX
923 || !s_apfnStdReq[pSetup->bRequest])
924 {
925 Log(("vusb: warning: standard req not implemented: message %u: val=%u idx=%u len=%u !!!\n",
926 pSetup->bRequest, pSetup->wValue, pSetup->wIndex, pSetup->wLength));
927 return false;
928 }
929
930 return s_apfnStdReq[pSetup->bRequest](pDev, EndPoint, pSetup, (uint8_t *)pvBuf, pcbBuf);
931}
932
933
934/**
935 * Add a device to the address hash
936 */
937static void vusbDevAddressHash(PVUSBDEV pDev)
938{
939 if (pDev->u8Address == VUSB_INVALID_ADDRESS)
940 return;
941 uint8_t u8Hash = vusbHashAddress(pDev->u8Address);
942 pDev->pNextHash = pDev->pHub->pRootHub->apAddrHash[u8Hash];
943 pDev->pHub->pRootHub->apAddrHash[u8Hash] = pDev;
944}
945
946/**
947 * Remove a device from the address hash
948 */
949static void vusbDevAddressUnHash(PVUSBDEV pDev)
950{
951 if (pDev->u8Address == VUSB_INVALID_ADDRESS)
952 return;
953
954 uint8_t u8Hash = vusbHashAddress(pDev->u8Address);
955 pDev->u8Address = VUSB_INVALID_ADDRESS;
956 pDev->u8NewAddress = VUSB_INVALID_ADDRESS;
957
958 PVUSBDEV pCur = pDev->pHub->pRootHub->apAddrHash[u8Hash];
959 if (pCur == pDev)
960 {
961 /* special case, we're at the head */
962 pDev->pHub->pRootHub->apAddrHash[u8Hash] = pDev->pNextHash;
963 pDev->pNextHash = NULL;
964 }
965 else
966 {
967 /* search the list */
968 PVUSBDEV pPrev;
969 for (pPrev = pCur, pCur = pCur->pNextHash;
970 pCur;
971 pPrev = pCur, pCur = pCur->pNextHash)
972 {
973 if (pCur == pDev)
974 {
975 pPrev->pNextHash = pCur->pNextHash;
976 pDev->pNextHash = NULL;
977 break;
978 }
979 }
980 }
981}
982
983/**
984 * Sets the address of a device.
985 *
986 * Called by status_completion() and vusbDevResetWorker().
987 */
988void vusbDevSetAddress(PVUSBDEV pDev, uint8_t u8Address)
989{
990 LogFlow(("vusbDevSetAddress: pDev=%p[%s]/%i u8Address=%#x\n",
991 pDev, pDev->pUsbIns->pszName, pDev->i16Port, u8Address));
992
993 /*
994 * Check that the device is in a valid state.
995 */
996 const VUSBDEVICESTATE enmState = vusbDevGetState(pDev);
997 VUSBDEV_ASSERT_VALID_STATE(enmState);
998 if ( enmState == VUSB_DEVICE_STATE_ATTACHED
999 || enmState == VUSB_DEVICE_STATE_DETACHED)
1000 {
1001 LogFlow(("vusbDevSetAddress: %s: fails because %d < POWERED\n", pDev->pUsbIns->pszName, pDev->enmState));
1002 return;
1003 }
1004 if (enmState == VUSB_DEVICE_STATE_RESET)
1005 {
1006 LogRel(("VUSB: %s: set address ignored, the device is resetting\n", pDev->pUsbIns->pszName));
1007 return;
1008 }
1009
1010 /*
1011 * Ok, get on with it.
1012 */
1013 if (pDev->u8Address == u8Address)
1014 return;
1015
1016 PVUSBROOTHUB pRh = vusbDevGetRh(pDev);
1017 AssertPtrReturnVoid(pRh);
1018 if (pDev->u8Address == VUSB_DEFAULT_ADDRESS)
1019 pRh->pDefaultAddress = NULL;
1020
1021 vusbDevAddressUnHash(pDev);
1022
1023 if (u8Address == VUSB_DEFAULT_ADDRESS)
1024 {
1025 if (pRh->pDefaultAddress != NULL)
1026 {
1027 vusbDevAddressUnHash(pRh->pDefaultAddress);
1028 vusbDevSetStateCmp(pRh->pDefaultAddress, VUSB_DEVICE_STATE_POWERED, VUSB_DEVICE_STATE_DEFAULT);
1029 Log(("2 DEFAULT ADDRS\n"));
1030 }
1031
1032 pRh->pDefaultAddress = pDev;
1033 vusbDevSetState(pDev, VUSB_DEVICE_STATE_DEFAULT);
1034 }
1035 else
1036 vusbDevSetState(pDev, VUSB_DEVICE_STATE_ADDRESS);
1037
1038 pDev->u8Address = u8Address;
1039 vusbDevAddressHash(pDev);
1040
1041 Log(("vusb: %p[%s]/%i: Assigned address %u\n",
1042 pDev, pDev->pUsbIns->pszName, pDev->i16Port, u8Address));
1043}
1044
1045
1046static DECLCALLBACK(int) vusbDevCancelAllUrbsWorker(PVUSBDEV pDev, bool fDetaching)
1047{
1048 /*
1049 * Iterate the URBs and cancel them.
1050 */
1051 PVUSBURBVUSB pVUsbUrb, pVUsbUrbNext;
1052 RTListForEachSafe(&pDev->LstAsyncUrbs, pVUsbUrb, pVUsbUrbNext, VUSBURBVUSBINT, NdLst)
1053 {
1054 PVUSBURB pUrb = pVUsbUrb->pUrb;
1055
1056 Assert(pUrb->pVUsb->pDev == pDev);
1057
1058 LogFlow(("%s: vusbDevCancelAllUrbs: CANCELING URB\n", pUrb->pszDesc));
1059 int rc = vusbUrbCancelWorker(pUrb, CANCELMODE_FAIL);
1060 AssertRC(rc);
1061 }
1062
1063 /*
1064 * Reap any URBs which became ripe during cancel now.
1065 */
1066 RTCritSectEnter(&pDev->CritSectAsyncUrbs);
1067 unsigned cReaped;
1068 do
1069 {
1070 cReaped = 0;
1071 pVUsbUrb = RTListGetFirst(&pDev->LstAsyncUrbs, VUSBURBVUSBINT, NdLst);
1072 while (pVUsbUrb)
1073 {
1074 PVUSBURBVUSB pNext = RTListGetNext(&pDev->LstAsyncUrbs, pVUsbUrb, VUSBURBVUSBINT, NdLst);
1075 PVUSBURB pUrb = pVUsbUrb->pUrb;
1076 Assert(pUrb->pVUsb->pDev == pDev);
1077
1078 PVUSBURB pRipe = NULL;
1079 if (pUrb->enmState == VUSBURBSTATE_REAPED)
1080 pRipe = pUrb;
1081 else if (pUrb->enmState == VUSBURBSTATE_CANCELLED)
1082#ifdef RT_OS_WINDOWS /** @todo Windows doesn't do cancelling, thus this kludge to prevent really bad
1083 * things from happening if we leave a pending URB behinds. */
1084 pRipe = pDev->pUsbIns->pReg->pfnUrbReap(pDev->pUsbIns, fDetaching ? 1500 : 0 /*ms*/);
1085#else
1086 pRipe = pDev->pUsbIns->pReg->pfnUrbReap(pDev->pUsbIns, fDetaching ? 10 : 0 /*ms*/);
1087#endif
1088 else
1089 AssertMsgFailed(("pUrb=%p enmState=%d\n", pUrb, pUrb->enmState));
1090 if (pRipe)
1091 {
1092 if ( pNext
1093 && pRipe == pNext->pUrb)
1094 pNext = RTListGetNext(&pDev->LstAsyncUrbs, pNext, VUSBURBVUSBINT, NdLst);
1095 vusbUrbRipe(pRipe);
1096 cReaped++;
1097 }
1098
1099 pVUsbUrb = pNext;
1100 }
1101 } while (cReaped > 0);
1102
1103 /*
1104 * If we're detaching, we'll have to orphan any leftover URBs.
1105 */
1106 if (fDetaching)
1107 {
1108 RTListForEachSafe(&pDev->LstAsyncUrbs, pVUsbUrb, pVUsbUrbNext, VUSBURBVUSBINT, NdLst)
1109 {
1110 PVUSBURB pUrb = pVUsbUrb->pUrb;
1111 Assert(pUrb->pVUsb->pDev == pDev);
1112
1113 AssertMsgFailed(("%s: Leaking left over URB! state=%d pDev=%p[%s]\n",
1114 pUrb->pszDesc, pUrb->enmState, pDev, pDev->pUsbIns->pszName));
1115 vusbUrbUnlink(pUrb);
1116 /* Unlink isn't enough, because boundary timer and detaching will try to reap it.
1117 * It was tested with MSD & iphone attachment to vSMP guest, if
1118 * it breaks anything, please add comment here, why we should unlink only.
1119 */
1120 pUrb->pVUsb->pfnFree(pUrb);
1121 }
1122 }
1123 RTCritSectLeave(&pDev->CritSectAsyncUrbs);
1124 return VINF_SUCCESS;
1125}
1126
1127/**
1128 * Cancels and completes (with CRC failure) all async URBs pending
1129 * on a device. This is typically done as part of a reset and
1130 * before detaching a device.
1131 *
1132 * @param fDetaching If set, we will unconditionally unlink (and leak)
1133 * any URBs which isn't reaped.
1134 */
1135static void vusbDevCancelAllUrbs(PVUSBDEV pDev, bool fDetaching)
1136{
1137 int rc = vusbDevIoThreadExecSync(pDev, (PFNRT)vusbDevCancelAllUrbsWorker, 2, pDev, fDetaching);
1138 AssertRC(rc);
1139}
1140
1141
1142static DECLCALLBACK(int) vusbDevUrbIoThread(RTTHREAD hThread, void *pvUser)
1143{
1144 PVUSBDEV pDev = (PVUSBDEV)pvUser;
1145
1146 /* Notify the starter that we are up and running. */
1147 RTThreadUserSignal(hThread);
1148
1149 LogFlowFunc(("Entering work loop\n"));
1150
1151 while (!ASMAtomicReadBool(&pDev->fTerminate))
1152 {
1153 if (vusbDevGetState(pDev) != VUSB_DEVICE_STATE_RESET)
1154 vusbUrbDoReapAsyncDev(pDev, RT_INDEFINITE_WAIT);
1155
1156 /* Process any URBs waiting to be cancelled first. */
1157 int rc = RTReqQueueProcess(pDev->hReqQueueSync, 0); /* Don't wait if there is nothing to do. */
1158 Assert(RT_SUCCESS(rc) || rc == VERR_TIMEOUT);
1159 }
1160
1161 return VINF_SUCCESS;
1162}
1163
1164int vusbDevUrbIoThreadWakeup(PVUSBDEV pDev)
1165{
1166 ASMAtomicXchgBool(&pDev->fWokenUp, true);
1167 return pDev->pUsbIns->pReg->pfnWakeup(pDev->pUsbIns);
1168}
1169
1170/**
1171 * Create the URB I/O thread.
1172 *
1173 * @returns VBox status code.
1174 * @param pDev The VUSB device.
1175 */
1176int vusbDevUrbIoThreadCreate(PVUSBDEV pDev)
1177{
1178 int rc = VINF_SUCCESS;
1179
1180 ASMAtomicXchgBool(&pDev->fTerminate, false);
1181 rc = RTThreadCreateF(&pDev->hUrbIoThread, vusbDevUrbIoThread, pDev, 0, RTTHREADTYPE_IO,
1182 RTTHREADFLAGS_WAITABLE, "USBDevIo-%d", pDev->i16Port);
1183 if (RT_SUCCESS(rc))
1184 {
1185 /* Wait for it to become active. */
1186 rc = RTThreadUserWait(pDev->hUrbIoThread, RT_INDEFINITE_WAIT);
1187 }
1188
1189 return rc;
1190}
1191
1192/**
1193 * Destro the URB I/O thread.
1194 *
1195 * @returns VBox status code.
1196 * @param pDev The VUSB device.
1197 */
1198int vusbDevUrbIoThreadDestroy(PVUSBDEV pDev)
1199{
1200 int rc = VINF_SUCCESS;
1201 int rcThread = VINF_SUCCESS;
1202
1203 ASMAtomicXchgBool(&pDev->fTerminate, true);
1204 vusbDevUrbIoThreadWakeup(pDev);
1205
1206 rc = RTThreadWait(pDev->hUrbIoThread, RT_INDEFINITE_WAIT, &rcThread);
1207 if (RT_SUCCESS(rc))
1208 rc = rcThread;
1209
1210 pDev->hUrbIoThread = NIL_RTTHREAD;
1211
1212 return rc;
1213}
1214
1215
1216/**
1217 * Detaches a device from the hub it's attached to.
1218 *
1219 * @returns VBox status code.
1220 * @param pDev The device to detach.
1221 *
1222 * @remark This can be called in any state but reset.
1223 */
1224int vusbDevDetach(PVUSBDEV pDev)
1225{
1226 LogFlow(("vusbDevDetach: pDev=%p[%s] enmState=%#x\n", pDev, pDev->pUsbIns->pszName, pDev->enmState));
1227 VUSBDEV_ASSERT_VALID_STATE(pDev->enmState);
1228 Assert(pDev->enmState != VUSB_DEVICE_STATE_RESET);
1229
1230 vusbDevCancelAllUrbs(pDev, true);
1231 vusbDevAddressUnHash(pDev);
1232
1233 PVUSBROOTHUB pRh = vusbDevGetRh(pDev);
1234 if (!pRh)
1235 AssertMsgFailedReturn(("Not attached!\n"), VERR_VUSB_DEVICE_NOT_ATTACHED);
1236 if (pRh->pDefaultAddress == pDev)
1237 pRh->pDefaultAddress = NULL;
1238
1239 pDev->pHub->pOps->pfnDetach(pDev->pHub, pDev);
1240 pDev->i16Port = -1;
1241 vusbDevSetState(pDev, VUSB_DEVICE_STATE_DETACHED);
1242 pDev->pHub = NULL;
1243
1244 /* Remove the configuration */
1245 pDev->pCurCfgDesc = NULL;
1246 for (unsigned i = 0; i < RT_ELEMENTS(pDev->aPipes); i++)
1247 vusbDevResetPipeData(&pDev->aPipes[i]);
1248 return VINF_SUCCESS;
1249}
1250
1251
1252/**
1253 * Destroys a device, detaching it from the hub if necessary.
1254 *
1255 * @param pDev The device.
1256 * @thread EMT
1257 */
1258void vusbDevDestroy(PVUSBDEV pDev)
1259{
1260 LogFlow(("vusbDevDestroy: pDev=%p[%s] enmState=%d\n", pDev, pDev->pUsbIns->pszName, pDev->enmState));
1261
1262 /*
1263 * Deal with pending async reset.
1264 * (anything but reset)
1265 */
1266 vusbDevSetStateCmp(pDev, VUSB_DEVICE_STATE_DEFAULT, VUSB_DEVICE_STATE_RESET);
1267
1268 /*
1269 * Detach and free resources.
1270 */
1271 if (pDev->pHub)
1272 vusbDevDetach(pDev);
1273 RTMemFree(pDev->paIfStates);
1274 TMR3TimerDestroy(pDev->pResetTimer);
1275 pDev->pResetTimer = NULL;
1276 for (unsigned i = 0; i < RT_ELEMENTS(pDev->aPipes); i++)
1277 {
1278 Assert(pDev->aPipes[i].pCtrl == NULL);
1279 RTCritSectDelete(&pDev->aPipes[i].CritSectCtrl);
1280 }
1281
1282 /*
1283 * Destroy I/O thread and request queue last because they might still be used
1284 * when cancelling URBs.
1285 */
1286 vusbDevUrbIoThreadDestroy(pDev);
1287
1288 int rc = RTReqQueueDestroy(pDev->hReqQueueSync);
1289 AssertRC(rc);
1290
1291 if (pDev->hSniffer != VUSBSNIFFER_NIL)
1292 VUSBSnifferDestroy(pDev->hSniffer);
1293
1294 vusbUrbPoolDestroy(&pDev->UrbPool);
1295
1296 RTCritSectDelete(&pDev->CritSectAsyncUrbs);
1297 /* Not using vusbDevSetState() deliberately here because it would assert on the state. */
1298 pDev->enmState = VUSB_DEVICE_STATE_DESTROYED;
1299}
1300
1301
1302/* -=-=-=-=-=- VUSBIDEVICE methods -=-=-=-=-=- */
1303
1304
1305/**
1306 * The actual reset has been done, do completion on EMT.
1307 *
1308 * There are several things we have to do now, like set default
1309 * config and address, and cleanup the state of control pipes.
1310 *
1311 * It's possible that the device has a delayed destroy request
1312 * pending when we get here. This can happen for async resetting.
1313 * We deal with it here, since we're now executing on the EMT
1314 * thread and the destruction will be properly serialized now.
1315 *
1316 * @param pDev The device that is being reset.
1317 * @param rc The vusbDevResetWorker return code.
1318 * @param pfnDone The done callback specified by the caller of vusbDevReset().
1319 * @param pvUser The user argument for the callback.
1320 */
1321static void vusbDevResetDone(PVUSBDEV pDev, int rc, PFNVUSBRESETDONE pfnDone, void *pvUser)
1322{
1323 VUSBDEV_ASSERT_VALID_STATE(pDev->enmState);
1324 Assert(pDev->enmState == VUSB_DEVICE_STATE_RESET);
1325
1326 /*
1327 * Do control pipe cleanup regardless of state and result.
1328 */
1329 for (unsigned i = 0; i < VUSB_PIPE_MAX; i++)
1330 if (pDev->aPipes[i].pCtrl)
1331 vusbMsgResetExtraData(pDev->aPipes[i].pCtrl);
1332
1333 /*
1334 * Switch to the default state.
1335 */
1336 vusbDevSetState(pDev, VUSB_DEVICE_STATE_DEFAULT);
1337 pDev->u16Status = 0;
1338 vusbDevDoSelectConfig(pDev, &g_Config0);
1339 if (!vusbDevIsRh(pDev))
1340 vusbDevSetAddress(pDev, VUSB_DEFAULT_ADDRESS);
1341 if (pfnDone)
1342 pfnDone(&pDev->IDevice, rc, pvUser);
1343}
1344
1345
1346/**
1347 * Timer callback for doing reset completion.
1348 *
1349 * @param pUsbIns The USB device instance.
1350 * @param pTimer The timer instance.
1351 * @param pvUser The VUSB device data.
1352 * @thread EMT
1353 */
1354static DECLCALLBACK(void) vusbDevResetDoneTimer(PPDMUSBINS pUsbIns, PTMTIMER pTimer, void *pvUser)
1355{
1356 PVUSBDEV pDev = (PVUSBDEV)pvUser;
1357 PVUSBRESETARGS pArgs = (PVUSBRESETARGS)pDev->pvArgs;
1358 Assert(pDev->pUsbIns == pUsbIns);
1359
1360 AssertPtr(pArgs);
1361
1362 /*
1363 * Reset-done processing and cleanup.
1364 */
1365 pDev->pvArgs = NULL;
1366 vusbDevResetDone(pDev, pArgs->rc, pArgs->pfnDone, pArgs->pvUser);
1367 RTMemFree(pArgs);
1368}
1369
1370
1371/**
1372 * Perform the actual reset.
1373 *
1374 * @thread EMT or a VUSB reset thread.
1375 */
1376static int vusbDevResetWorker(PVUSBDEV pDev, bool fResetOnLinux, bool fUseTimer, PVUSBRESETARGS pArgs)
1377{
1378 int rc = VINF_SUCCESS;
1379 uint64_t u64EndTS = TMTimerGet(pDev->pResetTimer) + TMTimerFromMilli(pDev->pResetTimer, 10);
1380
1381 if (pDev->pUsbIns->pReg->pfnUsbReset)
1382 rc = pDev->pUsbIns->pReg->pfnUsbReset(pDev->pUsbIns, fResetOnLinux);
1383
1384 if (pArgs)
1385 {
1386 pArgs->rc = rc;
1387 rc = VINF_SUCCESS;
1388 }
1389
1390 if (fUseTimer)
1391 {
1392 /*
1393 * We use a timer to communicate the result back to EMT.
1394 * This avoids suspend + poweroff issues, and it should give
1395 * us more accurate scheduling than making this thread sleep.
1396 */
1397 int rc2 = TMTimerSet(pDev->pResetTimer, u64EndTS);
1398 AssertReleaseRC(rc2);
1399 }
1400
1401 LogFlow(("vusbDevResetWorker: %s: returns %Rrc\n", pDev->pUsbIns->pszName, rc));
1402 return rc;
1403}
1404
1405
1406/**
1407 * Resets a device.
1408 *
1409 * Since a device reset shall take at least 10ms from the guest point of view,
1410 * it must be performed asynchronously. We create a thread which performs this
1411 * operation and ensures it will take at least 10ms.
1412 *
1413 * At times - like init - a synchronous reset is required, this can be done
1414 * by passing NULL for pfnDone.
1415 *
1416 * While the device is being reset it is in the VUSB_DEVICE_STATE_RESET state.
1417 * On completion it will be in the VUSB_DEVICE_STATE_DEFAULT state if successful,
1418 * or in the VUSB_DEVICE_STATE_DETACHED state if the rest failed.
1419 *
1420 * @returns VBox status code.
1421 *
1422 * @param pDev Pointer to the VUSB device interface.
1423 * @param fResetOnLinux Whether it's safe to reset the device(s) on a linux
1424 * host system. See discussion of logical reconnects elsewhere.
1425 * @param pfnDone Pointer to the completion routine. If NULL a synchronous
1426 * reset is preformed not respecting the 10ms.
1427 * @param pVM Pointer to the VM handle for performing the done function
1428 * on the EMT thread.
1429 * @thread EMT
1430 */
1431static DECLCALLBACK(int) vusbIDeviceReset(PVUSBIDEVICE pDevice, bool fResetOnLinux, PFNVUSBRESETDONE pfnDone, void *pvUser, PVM pVM)
1432{
1433 PVUSBDEV pDev = (PVUSBDEV)pDevice;
1434 Assert(!pfnDone || pVM);
1435 LogFlow(("vusb: reset: [%s]/%i\n", pDev->pUsbIns->pszName, pDev->i16Port));
1436
1437 /*
1438 * Only one reset operation at a time.
1439 */
1440 const VUSBDEVICESTATE enmStateOld = vusbDevSetState(pDev, VUSB_DEVICE_STATE_RESET);
1441 if (enmStateOld == VUSB_DEVICE_STATE_RESET)
1442 {
1443 LogRel(("VUSB: %s: reset request is ignored, the device is already resetting!\n", pDev->pUsbIns->pszName));
1444 return VERR_VUSB_DEVICE_IS_RESETTING;
1445 }
1446
1447 /*
1448 * First, cancel all async URBs.
1449 */
1450 vusbDevCancelAllUrbs(pDev, false);
1451
1452 /* Async or sync? */
1453 if (pfnDone)
1454 {
1455 /*
1456 * Async fashion.
1457 */
1458 PVUSBRESETARGS pArgs = (PVUSBRESETARGS)RTMemTmpAlloc(sizeof(*pArgs));
1459 if (pArgs)
1460 {
1461 pArgs->pDev = pDev;
1462 pArgs->pfnDone = pfnDone;
1463 pArgs->pvUser = pvUser;
1464 pArgs->rc = VINF_SUCCESS;
1465 AssertPtrNull(pDev->pvArgs);
1466 pDev->pvArgs = pArgs;
1467 int rc = vusbDevIoThreadExec(pDev, 0 /* fFlags */, (PFNRT)vusbDevResetWorker, 4, pDev, fResetOnLinux, true, pArgs);
1468 if (RT_SUCCESS(rc))
1469 return rc;
1470
1471 RTMemTmpFree(pArgs);
1472 }
1473 /* fall back to sync on failure */
1474 }
1475
1476 /*
1477 * Sync fashion.
1478 */
1479 int rc = vusbDevResetWorker(pDev, fResetOnLinux, false, NULL);
1480 vusbDevResetDone(pDev, rc, pfnDone, pvUser);
1481 return rc;
1482}
1483
1484
1485/**
1486 * Powers on the device.
1487 *
1488 * @returns VBox status code.
1489 * @param pInterface Pointer to the device interface structure.
1490 */
1491static DECLCALLBACK(int) vusbIDevicePowerOn(PVUSBIDEVICE pInterface)
1492{
1493 PVUSBDEV pDev = (PVUSBDEV)pInterface;
1494 LogFlow(("vusbDevPowerOn: pDev=%p[%s]\n", pDev, pDev->pUsbIns->pszName));
1495
1496 /*
1497 * Check that the device is in a valid state.
1498 */
1499 const VUSBDEVICESTATE enmState = vusbDevGetState(pDev);
1500 if (enmState == VUSB_DEVICE_STATE_DETACHED)
1501 {
1502 Log(("vusb: warning: attempt to power on detached device %p[%s]\n", pDev, pDev->pUsbIns->pszName));
1503 return VERR_VUSB_DEVICE_NOT_ATTACHED;
1504 }
1505 if (enmState == VUSB_DEVICE_STATE_RESET)
1506 {
1507 LogRel(("VUSB: %s: power on ignored, the device is resetting!\n", pDev->pUsbIns->pszName));
1508 return VERR_VUSB_DEVICE_IS_RESETTING;
1509 }
1510
1511 /*
1512 * Do the job.
1513 */
1514 if (enmState == VUSB_DEVICE_STATE_ATTACHED)
1515 vusbDevSetState(pDev, VUSB_DEVICE_STATE_POWERED);
1516
1517 return VINF_SUCCESS;
1518}
1519
1520
1521/**
1522 * Powers off the device.
1523 *
1524 * @returns VBox status code.
1525 * @param pInterface Pointer to the device interface structure.
1526 */
1527static DECLCALLBACK(int) vusbIDevicePowerOff(PVUSBIDEVICE pInterface)
1528{
1529 PVUSBDEV pDev = (PVUSBDEV)pInterface;
1530 LogFlow(("vusbDevPowerOff: pDev=%p[%s]\n", pDev, pDev->pUsbIns->pszName));
1531
1532 /*
1533 * Check that the device is in a valid state.
1534 */
1535 const VUSBDEVICESTATE enmState = vusbDevGetState(pDev);
1536 if (enmState == VUSB_DEVICE_STATE_DETACHED)
1537 {
1538 Log(("vusb: warning: attempt to power off detached device %p[%s]\n", pDev, pDev->pUsbIns->pszName));
1539 return VERR_VUSB_DEVICE_NOT_ATTACHED;
1540 }
1541 if (enmState == VUSB_DEVICE_STATE_RESET)
1542 {
1543 LogRel(("VUSB: %s: power off ignored, the device is resetting!\n", pDev->pUsbIns->pszName));
1544 return VERR_VUSB_DEVICE_IS_RESETTING;
1545 }
1546
1547 /*
1548 * If it's a root hub, we will have to cancel all URBs and reap them.
1549 */
1550 if (vusbDevIsRh(pDev))
1551 {
1552 PVUSBROOTHUB pRh = (PVUSBROOTHUB)pDev;
1553 VUSBIRhCancelAllUrbs(&pRh->IRhConnector);
1554 VUSBIRhReapAsyncUrbs(&pRh->IRhConnector, pInterface, 0);
1555 }
1556
1557 vusbDevSetState(pDev, VUSB_DEVICE_STATE_ATTACHED);
1558 return VINF_SUCCESS;
1559}
1560
1561
1562/**
1563 * Get the state of the device.
1564 *
1565 * @returns Device state.
1566 * @param pInterface Pointer to the device interface structure.
1567 */
1568static DECLCALLBACK(VUSBDEVICESTATE) vusbIDeviceGetState(PVUSBIDEVICE pInterface)
1569{
1570 return vusbDevGetState((PVUSBDEV)pInterface);
1571}
1572
1573
1574/**
1575 * @interface_method_impl{VUSBIDEVICE,pfnIsSavedStateSupported}
1576 */
1577static DECLCALLBACK(bool) vusbIDeviceIsSavedStateSupported(PVUSBIDEVICE pInterface)
1578{
1579 PVUSBDEV pDev = (PVUSBDEV)pInterface;
1580 bool fSavedStateSupported = RT_BOOL(pDev->pUsbIns->pReg->fFlags & PDM_USBREG_SAVED_STATE_SUPPORTED);
1581
1582 LogFlowFunc(("pInterface=%p\n", pInterface));
1583
1584 LogFlowFunc(("returns %RTbool\n", fSavedStateSupported));
1585 return fSavedStateSupported;
1586}
1587
1588
1589/**
1590 * @interface_method_impl{VUSBIDEVICE,pfnGetState}
1591 */
1592static DECLCALLBACK(VUSBSPEED) vusbIDeviceGetSpeed(PVUSBIDEVICE pInterface)
1593{
1594 PVUSBDEV pDev = (PVUSBDEV)pInterface;
1595 VUSBSPEED enmSpeed = pDev->pUsbIns->enmSpeed;
1596
1597 LogFlowFunc(("pInterface=%p, returns %u\n", pInterface, enmSpeed));
1598 return enmSpeed;
1599}
1600
1601
1602/**
1603 * The maximum number of interfaces the device can have in all of it's configuration.
1604 *
1605 * @returns Number of interfaces.
1606 * @param pDev The device.
1607 */
1608size_t vusbDevMaxInterfaces(PVUSBDEV pDev)
1609{
1610 uint8_t cMax = 0;
1611 unsigned i = pDev->pDescCache->pDevice->bNumConfigurations;
1612 while (i-- > 0)
1613 {
1614 if (pDev->pDescCache->paConfigs[i].Core.bNumInterfaces > cMax)
1615 cMax = pDev->pDescCache->paConfigs[i].Core.bNumInterfaces;
1616 }
1617
1618 return cMax;
1619}
1620
1621
1622/**
1623 * Executes a given function on the I/O thread.
1624 *
1625 * @returns IPRT status code.
1626 * @param pDev The USB device instance data.
1627 * @param fFlags Combination of VUSB_DEV_IO_THREAD_EXEC_FLAGS_*
1628 * @param pfnFunction The function to execute.
1629 * @param cArgs Number of arguments to the function.
1630 * @param Args The parameter list.
1631 *
1632 * @remarks See remarks on RTReqQueueCallV
1633 */
1634DECLHIDDEN(int) vusbDevIoThreadExecV(PVUSBDEV pDev, uint32_t fFlags, PFNRT pfnFunction, unsigned cArgs, va_list Args)
1635{
1636 int rc = VINF_SUCCESS;
1637 PRTREQ hReq = NULL;
1638
1639 Assert(pDev->hUrbIoThread != NIL_RTTHREAD);
1640 if (RT_LIKELY(pDev->hUrbIoThread != NIL_RTTHREAD))
1641 {
1642 uint32_t fReqFlags = RTREQFLAGS_IPRT_STATUS;
1643
1644 if (!(fFlags & VUSB_DEV_IO_THREAD_EXEC_FLAGS_SYNC))
1645 fReqFlags |= RTREQFLAGS_NO_WAIT;
1646
1647 rc = RTReqQueueCallV(pDev->hReqQueueSync, &hReq, 0 /* cMillies */, fReqFlags, pfnFunction, cArgs, Args);
1648 Assert(RT_SUCCESS(rc) || rc == VERR_TIMEOUT);
1649
1650 /* In case we are called on the I/O thread just process the request. */
1651 if ( pDev->hUrbIoThread == RTThreadSelf()
1652 && (fFlags & VUSB_DEV_IO_THREAD_EXEC_FLAGS_SYNC))
1653 {
1654 int rc2 = RTReqQueueProcess(pDev->hReqQueueSync, 0);
1655 Assert(RT_SUCCESS(rc2) || rc2 == VERR_TIMEOUT);
1656 }
1657 else
1658 vusbDevUrbIoThreadWakeup(pDev);
1659
1660 if ( rc == VERR_TIMEOUT
1661 && (fFlags & VUSB_DEV_IO_THREAD_EXEC_FLAGS_SYNC))
1662 {
1663 rc = RTReqWait(hReq, RT_INDEFINITE_WAIT);
1664 AssertRC(rc);
1665 }
1666 RTReqRelease(hReq);
1667 }
1668 else
1669 rc = VERR_INVALID_STATE;
1670
1671 return rc;
1672}
1673
1674
1675/**
1676 * Executes a given function on the I/O thread.
1677 *
1678 * @returns IPRT status code.
1679 * @param pDev The USB device instance data.
1680 * @param fFlags Combination of VUSB_DEV_IO_THREAD_EXEC_FLAGS_*
1681 * @param pfnFunction The function to execute.
1682 * @param cArgs Number of arguments to the function.
1683 * @param ... The parameter list.
1684 *
1685 * @remarks See remarks on RTReqQueueCallV
1686 */
1687DECLHIDDEN(int) vusbDevIoThreadExec(PVUSBDEV pDev, uint32_t fFlags, PFNRT pfnFunction, unsigned cArgs, ...)
1688{
1689 int rc = VINF_SUCCESS;
1690 va_list va;
1691
1692 va_start(va, cArgs);
1693 rc = vusbDevIoThreadExecV(pDev, fFlags, pfnFunction, cArgs, va);
1694 va_end(va);
1695 return rc;
1696}
1697
1698
1699/**
1700 * Executes a given function synchronously on the I/O thread waiting for it to complete.
1701 *
1702 * @returns IPRT status code.
1703 * @param pDev The USB device instance data
1704 * @param pfnFunction The function to execute.
1705 * @param cArgs Number of arguments to the function.
1706 * @param ... The parameter list.
1707 *
1708 * @remarks See remarks on RTReqQueueCallV
1709 */
1710DECLHIDDEN(int) vusbDevIoThreadExecSync(PVUSBDEV pDev, PFNRT pfnFunction, unsigned cArgs, ...)
1711{
1712 int rc = VINF_SUCCESS;
1713 va_list va;
1714
1715 va_start(va, cArgs);
1716 rc = vusbDevIoThreadExecV(pDev, VUSB_DEV_IO_THREAD_EXEC_FLAGS_SYNC, pfnFunction, cArgs, va);
1717 va_end(va);
1718 return rc;
1719}
1720
1721
1722static DECLCALLBACK(int) vusbDevGetDescriptorCacheWorker(PPDMUSBINS pUsbIns, PCPDMUSBDESCCACHE *ppDescCache)
1723{
1724 *ppDescCache = pUsbIns->pReg->pfnUsbGetDescriptorCache(pUsbIns);
1725 return VINF_SUCCESS;
1726}
1727
1728/**
1729 * Initialize a new VUSB device.
1730 *
1731 * @returns VBox status code.
1732 * @param pDev The VUSB device to initialize.
1733 * @param pUsbIns Pointer to the PDM USB Device instance.
1734 */
1735int vusbDevInit(PVUSBDEV pDev, PPDMUSBINS pUsbIns, const char *pszCaptureFilename)
1736{
1737 /*
1738 * Initialize the device data members.
1739 * (All that are Non-Zero at least.)
1740 */
1741 Assert(!pDev->IDevice.pfnReset);
1742 Assert(!pDev->IDevice.pfnPowerOn);
1743 Assert(!pDev->IDevice.pfnPowerOff);
1744 Assert(!pDev->IDevice.pfnGetState);
1745 Assert(!pDev->IDevice.pfnIsSavedStateSupported);
1746
1747 pDev->IDevice.pfnReset = vusbIDeviceReset;
1748 pDev->IDevice.pfnPowerOn = vusbIDevicePowerOn;
1749 pDev->IDevice.pfnPowerOff = vusbIDevicePowerOff;
1750 pDev->IDevice.pfnGetState = vusbIDeviceGetState;
1751 pDev->IDevice.pfnIsSavedStateSupported = vusbIDeviceIsSavedStateSupported;
1752 pDev->IDevice.pfnGetSpeed = vusbIDeviceGetSpeed;
1753 pDev->pUsbIns = pUsbIns;
1754 pDev->pNext = NULL;
1755 pDev->pNextHash = NULL;
1756 pDev->pHub = NULL;
1757 pDev->enmState = VUSB_DEVICE_STATE_DETACHED;
1758 pDev->u8Address = VUSB_INVALID_ADDRESS;
1759 pDev->u8NewAddress = VUSB_INVALID_ADDRESS;
1760 pDev->i16Port = -1;
1761 pDev->u16Status = 0;
1762 pDev->pDescCache = NULL;
1763 pDev->pCurCfgDesc = NULL;
1764 pDev->paIfStates = NULL;
1765 RTListInit(&pDev->LstAsyncUrbs);
1766 memset(&pDev->aPipes[0], 0, sizeof(pDev->aPipes));
1767 for (unsigned i = 0; i < RT_ELEMENTS(pDev->aPipes); i++)
1768 {
1769 int rc = RTCritSectInit(&pDev->aPipes[i].CritSectCtrl);
1770 AssertRCReturn(rc, rc);
1771 }
1772 pDev->pResetTimer = NULL;
1773 pDev->hSniffer = VUSBSNIFFER_NIL;
1774
1775 int rc = RTCritSectInit(&pDev->CritSectAsyncUrbs);
1776 AssertRCReturn(rc, rc);
1777
1778 /* Create the URB pool. */
1779 rc = vusbUrbPoolInit(&pDev->UrbPool);
1780 AssertRCReturn(rc, rc);
1781
1782 /* Setup request queue executing synchronous tasks on the I/O thread. */
1783 rc = RTReqQueueCreate(&pDev->hReqQueueSync);
1784 AssertRCReturn(rc, rc);
1785
1786 /* Create I/O thread. */
1787 rc = vusbDevUrbIoThreadCreate(pDev);
1788 AssertRCReturn(rc, rc);
1789
1790 /*
1791 * Create the reset timer.
1792 */
1793 rc = PDMUsbHlpTMTimerCreate(pDev->pUsbIns, TMCLOCK_VIRTUAL, vusbDevResetDoneTimer, pDev, 0 /*fFlags*/,
1794 "USB Device Reset Timer", &pDev->pResetTimer);
1795 AssertRCReturn(rc, rc);
1796
1797 if (pszCaptureFilename)
1798 {
1799 rc = VUSBSnifferCreate(&pDev->hSniffer, 0, pszCaptureFilename, NULL, NULL);
1800 AssertRCReturn(rc, rc);
1801 }
1802
1803 /*
1804 * Get the descriptor cache from the device. (shall cannot fail)
1805 */
1806 rc = vusbDevIoThreadExecSync(pDev, (PFNRT)vusbDevGetDescriptorCacheWorker, 2, pUsbIns, &pDev->pDescCache);
1807 AssertRC(rc);
1808 AssertPtr(pDev->pDescCache);
1809#ifdef VBOX_STRICT
1810 if (pDev->pDescCache->fUseCachedStringsDescriptors)
1811 {
1812 int32_t iPrevId = -1;
1813 for (unsigned iLang = 0; iLang < pDev->pDescCache->cLanguages; iLang++)
1814 {
1815 Assert((int32_t)pDev->pDescCache->paLanguages[iLang].idLang > iPrevId);
1816 iPrevId = pDev->pDescCache->paLanguages[iLang].idLang;
1817
1818 int32_t idxPrevStr = -1;
1819 PCPDMUSBDESCCACHESTRING paStrings = pDev->pDescCache->paLanguages[iLang].paStrings;
1820 unsigned cStrings = pDev->pDescCache->paLanguages[iLang].cStrings;
1821 for (unsigned iStr = 0; iStr < cStrings; iStr++)
1822 {
1823 Assert((int32_t)paStrings[iStr].idx > idxPrevStr);
1824 idxPrevStr = paStrings[iStr].idx;
1825 size_t cch = strlen(paStrings[iStr].psz);
1826 Assert(cch <= 127);
1827 }
1828 }
1829 }
1830#endif
1831
1832 /*
1833 * Allocate memory for the interface states.
1834 */
1835 size_t cbIface = vusbDevMaxInterfaces(pDev) * sizeof(*pDev->paIfStates);
1836 pDev->paIfStates = (PVUSBINTERFACESTATE)RTMemAllocZ(cbIface);
1837 AssertMsgReturn(pDev->paIfStates, ("RTMemAllocZ(%d) failed\n", cbIface), VERR_NO_MEMORY);
1838
1839 return VINF_SUCCESS;
1840}
1841
1842/*
1843 * Local Variables:
1844 * mode: c
1845 * c-file-style: "bsd"
1846 * c-basic-offset: 4
1847 * tab-width: 4
1848 * indent-tabs-mode: s
1849 * End:
1850 */
1851
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