VirtualBox

source: vbox/trunk/src/VBox/Main/src-server/HostUSBDeviceImpl.cpp@ 58414

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

USBIdDatabase*: Size optimization reducing VBoxSVC by xxxx bytes on 64-bit Windows (debug) - just for the fun of it. :-)

  • Use parallel arrays instead of structures/classes with alignment padding (4 or 6 bytes on 64-bit hosts).
  • Exchanging the vendor id in the product table for a product table index and length in the vendor entry.
  • Use a string table with 32-bit references that includes the size.
  • The 32-bit references saves relocations.
  • The string table skips the terminator char, saving 1 byte per string.
  • The string table eliminates duplicate strings and duplicate substrings.
  • Very simple compression that replaces the 127 most frequently used words.
  • Replaced lower_bound with handcoded binary lookup (easier for parallel arrays, seems to reduce code size too).

Before:
VBoxSVC.exe w/ db: 10145792 bytes,
VBoxSVC.exe w/o db: 9436672 bytes,
Database cost: 709120 bytes.

After:
VBoxSVC.exe w/ db: 9797120 bytes
VBoxSVC.exe w/o db: 9439232 bytes
Database cost: 357888 bytes

Saving 351232 bytes (49%).

PS. Relocations saving estimated to be around 30987 bytes (a bit uncertain because of string pooling by the compiler / linker).

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 96.5 KB
Line 
1/* $Id: HostUSBDeviceImpl.cpp 58016 2015-10-03 18:47:11Z vboxsync $ */
2/** @file
3 * VirtualBox IHostUSBDevice COM interface implementation.
4 */
5
6/*
7 * Copyright (C) 2005-2014 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#include <iprt/types.h> /* for UINT64_C */
20
21#include "HostUSBDeviceImpl.h"
22#include "MachineImpl.h"
23#include "HostImpl.h"
24#include "VirtualBoxErrorInfoImpl.h"
25#include "USBProxyService.h"
26#include "USBIdDatabase.h"
27
28#include "AutoCaller.h"
29#include "Logging.h"
30
31#include <VBox/err.h>
32#include <iprt/cpp/utils.h>
33
34// constructor / destructor
35/////////////////////////////////////////////////////////////////////////////
36
37DEFINE_EMPTY_CTOR_DTOR(HostUSBDevice)
38
39HRESULT HostUSBDevice::FinalConstruct()
40{
41 mUSBProxyService = NULL;
42 mUsb = NULL;
43
44 return BaseFinalConstruct();
45}
46
47void HostUSBDevice::FinalRelease()
48{
49 uninit();
50 BaseFinalRelease();
51}
52
53// public initializer/uninitializer for internal purposes only
54/////////////////////////////////////////////////////////////////////////////
55
56/**
57 * Initializes the USB device object.
58 *
59 * @returns COM result indicator
60 * @param aUsb Pointer to the usb device structure for which the object is to be a wrapper.
61 * This structure is now fully owned by the HostUSBDevice object and will be
62 * freed when it is destructed.
63 * @param aUSBProxyService Pointer to the USB Proxy Service object.
64 */
65HRESULT HostUSBDevice::init(PUSBDEVICE aUsb, USBProxyService *aUSBProxyService)
66{
67 ComAssertRet(aUsb, E_INVALIDARG);
68
69 /* Enclose the state transition NotReady->InInit->Ready */
70 AutoInitSpan autoInitSpan(this);
71 AssertReturn(autoInitSpan.isOk(), E_FAIL);
72
73 /*
74 * We need a unique ID for this VBoxSVC session.
75 * The UUID isn't stored anywhere.
76 */
77 unconst(mId).create();
78
79 /*
80 * Set the initial device state.
81 */
82 AssertMsgReturn( aUsb->enmState >= USBDEVICESTATE_UNSUPPORTED
83 && aUsb->enmState < USBDEVICESTATE_USED_BY_GUEST, /* used-by-guest is not a legal initial state. */
84 ("%d\n", aUsb->enmState), E_FAIL);
85 mUniState = (HostUSBDeviceState)aUsb->enmState;
86 mUniSubState = kHostUSBDeviceSubState_Default;
87 mPendingUniState = kHostUSBDeviceState_Invalid;
88 mPrevUniState = mUniState;
89 mIsPhysicallyDetached = false;
90
91 /* Other data members */
92 mUSBProxyService = aUSBProxyService;
93 mUsb = aUsb;
94
95 /* Set the name. */
96 mNameObj = i_getName();
97 mName = mNameObj.c_str();
98
99 /* Confirm the successful initialization */
100 autoInitSpan.setSucceeded();
101
102 return S_OK;
103}
104
105/**
106 * Uninitializes the instance and sets the ready flag to FALSE.
107 * Called either from FinalRelease() or by the parent when it gets destroyed.
108 */
109void HostUSBDevice::uninit()
110{
111 /* Enclose the state transition Ready->InUninit->NotReady */
112 AutoUninitSpan autoUninitSpan(this);
113 if (autoUninitSpan.uninitDone())
114 return;
115
116 if (mUsb != NULL)
117 {
118 USBProxyService::freeDevice(mUsb);
119 mUsb = NULL;
120 }
121
122 mUSBProxyService = NULL;
123 mUniState = kHostUSBDeviceState_Invalid;
124}
125
126// Wrapped IUSBDevice properties
127/////////////////////////////////////////////////////////////////////////////
128HRESULT HostUSBDevice::getId(com::Guid &aId)
129{
130 /* mId is constant during life time, no need to lock */
131 aId = mId;
132
133 return S_OK;
134}
135
136
137HRESULT HostUSBDevice::getVendorId(USHORT *aVendorId)
138{
139 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
140
141 *aVendorId = mUsb->idVendor;
142
143 return S_OK;
144}
145
146HRESULT HostUSBDevice::getProductId(USHORT *aProductId)
147{
148 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
149
150 *aProductId = mUsb->idProduct;
151
152 return S_OK;
153}
154
155
156HRESULT HostUSBDevice::getRevision(USHORT *aRevision)
157{
158 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
159
160 *aRevision = mUsb->bcdDevice;
161
162 return S_OK;
163}
164
165HRESULT HostUSBDevice::getManufacturer(com::Utf8Str &aManufacturer)
166{
167 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
168
169 aManufacturer = mUsb->pszManufacturer;
170 if (mUsb->pszManufacturer == NULL || mUsb->pszManufacturer[0] == 0)
171 aManufacturer = USBIdDatabase::findVendor(mUsb->idVendor);
172 return S_OK;
173}
174
175
176HRESULT HostUSBDevice::getProduct(com::Utf8Str &aProduct)
177{
178 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
179
180 aProduct = mUsb->pszProduct;
181 if (mUsb->pszProduct == NULL || mUsb->pszProduct[0] == 0)
182 aProduct = USBIdDatabase::findProduct(mUsb->idVendor, mUsb->idProduct);
183 return S_OK;
184}
185
186
187HRESULT HostUSBDevice::getSerialNumber(com::Utf8Str &aSerialNumber)
188{
189 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
190
191 aSerialNumber = mUsb->pszSerialNumber;
192
193 return S_OK;
194}
195
196HRESULT HostUSBDevice::getAddress(com::Utf8Str &aAddress)
197{
198 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
199 aAddress = mUsb->pszAddress;
200 return S_OK;
201}
202
203
204HRESULT HostUSBDevice::getPort(USHORT *aPort)
205{
206 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
207
208#if !defined(RT_OS_WINDOWS) /// @todo check up the bPort value on Windows before enabling this.
209 *aPort = mUsb->bPort;
210#else
211 *aPort = 0;
212#endif
213
214 return S_OK;
215}
216
217
218HRESULT HostUSBDevice::getVersion(USHORT *aVersion)
219{
220 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
221
222 *aVersion = mUsb->bcdUSB >> 8;
223
224 return S_OK;
225}
226
227
228HRESULT HostUSBDevice::getSpeed(USBConnectionSpeed_T *aSpeed)
229{
230 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
231
232 /* If the speed is unknown (which it shouldn't be), make a guess
233 * which will be correct for USB 1 and 3 devices, but may be wrong
234 * for USB 2.0 devices
235 */
236 switch (mUsb->enmSpeed)
237 {
238 case USBDEVICESPEED_LOW: *aSpeed = USBConnectionSpeed_Low; break;
239 case USBDEVICESPEED_FULL: *aSpeed = USBConnectionSpeed_Full; break;
240 case USBDEVICESPEED_HIGH: *aSpeed = USBConnectionSpeed_High; break;
241 case USBDEVICESPEED_SUPER: *aSpeed = USBConnectionSpeed_Super; break;
242// case USBDEVICESPEED_SUPERPLUS: *aSpeed = USBConnectionSpeed_SuperPlus; break;
243 default:
244 switch (mUsb->bcdUSB >> 8)
245 {
246 case 3: *aSpeed = USBConnectionSpeed_Super; break;
247 case 2: *aSpeed = USBConnectionSpeed_High; break;
248 default: *aSpeed = USBConnectionSpeed_Full;
249 }
250 }
251
252 return S_OK;
253}
254
255
256HRESULT HostUSBDevice::getPortVersion(USHORT *aPortVersion)
257{
258 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
259 /* Port version is 2 (EHCI) if and only if the device runs at high speed;
260 * if speed is unknown, fall back to the old and inaccurate method.
261 */
262 if (mUsb->enmSpeed == USBDEVICESPEED_UNKNOWN)
263 *aPortVersion = mUsb->bcdUSB >> 8;
264 else
265 {
266 switch (mUsb->enmSpeed)
267 {
268 case USBDEVICESPEED_SUPER:
269 *aPortVersion = 3;
270 break;
271 case USBDEVICESPEED_HIGH:
272 *aPortVersion = 2;
273 break;
274 case USBDEVICESPEED_FULL:
275 case USBDEVICESPEED_LOW:
276 case USBDEVICESPEED_VARIABLE:
277 *aPortVersion = 1;
278 break;
279 default:
280 AssertMsgFailed(("Invalid USB speed: %d\n", mUsb->enmSpeed));
281 *aPortVersion = 1;
282 }
283 }
284
285 return S_OK;
286}
287
288
289HRESULT HostUSBDevice::getRemote(BOOL *aRemote)
290{
291 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
292
293 *aRemote = FALSE;
294
295 return S_OK;
296}
297
298
299HRESULT HostUSBDevice::getState(USBDeviceState_T *aState)
300{
301 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
302
303 *aState = i_canonicalState();
304
305 return S_OK;
306}
307
308
309// public methods only for internal purposes
310////////////////////////////////////////////////////////////////////////////////
311
312/**
313 * @note Locks this object for reading.
314 */
315com::Utf8Str HostUSBDevice::i_getName()
316{
317 Utf8Str name;
318
319 AutoCaller autoCaller(this);
320 AssertComRCReturn(autoCaller.rc(), name);
321
322 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
323
324 bool haveManufacturer = mUsb->pszManufacturer && *mUsb->pszManufacturer;
325 bool haveProduct = mUsb->pszProduct && *mUsb->pszProduct;
326 if (haveManufacturer && haveProduct)
327 name = Utf8StrFmt("%s %s", mUsb->pszManufacturer, mUsb->pszProduct);
328 else if (haveManufacturer)
329 name = mUsb->pszManufacturer;
330 else if (haveProduct)
331 name = mUsb->pszProduct;
332 else
333 {
334 Utf8Str strProduct;
335 Utf8Str strVendor = USBIdDatabase::findVendorAndProduct(mUsb->idVendor, mUsb->idProduct, &strProduct);
336 if (strVendor.isNotEmpty() && strProduct.isNotEmpty())
337 name = Utf8StrFmt("%s %s", strVendor.c_str(), strProduct.c_str());
338 else
339 {
340 LogRel(("USB: Unknown USB device detected (idVendor: 0x%04x, idProduct: 0x%04x). Please, report the idVendor and idProduct to virtualbox.org.\n",
341 mUsb->idVendor, mUsb->idProduct));
342 if (strVendor.isNotEmpty())
343 name = strVendor;
344 else
345 {
346 Assert(strProduct.isEmpty());
347 name = "<unknown>";
348 }
349 }
350 }
351
352 return name;
353}
354
355/**
356 * Requests the USB proxy service capture the device (from the host)
357 * and attach it to a VM.
358 *
359 * As a convenience, this method will operate like attachToVM() if the device
360 * is already held by the proxy. Note that it will then perform IPC to the VM
361 * process, which means it will temporarily release all locks. (Is this a good idea?)
362 *
363 * @param aMachine Machine this device should be attach to.
364 * @param aSetError Whether to set full error message or not to bother.
365 * @param aCaptureFilename The filename to capture the USB traffic to.
366 * @param aMaskedIfs The interfaces to hide from the guest.
367 *
368 * @returns Status indicating whether it was successfully captured and/or attached.
369 * @retval S_OK on success.
370 * @retval E_UNEXPECTED if the device state doesn't permit for any attaching.
371 * @retval E_* as appropriate.
372 */
373HRESULT HostUSBDevice::i_requestCaptureForVM(SessionMachine *aMachine, bool aSetError,
374 const com::Utf8Str &aCaptureFilename, ULONG aMaskedIfs /* = 0*/)
375{
376 /*
377 * Validate preconditions and input.
378 */
379 AssertReturn(aMachine, E_INVALIDARG);
380 AssertReturn(!isWriteLockOnCurrentThread(), E_FAIL);
381 AssertReturn(!aMachine->isWriteLockOnCurrentThread(), E_FAIL);
382
383 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
384 LogFlowThisFunc(("{%s} aMachine=%p aMaskedIfs=%#x\n", mName, aMachine, aMaskedIfs));
385
386 if (aSetError)
387 {
388 if (mUniState == kHostUSBDeviceState_Unsupported)
389 return setError(E_INVALIDARG,
390 tr("USB device '%s' with UUID {%RTuuid} cannot be accessed by guest computers"),
391 mName, mId.raw());
392 if (mUniState == kHostUSBDeviceState_UsedByHost)
393 return setError(E_INVALIDARG,
394 tr("USB device '%s' with UUID {%RTuuid} is being exclusively used by the host computer"),
395 mName, mId.raw());
396 if (mUniState == kHostUSBDeviceState_UsedByVM)
397 {
398 /* Machine::name() requires a read lock */
399 alock.release();
400 AutoReadLock machLock(mMachine COMMA_LOCKVAL_SRC_POS);
401 return setError(E_INVALIDARG,
402 tr("USB device '%s' with UUID {%RTuuid} is already captured by the virtual machine '%s'"),
403 mName, mId.raw(), mMachine->i_getName().c_str());
404 }
405 if (mUniState >= kHostUSBDeviceState_FirstTransitional)
406 return setError(E_INVALIDARG,
407 tr("USB device '%s' with UUID {%RTuuid} is busy with a previous request. Please try again later"),
408 mName, mId.raw());
409 if ( mUniState != kHostUSBDeviceState_Unused
410 && mUniState != kHostUSBDeviceState_HeldByProxy
411 && mUniState != kHostUSBDeviceState_Capturable)
412 return setError(E_INVALIDARG,
413 tr("USB device '%s' with UUID {%RTuuid} is not in the right state for capturing (%s)"),
414 mName, mId.raw(), i_getStateName());
415 }
416
417 AssertReturn( mUniState == kHostUSBDeviceState_HeldByProxy
418 || mUniState == kHostUSBDeviceState_Unused
419 || mUniState == kHostUSBDeviceState_Capturable,
420 E_UNEXPECTED);
421 Assert(mMachine.isNull());
422
423 /*
424 * If it's already held by the proxy, we'll simply call
425 * attachToVM synchronously.
426 */
427 if (mUniState == kHostUSBDeviceState_HeldByProxy)
428 {
429 alock.release();
430 HRESULT hrc = i_attachToVM(aMachine, aCaptureFilename, aMaskedIfs);
431 return SUCCEEDED(hrc);
432 }
433
434 /*
435 * Need to capture the device before it can be used.
436 *
437 * The device will be attached to the VM by the USB proxy service thread
438 * when the request succeeds (i.e. asynchronously).
439 */
440 LogFlowThisFunc(("{%s} capturing the device.\n", mName));
441#if defined(RT_OS_DARWIN) || defined(RT_OS_WINDOWS) || defined(RT_OS_SOLARIS) /* PORTME */
442 i_setState(kHostUSBDeviceState_Capturing, kHostUSBDeviceState_UsedByVM, kHostUSBDeviceSubState_AwaitingDetach);
443#else
444 i_setState(kHostUSBDeviceState_Capturing, kHostUSBDeviceState_UsedByVM);
445#endif
446 mMachine = aMachine;
447 mMaskedIfs = aMaskedIfs;
448 mCaptureFilename = aCaptureFilename;
449 alock.release();
450 int rc = mUSBProxyService->captureDevice(this);
451 if (RT_FAILURE(rc))
452 {
453 alock.acquire();
454 i_failTransition(kHostUSBDeviceState_Invalid);
455 mMachine.setNull();
456 if (rc == VERR_SHARING_VIOLATION)
457 return setError(E_FAIL,
458 tr("USB device '%s' with UUID {%RTuuid} is in use by someone else"),
459 mName, mId.raw());
460 return E_FAIL;
461 }
462
463 return S_OK;
464}
465
466/**
467 * Attempts to attach the USB device to a VM.
468 *
469 * The device must be in the HeldByProxy state or about to exit the
470 * Capturing state.
471 *
472 * This method will make an IPC to the VM process and do the actual
473 * attaching. While in the IPC locks will be abandond.
474 *
475 * @returns Status indicating whether it was successfully attached or not.
476 * @retval S_OK on success.
477 * @retval E_UNEXPECTED if the device state doesn't permit for any attaching.
478 * @retval E_* as appropriate.
479 *
480 * @param aMachine Machine this device should be attach to.
481 * @param aMaskedIfs The interfaces to hide from the guest.
482 */
483HRESULT HostUSBDevice::i_attachToVM(SessionMachine *aMachine, const com::Utf8Str &aCaptureFilename,
484 ULONG aMaskedIfs /* = 0*/)
485{
486 AssertReturn(!isWriteLockOnCurrentThread(), E_FAIL);
487 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
488 /*
489 * Validate and update the state.
490 */
491 AssertReturn( mUniState == kHostUSBDeviceState_Capturing
492 || mUniState == kHostUSBDeviceState_HeldByProxy
493 || mUniState == kHostUSBDeviceState_AttachingToVM,
494 E_UNEXPECTED);
495 i_setState(kHostUSBDeviceState_AttachingToVM, kHostUSBDeviceState_UsedByVM);
496
497 /*
498 * The VM process will query the object, so grab a reference to ourselves and release the locks.
499 */
500 ComPtr<IUSBDevice> d = this;
501
502 /*
503 * Call the VM process (IPC) and request it to attach the device.
504 *
505 * There are many reasons for this to fail, so, as a consequence we don't
506 * assert the return code as it will crash the daemon and annoy the heck
507 * out of people.
508 */
509 LogFlowThisFunc(("{%s} Calling machine->onUSBDeviceAttach()...\n", mName));
510 alock.release();
511 HRESULT hrc = aMachine->i_onUSBDeviceAttach(d, NULL, aMaskedIfs, aCaptureFilename);
512 LogFlowThisFunc(("{%s} Done machine->onUSBDeviceAttach()=%08X\n", mName, hrc));
513
514 /*
515 * As we re-acquire the lock, we'll have to check if the device was
516 * physically detached while we were busy.
517 */
518 alock.acquire();
519
520 if (SUCCEEDED(hrc))
521 {
522 mMachine = aMachine;
523 if (!mIsPhysicallyDetached)
524 i_setState(kHostUSBDeviceState_UsedByVM);
525 else
526 {
527 alock.release();
528 i_detachFromVM(kHostUSBDeviceState_PhysDetached);
529 hrc = E_UNEXPECTED;
530 }
531 }
532 else
533 {
534 mMachine.setNull();
535 if (!mIsPhysicallyDetached)
536 {
537 i_setState(kHostUSBDeviceState_HeldByProxy);
538 if (hrc == E_UNEXPECTED)
539 hrc = E_FAIL; /* No confusion. */
540 }
541 else
542 {
543 alock.release();
544 i_onPhysicalDetachedInternal();
545 hrc = E_UNEXPECTED;
546 }
547 }
548 return hrc;
549}
550
551
552/**
553 * Detaches the device from the VM.
554 *
555 * This is used for a special scenario in attachToVM() and from
556 * onPhysicalDetachedInternal().
557 *
558 * @param aFinalState The final state (PhysDetached).
559 */
560void HostUSBDevice::i_detachFromVM(HostUSBDeviceState aFinalState)
561{
562 NOREF(aFinalState);
563
564 /*
565 * Assert preconditions.
566 */
567 Assert(aFinalState == kHostUSBDeviceState_PhysDetached);
568 AssertReturnVoid(!isWriteLockOnCurrentThread());
569 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
570 Assert( mUniState == kHostUSBDeviceState_AttachingToVM
571 || mUniState == kHostUSBDeviceState_UsedByVM);
572 Assert(!mMachine.isNull());
573
574 /*
575 * Change the state and abandon the locks. The VM may query
576 * data and we don't want to deadlock - the state protects us,
577 * so, it's not a bit issue here.
578 */
579 i_setState(kHostUSBDeviceState_PhysDetachingFromVM, kHostUSBDeviceState_PhysDetached);
580
581 /*
582 * Call the VM process (IPC) and request it to detach the device.
583 *
584 * There are many reasons for this to fail, so, as a consequence we don't
585 * assert the return code as it will crash the daemon and annoy the heck
586 * out of people.
587 */
588 alock.release();
589 LogFlowThisFunc(("{%s} Calling machine->onUSBDeviceDetach()...\n", mName));
590 HRESULT hrc = mMachine->i_onUSBDeviceDetach(mId.toUtf16().raw(), NULL);
591 LogFlowThisFunc(("{%s} Done machine->onUSBDeviceDetach()=%Rhrc\n", mName, hrc));
592 NOREF(hrc);
593
594 /*
595 * Re-acquire the locks and complete the transition.
596 */
597 alock.acquire();
598 i_advanceTransition();
599}
600
601/**
602 * Called when the VM process to inform us about the device being
603 * detached from it.
604 *
605 * This is NOT called when we detach the device via onUSBDeviceDetach.
606 *
607 *
608 * @param[in] aMachine The machine making the request.
609 * This must be the machine this device is currently attached to.
610 * @param[in] aDone When set to false, the VM just informs us that it's about
611 * to detach this device but hasn't done it just yet.
612 * When set to true, the VM informs us that it has completed
613 * the detaching of this device.
614 * @param[out] aRunFilters Whether to run filters.
615 * @param[in] aAbnormal Set if we're cleaning up after a crashed VM.
616 *
617 * @returns S_OK on success, and E_UNEXPECTED if the device isn't in the right state.
618 *
619 * @note Must be called from under the object write lock.
620 */
621HRESULT HostUSBDevice::i_onDetachFromVM(SessionMachine *aMachine, bool aDone, bool *aRunFilters, bool aAbnormal /*= true*/)
622{
623 LogFlowThisFunc(("{%s} state=%s aDone=%RTbool aAbnormal=%RTbool\n", mName, i_getStateName(), aDone, aAbnormal));
624
625 /*
626 * Validate preconditions.
627 */
628 AssertPtrReturn(aRunFilters, E_INVALIDARG);
629 AssertReturn(isWriteLockOnCurrentThread(), E_FAIL);
630 if (!aDone)
631 {
632 if (mUniState != kHostUSBDeviceState_UsedByVM)
633 return setError(E_INVALIDARG,
634 tr("USB device '%s' with UUID {%RTuuid} is busy (state '%s'). Please try again later"),
635 mName, mId.raw(), i_getStateName());
636 }
637 else
638 AssertMsgReturn( mUniState == kHostUSBDeviceState_DetachingFromVM /** @todo capturing for VM
639 ends up here on termination. */
640 || (mUniState == kHostUSBDeviceState_UsedByVM && aAbnormal),
641 ("{%s} %s\n", mName, i_getStateName()), E_UNEXPECTED);
642 AssertMsgReturn((mMachine == aMachine), ("%p != %p\n", (void *)mMachine, aMachine), E_FAIL);
643
644 /*
645 * Change the state.
646 */
647 if (!aDone)
648 {
649 *aRunFilters = i_startTransition(kHostUSBDeviceState_DetachingFromVM, kHostUSBDeviceState_HeldByProxy);
650 /* PORTME: This might require host specific changes if you re-enumerate the device. */
651 }
652 else if (aAbnormal && mUniState == kHostUSBDeviceState_UsedByVM)
653 {
654 /* Fast forward thru the DetachingFromVM state and on to HeldByProxy. */
655 /** @todo need to update the state machine to handle crashed VMs. */
656 i_startTransition(kHostUSBDeviceState_DetachingFromVM, kHostUSBDeviceState_HeldByProxy);
657 *aRunFilters = i_advanceTransition();
658 mMachine.setNull();
659 /* PORTME: ditto / trouble if you depend on the VM process to do anything. */
660 }
661 else
662 {
663 /* normal completion. */
664 Assert(mUniSubState == kHostUSBDeviceSubState_Default); /* PORTME: ditto */
665 *aRunFilters = i_advanceTransition();
666 mMachine.setNull();
667 }
668
669 return S_OK;
670}
671
672/**
673 * Requests the USB proxy service to release the device back to the host.
674 *
675 * This method will ignore (not assert) calls for devices that already
676 * belong to the host because it simplifies the usage a bit.
677 *
678 * @returns COM status code.
679 * @retval S_OK on success.
680 * @retval E_UNEXPECTED on bad state.
681 * @retval E_* as appropriate.
682 *
683 * @note Must be called without holding the object lock.
684 */
685HRESULT HostUSBDevice::i_requestReleaseToHost()
686{
687 /*
688 * Validate preconditions.
689 */
690 AssertReturn(!isWriteLockOnCurrentThread(), E_FAIL);
691 Assert(mMachine.isNull());
692
693 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
694 LogFlowThisFunc(("{%s}\n", mName));
695 if ( mUniState == kHostUSBDeviceState_Unused
696 || mUniState == kHostUSBDeviceState_Capturable)
697 return S_OK;
698 AssertMsgReturn(mUniState == kHostUSBDeviceState_HeldByProxy, ("{%s} %s\n", mName, i_getStateName()), E_UNEXPECTED);
699
700 /*
701 * Try release it.
702 */
703#if defined(RT_OS_DARWIN) || defined(RT_OS_WINDOWS) /* PORTME */
704 i_startTransition(kHostUSBDeviceState_ReleasingToHost, kHostUSBDeviceState_Unused, kHostUSBDeviceSubState_AwaitingDetach);
705#else
706 i_startTransition(kHostUSBDeviceState_ReleasingToHost, kHostUSBDeviceState_Unused);
707#endif
708 alock.release();
709 int rc = mUSBProxyService->releaseDevice(this);
710 if (RT_FAILURE(rc))
711 {
712 alock.acquire();
713 i_failTransition(kHostUSBDeviceState_Invalid);
714 return E_FAIL;
715 }
716 return S_OK;
717}
718
719/**
720 * Requests the USB proxy service to capture and hold the device.
721 *
722 * The device must be owned by the host at the time of the call. But for
723 * the callers convenience, calling this method on a device that is already
724 * being held will success without any assertions.
725 *
726 * @returns COM status code.
727 * @retval S_OK on success.
728 * @retval E_UNEXPECTED on bad state.
729 * @retval E_* as appropriate.
730 *
731 * @note Must be called without holding the object lock.
732 */
733HRESULT HostUSBDevice::i_requestHold()
734{
735 /*
736 * Validate preconditions.
737 */
738 AssertReturn(!isWriteLockOnCurrentThread(), E_FAIL);
739 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
740 LogFlowThisFunc(("{%s}\n", mName));
741 AssertMsgReturn( mUniState == kHostUSBDeviceState_Unused
742 || mUniState == kHostUSBDeviceState_Capturable
743 || mUniState == kHostUSBDeviceState_HeldByProxy,
744 ("{%s} %s\n", mName, i_getStateName()),
745 E_UNEXPECTED);
746
747 Assert(mMachine.isNull());
748 mMachine.setNull();
749
750 if (mUniState == kHostUSBDeviceState_HeldByProxy)
751 return S_OK;
752
753 /*
754 * Do the job.
755 */
756#if defined(RT_OS_DARWIN) || defined(RT_OS_WINDOWS) /* PORTME */
757 i_startTransition(kHostUSBDeviceState_Capturing, kHostUSBDeviceState_HeldByProxy, kHostUSBDeviceSubState_AwaitingDetach);
758#else
759 i_startTransition(kHostUSBDeviceState_Capturing, kHostUSBDeviceState_HeldByProxy);
760#endif
761 alock.release();
762 int rc = mUSBProxyService->captureDevice(this);
763 if (RT_FAILURE(rc))
764 {
765 alock.acquire();
766 i_failTransition(kHostUSBDeviceState_Invalid);
767 return E_FAIL;
768 }
769 return S_OK;
770}
771
772
773/**
774 * Check a detach detected by the USB Proxy Service to see if
775 * it's a real one or just a logical following a re-enumeration.
776 *
777 * This will work the internal sub state of the device and do time
778 * outs, so it does more than just querying data!
779 *
780 * @returns true if it was actually detached, false if it's just a re-enumeration.
781 */
782bool HostUSBDevice::i_wasActuallyDetached()
783{
784 /*
785 * This only applies to the detach and re-attach states.
786 */
787 switch (mUniState)
788 {
789 case kHostUSBDeviceState_Capturing:
790 case kHostUSBDeviceState_ReleasingToHost:
791 case kHostUSBDeviceState_AttachingToVM:
792 case kHostUSBDeviceState_DetachingFromVM:
793 switch (mUniSubState)
794 {
795 /*
796 * If we're awaiting a detach, the this has now occurred
797 * and the state should be advanced.
798 */
799 case kHostUSBDeviceSubState_AwaitingDetach:
800 i_advanceTransition();
801 return false; /* not physically detached. */
802
803 /*
804 * Check for timeouts.
805 */
806 case kHostUSBDeviceSubState_AwaitingReAttach:
807 {
808#ifndef RT_OS_WINDOWS /* check the implementation details here. */
809 uint64_t elapsedNanoseconds = RTTimeNanoTS() - mLastStateChangeTS;
810 if (elapsedNanoseconds > UINT64_C(60000000000)) /* 60 seconds */
811 {
812 LogRel(("USB: Async operation timed out for device %s (state: %s)\n", mName, i_getStateName()));
813 i_failTransition(kHostUSBDeviceState_PhysDetached);
814 }
815#endif
816 return false; /* not physically detached. */
817 }
818
819 /* not applicable.*/
820 case kHostUSBDeviceSubState_Default:
821 break;
822 }
823 break;
824
825 /* not applicable. */
826 case kHostUSBDeviceState_Unsupported:
827 case kHostUSBDeviceState_UsedByHost:
828 case kHostUSBDeviceState_Capturable:
829 case kHostUSBDeviceState_Unused:
830 case kHostUSBDeviceState_HeldByProxy:
831 case kHostUSBDeviceState_UsedByVM:
832 case kHostUSBDeviceState_PhysDetachingFromVM:
833 case kHostUSBDeviceState_PhysDetached:
834 break;
835
836 default:
837 AssertLogRelMsgFailed(("this=%p %s\n", this, i_getStateName()));
838 break;
839 }
840
841 /* It was detached. */
842 return true;
843}
844
845/**
846 * Notification from the USB Proxy that the device was physically detached.
847 *
848 * If a transition is pending, mIsPhysicallyDetached will be set and
849 * handled when the transition advances forward.
850 *
851 * Otherwise the device will be detached from any VM currently using it - this
852 * involves IPC and will temporarily abandon locks - and all the device data
853 * reset.
854 */
855void HostUSBDevice::i_onPhysicalDetached()
856{
857 AssertReturnVoid(!isWriteLockOnCurrentThread());
858 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
859 LogFlowThisFunc(("{%s}\n", mName));
860
861 mIsPhysicallyDetached = true;
862 if (mUniState < kHostUSBDeviceState_FirstTransitional)
863 {
864 alock.release();
865 i_onPhysicalDetachedInternal();
866 }
867}
868
869
870/**
871 * Do the physical detach work for a device in a stable state or
872 * at a transition state change.
873 *
874 * See onPhysicalDetach() for details.
875 */
876void HostUSBDevice::i_onPhysicalDetachedInternal()
877{
878 AssertReturnVoid(!isWriteLockOnCurrentThread());
879 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
880 LogFlowThisFunc(("{%s}\n", mName));
881 Assert(mIsPhysicallyDetached);
882
883 /*
884 * Do we need to detach it from the VM first?
885 */
886 if ( !mMachine.isNull()
887 && ( mUniState == kHostUSBDeviceState_UsedByVM
888 || mUniState == kHostUSBDeviceState_AttachingToVM))
889 {
890 alock.release();
891 i_detachFromVM(kHostUSBDeviceState_PhysDetached);
892 alock.acquire();
893 }
894 else
895 AssertMsg(mMachine.isNull(), ("%s\n", i_getStateName()));
896
897 /*
898 * Reset the data and enter the final state.
899 */
900 mMachine.setNull();
901 i_setState(kHostUSBDeviceState_PhysDetached);
902}
903
904
905/**
906 * Returns true if this device matches the given filter data.
907 *
908 * @note It is assumed, that the filter data owner is appropriately
909 * locked before calling this method.
910 *
911 * @note
912 * This method MUST correlate with
913 * USBController::hasMatchingFilter (IUSBDevice *)
914 * in the sense of the device matching logic.
915 *
916 * @note Locks this object for reading.
917 */
918bool HostUSBDevice::i_isMatch(const USBDeviceFilter::Data &aData)
919{
920 AutoCaller autoCaller(this);
921 AssertComRCReturn(autoCaller.rc(), false);
922
923 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
924
925 if (!aData.mActive)
926 return false;
927
928 if (!aData.mRemote.isMatch(FALSE))
929 return false;
930
931 if (!USBFilterMatchDevice(&aData.mUSBFilter, mUsb))
932 return false;
933
934 /* Don't match busy devices with a 100% wildcard filter - this will
935 later become a filter prop (ring-3 only). */
936 if ( mUsb->enmState == USBDEVICESTATE_USED_BY_HOST_CAPTURABLE
937 && !USBFilterHasAnySubstatialCriteria(&aData.mUSBFilter))
938 return false;
939
940 LogFlowThisFunc(("returns true\n"));
941 return true;
942}
943
944/**
945 * Compares this device with a USBDEVICE and decides if the match or which comes first.
946 *
947 * This will take into account device re-attaching and omit the bits
948 * that may change during a device re-enumeration.
949 *
950 * @param aDev2 Device 2.
951 *
952 * @returns < 0 if this should come before aDev2.
953 * @returns 0 if this and aDev2 are equal.
954 * @returns > 0 if this should come after aDev2.
955 *
956 * @note Must be called from under the object write lock.
957 */
958int HostUSBDevice::i_compare(PCUSBDEVICE aDev2)
959{
960 AssertReturn(isWriteLockOnCurrentThread(), -1);
961 //Log3(("%Rfn: %p {%s}\n", __PRETTY_FUNCTION__, this, mName));
962 return i_compare(mUsb, aDev2,
963 mUniSubState == kHostUSBDeviceSubState_AwaitingDetach /* (In case we don't get the detach notice.) */
964 || mUniSubState == kHostUSBDeviceSubState_AwaitingReAttach);
965}
966
967/**
968 * Compares two USBDEVICE structures and decides if the match or which comes first.
969 *
970 * @param aDev1 Device 1.
971 * @param aDev2 Device 2.
972 * @param aIsAwaitingReAttach Whether to omit bits that will change in a device
973 * re-enumeration (true) or not (false).
974 *
975 * @returns < 0 if aDev1 should come before aDev2.
976 * @returns 0 if aDev1 and aDev2 are equal.
977 * @returns > 0 if aDev1 should come after aDev2.
978 */
979/*static*/
980int HostUSBDevice::i_compare(PCUSBDEVICE aDev1, PCUSBDEVICE aDev2, bool aIsAwaitingReAttach /*= false */)
981{
982 /*
983 * Things that stays the same everywhere.
984 *
985 * The more uniquely these properties identifies a device the less the chance
986 * that we mix similar devices during re-enumeration. Bus+port would help
987 * provide ~99.8% accuracy if the host can provide those attributes.
988 */
989 int iDiff = aDev1->idVendor - aDev2->idVendor;
990 if (iDiff)
991 return iDiff;
992
993 iDiff = aDev1->idProduct - aDev2->idProduct;
994 if (iDiff)
995 return iDiff;
996
997 iDiff = aDev1->bcdDevice - aDev2->bcdDevice;
998 if (iDiff)
999 {
1000 //Log3(("compare: bcdDevice: %#x != %#x\n", aDev1->bcdDevice, aDev2->bcdDevice));
1001 return iDiff;
1002 }
1003
1004#ifdef RT_OS_WINDOWS /* the string query may fail on windows during replugging, ignore serial mismatch if this is the case. */
1005 if ( aDev1->u64SerialHash != aDev2->u64SerialHash
1006 && ( !aIsAwaitingReAttach
1007 || (aDev2->pszSerialNumber && *aDev2->pszSerialNumber)
1008 || (aDev2->pszManufacturer && *aDev2->pszManufacturer)
1009 || (aDev2->pszProduct && *aDev2->pszProduct))
1010 )
1011#else
1012 if (aDev1->u64SerialHash != aDev2->u64SerialHash)
1013#endif
1014 {
1015 //Log3(("compare: u64SerialHash: %#llx != %#llx\n", aDev1->u64SerialHash, aDev2->u64SerialHash));
1016 return aDev1->u64SerialHash < aDev2->u64SerialHash ? -1 : 1;
1017 }
1018
1019 /* The hub/bus + port should help a lot in a re-attach situation. */
1020#ifdef RT_OS_WINDOWS
1021 iDiff = strcmp(aDev1->pszHubName, aDev2->pszHubName);
1022 if (iDiff)
1023 {
1024 //Log3(("compare: HubName: %s != %s\n", aDev1->pszHubName, aDev2->pszHubName));
1025 return iDiff;
1026 }
1027#else
1028 iDiff = aDev1->bBus - aDev2->bBus;
1029 if (iDiff)
1030 {
1031 //Log3(("compare: bBus: %#x != %#x\n", aDev1->bBus, aDev2->bBus));
1032 return iDiff;
1033 }
1034#endif
1035
1036 iDiff = aDev1->bPort - aDev2->bPort; /* shouldn't change anywhere and help pinpoint it very accurately. */
1037 if (iDiff)
1038 {
1039 //Log3(("compare: bPort: %#x != %#x\n", aDev1->bPort, aDev2->bPort));
1040 return iDiff;
1041 }
1042
1043 /*
1044 * Things that usually doesn't stay the same when re-enumerating
1045 * a device. The fewer things in the category the better chance
1046 * that we avoid messing up when more than one device of the same
1047 * kind is attached.
1048 */
1049 if (aIsAwaitingReAttach)
1050 {
1051 //Log3(("aDev1=%p == aDev2=%p\n", aDev1, aDev2));
1052 return 0;
1053 }
1054 /* device number always changes. */
1055 return strcmp(aDev1->pszAddress, aDev2->pszAddress);
1056}
1057
1058/**
1059 * Updates the state of the device.
1060 *
1061 * If this method returns @c true, Host::onUSBDeviceStateChanged() will be
1062 * called to process the state change (complete the state change request,
1063 * inform the VM process etc.).
1064 *
1065 * If this method returns @c false, it is assumed that the given state change
1066 * is "minor": it doesn't require any further action other than update the
1067 * mState field with the actual state value.
1068 *
1069 * Regardless of the return value, this method always takes ownership of the
1070 * new USBDEVICE structure passed in and updates the pNext and pPrev fiends in
1071 * it using the values of the old structure.
1072 *
1073 * @param[in] aDev The current device state as seen by the proxy backend.
1074 * @param[out] aRunFilters Whether the state change should be accompanied by
1075 * running filters on the device.
1076 * @param[out] aIgnoreMachine Machine to ignore when running filters.
1077 *
1078 * @returns Whether the Host object should be bothered with this state change.
1079 *
1080 * @todo Just do everything here, that is, call filter runners and everything that
1081 * works by state change. Using 3 return codes/parameters is just plain ugly.
1082 */
1083bool HostUSBDevice::i_updateState(PCUSBDEVICE aDev, bool *aRunFilters, SessionMachine **aIgnoreMachine)
1084{
1085 *aRunFilters = false;
1086 *aIgnoreMachine = NULL;
1087
1088 /*
1089 * Locking.
1090 */
1091 AssertReturn(!isWriteLockOnCurrentThread(), false);
1092 AutoCaller autoCaller(this);
1093 AssertComRCReturn(autoCaller.rc(), false);
1094 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1095
1096 /*
1097 * Replace the existing structure by the new one.
1098 */
1099 const USBDEVICESTATE enmOldState = mUsb->enmState; NOREF(enmOldState);
1100 if (mUsb != aDev)
1101 {
1102#ifdef RT_OS_WINDOWS
1103 /* we used this logic of string comparison in HostUSBDevice::compare
1104 * now we need to preserve strings from the old device if the new device has zero strings
1105 * this ensures the device is correctly matched later on
1106 * otherwise we may end up with a phantom misconfigured device instance */
1107 if ((mUniSubState == kHostUSBDeviceSubState_AwaitingDetach /* (In case we don't get the detach notice.) */
1108 || mUniSubState == kHostUSBDeviceSubState_AwaitingReAttach)
1109 && (!aDev->pszSerialNumber || !*aDev->pszSerialNumber)
1110 && (!aDev->pszManufacturer || !*aDev->pszManufacturer)
1111 && (!aDev->pszProduct || !*aDev->pszProduct))
1112 {
1113 aDev->u64SerialHash = mUsb->u64SerialHash;
1114
1115 if (mUsb->pszSerialNumber && *mUsb->pszSerialNumber)
1116 {
1117 if (aDev->pszSerialNumber)
1118 RTStrFree((char *)aDev->pszSerialNumber);
1119
1120 /* since we're going to free old device later on,
1121 * we can just assign the string from it to the new device
1122 * and zero up the string filed for the old device */
1123 aDev->pszSerialNumber = mUsb->pszSerialNumber;
1124 mUsb->pszSerialNumber = NULL;
1125 }
1126
1127 if (mUsb->pszManufacturer && *mUsb->pszManufacturer)
1128 {
1129 if (aDev->pszManufacturer)
1130 RTStrFree((char *)aDev->pszManufacturer);
1131
1132 /* since we're going to free old device later on,
1133 * we can just assign the string from it to the new device
1134 * and zero up the string filed for the old device */
1135 aDev->pszManufacturer = mUsb->pszManufacturer;
1136 mUsb->pszManufacturer = NULL;
1137 }
1138
1139 if (mUsb->pszProduct && *mUsb->pszProduct)
1140 {
1141 if (aDev->pszProduct)
1142 RTStrFree((char *)aDev->pszProduct);
1143
1144 /* since we're going to free old device later on,
1145 * we can just assign the string from it to the new device
1146 * and zero up the string filed for the old device */
1147 aDev->pszProduct = mUsb->pszProduct;
1148 mUsb->pszProduct = NULL;
1149 }
1150 }
1151#endif
1152 aDev->pNext = mUsb->pNext;
1153 aDev->pPrev = mUsb->pPrev;
1154 USBProxyService::freeDevice(mUsb);
1155 mUsb = aDev;
1156 }
1157
1158/** @def HOSTUSBDEVICE_FUZZY_STATE
1159 * Defined on hosts where we have a driver that keeps proper device states.
1160 */
1161# if defined(RT_OS_LINUX) || defined(DOXYGEN_RUNNING)
1162# define HOSTUSBDEVICE_FUZZY_STATE 1
1163# else
1164# undef HOSTUSBDEVICE_FUZZY_STATE
1165# endif
1166 /*
1167 * For some hosts we'll have to be pretty careful here because
1168 * they don't always have a clue what is going on. This is
1169 * particularly true on linux and solaris, while windows and
1170 * darwin generally knows a bit more.
1171 */
1172 bool fIsImportant = false;
1173 if (enmOldState != mUsb->enmState)
1174 {
1175 LogFlowThisFunc(("%p {%s} %s\n", this, mName, i_getStateName()));
1176 switch (mUsb->enmState)
1177 {
1178 /*
1179 * Little fuzziness here, except where we fake capture.
1180 */
1181 case USBDEVICESTATE_USED_BY_HOST:
1182 switch (mUniState)
1183 {
1184 /* Host drivers installed, that's fine. */
1185 case kHostUSBDeviceState_Capturable:
1186 case kHostUSBDeviceState_Unused:
1187 LogThisFunc(("{%s} %s -> %s\n", mName, i_getStateName(), i_stateName(kHostUSBDeviceState_UsedByHost)));
1188 *aRunFilters = i_setState(kHostUSBDeviceState_UsedByHost);
1189 break;
1190 case kHostUSBDeviceState_UsedByHost:
1191 break;
1192
1193 /* Can only mean that we've failed capturing it. */
1194 case kHostUSBDeviceState_Capturing:
1195 LogThisFunc(("{%s} capture failed! (#1)\n", mName));
1196 mUSBProxyService->captureDeviceCompleted(this, false /* aSuccess */);
1197 *aRunFilters = i_failTransition(kHostUSBDeviceState_UsedByHost);
1198 mMachine.setNull();
1199 break;
1200
1201 /* Guess we've successfully released it. */
1202 case kHostUSBDeviceState_ReleasingToHost:
1203 LogThisFunc(("{%s} %s -> %s\n", mName, i_getStateName(), i_stateName(kHostUSBDeviceState_UsedByHost)));
1204 mUSBProxyService->releaseDeviceCompleted(this, true /* aSuccess */);
1205 *aRunFilters = i_setState(kHostUSBDeviceState_UsedByHost);
1206 break;
1207
1208 /* These are IPC states and should be left alone. */
1209 case kHostUSBDeviceState_AttachingToVM:
1210 case kHostUSBDeviceState_DetachingFromVM:
1211 case kHostUSBDeviceState_PhysDetachingFromVM:
1212 LogThisFunc(("{%s} %s - changed to USED_BY_HOST...\n", mName, i_getStateName()));
1213 break;
1214
1215#ifdef HOSTUSBDEVICE_FUZZY_STATE
1216 /* Fake: We can't prevent anyone from grabbing it. */
1217 case kHostUSBDeviceState_HeldByProxy:
1218 LogThisFunc(("{%s} %s -> %s!\n", mName, i_getStateName(), i_stateName(kHostUSBDeviceState_UsedByHost)));
1219 *aRunFilters = i_setState(kHostUSBDeviceState_UsedByHost);
1220 break;
1221 //case kHostUSBDeviceState_UsedByVM:
1222 // /** @todo needs to be detached from the VM. */
1223 // break;
1224#endif
1225 /* Not supposed to happen... */
1226#ifndef HOSTUSBDEVICE_FUZZY_STATE
1227 case kHostUSBDeviceState_HeldByProxy:
1228#endif
1229 case kHostUSBDeviceState_UsedByVM:
1230 case kHostUSBDeviceState_PhysDetached:
1231 case kHostUSBDeviceState_Unsupported:
1232 default:
1233 AssertMsgFailed(("{%s} %s\n", mName, i_getStateName()));
1234 break;
1235 }
1236 break;
1237
1238 /*
1239 * It changed to capturable. Fuzzy hosts might easily
1240 * confuse UsedByVM with this one.
1241 */
1242 case USBDEVICESTATE_USED_BY_HOST_CAPTURABLE:
1243 switch (mUniState)
1244 {
1245 /* No change. */
1246#ifdef HOSTUSBDEVICE_FUZZY_STATE
1247 case kHostUSBDeviceState_HeldByProxy:
1248 case kHostUSBDeviceState_UsedByVM:
1249#endif
1250 case kHostUSBDeviceState_Capturable:
1251 break;
1252
1253 /* Changed! */
1254 case kHostUSBDeviceState_UsedByHost:
1255 fIsImportant = true;
1256 case kHostUSBDeviceState_Unused:
1257 LogThisFunc(("{%s} %s -> %s\n", mName, i_getStateName(), i_stateName(kHostUSBDeviceState_Capturable)));
1258 *aRunFilters = i_setState(kHostUSBDeviceState_Capturable);
1259 break;
1260
1261 /* Can only mean that we've failed capturing it. */
1262 case kHostUSBDeviceState_Capturing:
1263 LogThisFunc(("{%s} capture failed! (#2)\n", mName));
1264 mUSBProxyService->captureDeviceCompleted(this, false /* aSuccess */);
1265 *aRunFilters = i_failTransition(kHostUSBDeviceState_Capturable);
1266 mMachine.setNull();
1267 break;
1268
1269 /* Guess we've successfully released it. */
1270 case kHostUSBDeviceState_ReleasingToHost:
1271 LogThisFunc(("{%s} %s -> %s\n", mName, i_getStateName(), i_stateName(kHostUSBDeviceState_Capturable)));
1272 mUSBProxyService->releaseDeviceCompleted(this, true /* aSuccess */);
1273 *aRunFilters = i_setState(kHostUSBDeviceState_Capturable);
1274 break;
1275
1276 /* These are IPC states and should be left alone. */
1277 case kHostUSBDeviceState_AttachingToVM:
1278 case kHostUSBDeviceState_DetachingFromVM:
1279 case kHostUSBDeviceState_PhysDetachingFromVM:
1280 LogThisFunc(("{%s} %s - changed to USED_BY_HOST_CAPTURABLE...\n", mName, i_getStateName()));
1281 break;
1282
1283 /* Not supposed to happen*/
1284#ifndef HOSTUSBDEVICE_FUZZY_STATE
1285 case kHostUSBDeviceState_HeldByProxy:
1286 case kHostUSBDeviceState_UsedByVM:
1287#endif
1288 case kHostUSBDeviceState_Unsupported:
1289 case kHostUSBDeviceState_PhysDetached:
1290 default:
1291 AssertMsgFailed(("{%s} %s\n", mName, i_getStateName()));
1292 break;
1293 }
1294 break;
1295
1296
1297 /*
1298 * It changed to capturable. Fuzzy hosts might easily
1299 * confuse UsedByVM and HeldByProxy with this one.
1300 */
1301 case USBDEVICESTATE_UNUSED:
1302 switch (mUniState)
1303 {
1304 /* No change. */
1305#ifdef HOSTUSBDEVICE_FUZZY_STATE
1306 case kHostUSBDeviceState_HeldByProxy:
1307 case kHostUSBDeviceState_UsedByVM:
1308#endif
1309 case kHostUSBDeviceState_Unused:
1310 break;
1311
1312 /* Changed! */
1313 case kHostUSBDeviceState_UsedByHost:
1314 case kHostUSBDeviceState_Capturable:
1315 fIsImportant = true;
1316 LogThisFunc(("{%s} %s -> %s\n", mName, i_getStateName(), i_stateName(kHostUSBDeviceState_Unused)));
1317 *aRunFilters = i_setState(kHostUSBDeviceState_Unused);
1318 break;
1319
1320 /* Can mean that we've failed capturing it, but on windows it is the detach signal. */
1321 case kHostUSBDeviceState_Capturing:
1322#if defined(RT_OS_WINDOWS)
1323 if (mUniSubState == kHostUSBDeviceSubState_AwaitingDetach)
1324 {
1325 LogThisFunc(("{%s} capture advancing thru UNUSED...\n", mName));
1326 *aRunFilters = i_advanceTransition();
1327 }
1328 else
1329#endif
1330 {
1331 LogThisFunc(("{%s} capture failed! (#3)\n", mName));
1332 mUSBProxyService->captureDeviceCompleted(this, false /* aSuccess */);
1333 *aRunFilters = i_failTransition(kHostUSBDeviceState_Unused);
1334 mMachine.setNull();
1335 }
1336 break;
1337
1338 /* Guess we've successfully released it. */
1339 case kHostUSBDeviceState_ReleasingToHost:
1340 LogThisFunc(("{%s} %s -> %s\n", mName, i_getStateName(), i_stateName(kHostUSBDeviceState_Unused)));
1341 mUSBProxyService->releaseDeviceCompleted(this, true /* aSuccess */);
1342 *aRunFilters = i_setState(kHostUSBDeviceState_Unused);
1343 break;
1344
1345 /* These are IPC states and should be left alone. */
1346 case kHostUSBDeviceState_AttachingToVM:
1347 case kHostUSBDeviceState_DetachingFromVM:
1348 case kHostUSBDeviceState_PhysDetachingFromVM:
1349 LogThisFunc(("{%s} %s - changed to UNUSED...\n", mName, i_getStateName()));
1350 break;
1351
1352 /* Not supposed to happen*/
1353#ifndef HOSTUSBDEVICE_FUZZY_STATE
1354 case kHostUSBDeviceState_HeldByProxy:
1355 case kHostUSBDeviceState_UsedByVM:
1356#endif
1357 case kHostUSBDeviceState_Unsupported:
1358 case kHostUSBDeviceState_PhysDetached:
1359 default:
1360 AssertMsgFailed(("{%s} %s\n", mName, i_getStateName()));
1361 break;
1362 }
1363 break;
1364
1365 /*
1366 * This is pretty straight forward, except that everyone
1367 * might sometimes confuse this and the UsedByVM state.
1368 */
1369 case USBDEVICESTATE_HELD_BY_PROXY:
1370 switch (mUniState)
1371 {
1372 /* No change. */
1373 case kHostUSBDeviceState_HeldByProxy:
1374 break;
1375 case kHostUSBDeviceState_UsedByVM:
1376 LogThisFunc(("{%s} %s - changed to HELD_BY_PROXY...\n", mName, i_getStateName()));
1377 break;
1378
1379 /* Guess we've successfully captured it. */
1380 case kHostUSBDeviceState_Capturing:
1381 LogThisFunc(("{%s} capture succeeded!\n", mName));
1382 mUSBProxyService->captureDeviceCompleted(this, true /* aSuccess */);
1383 *aRunFilters = i_advanceTransition(true /* fast forward thru re-attach */);
1384
1385 /* Take action if we're supposed to attach it to a VM. */
1386 if (mUniState == kHostUSBDeviceState_AttachingToVM)
1387 {
1388 alock.release();
1389 i_attachToVM(mMachine, mCaptureFilename, mMaskedIfs);
1390 alock.acquire();
1391 }
1392 break;
1393
1394 /* Can only mean that we've failed capturing it. */
1395 case kHostUSBDeviceState_ReleasingToHost:
1396 LogThisFunc(("{%s} %s failed!\n", mName, i_getStateName()));
1397 mUSBProxyService->releaseDeviceCompleted(this, false /* aSuccess */);
1398 *aRunFilters = i_setState(kHostUSBDeviceState_HeldByProxy);
1399 break;
1400
1401 /* These are IPC states and should be left alone. */
1402 case kHostUSBDeviceState_AttachingToVM:
1403 case kHostUSBDeviceState_DetachingFromVM:
1404 case kHostUSBDeviceState_PhysDetachingFromVM:
1405 LogThisFunc(("{%s} %s - changed to HELD_BY_PROXY...\n", mName, i_getStateName()));
1406 break;
1407
1408 /* Not supposed to happen. */
1409 case kHostUSBDeviceState_Unsupported:
1410 case kHostUSBDeviceState_UsedByHost:
1411 case kHostUSBDeviceState_Capturable:
1412 case kHostUSBDeviceState_Unused:
1413 case kHostUSBDeviceState_PhysDetached:
1414 default:
1415 AssertMsgFailed(("{%s} %s\n", mName, i_getStateName()));
1416 break;
1417 }
1418 break;
1419
1420 /*
1421 * This is very straight forward and only Darwin implements it.
1422 */
1423 case USBDEVICESTATE_USED_BY_GUEST:
1424 switch (mUniState)
1425 {
1426 /* No change. */
1427 case kHostUSBDeviceState_HeldByProxy:
1428 LogThisFunc(("{%s} %s - changed to USED_BY_GUEST...\n", mName, i_getStateName()));
1429 break;
1430 case kHostUSBDeviceState_UsedByVM:
1431 break;
1432
1433 /* These are IPC states and should be left alone. */
1434 case kHostUSBDeviceState_AttachingToVM:
1435 case kHostUSBDeviceState_DetachingFromVM:
1436 case kHostUSBDeviceState_PhysDetachingFromVM:
1437 LogThisFunc(("{%s} %s - changed to USED_BY_GUEST...\n", mName, i_getStateName()));
1438 break;
1439
1440 /* Not supposed to happen. */
1441 case kHostUSBDeviceState_Unsupported:
1442 case kHostUSBDeviceState_Capturable:
1443 case kHostUSBDeviceState_Unused:
1444 case kHostUSBDeviceState_UsedByHost:
1445 case kHostUSBDeviceState_PhysDetached:
1446 case kHostUSBDeviceState_ReleasingToHost:
1447 case kHostUSBDeviceState_Capturing:
1448 default:
1449 AssertMsgFailed(("{%s} %s\n", mName, i_getStateName()));
1450 break;
1451 }
1452 break;
1453
1454 /*
1455 * This is not supposed to happen and indicates a bug in the backend!
1456 */
1457 case USBDEVICESTATE_UNSUPPORTED:
1458 AssertMsgFailed(("enmOldState=%d {%s} %s\n", enmOldState, mName, i_getStateName()));
1459 break;
1460 default:
1461 AssertMsgFailed(("enmState=%d {%s} %s\n", mUsb->enmState, mName, i_getStateName()));
1462 break;
1463 }
1464 }
1465 else if ( mUniSubState == kHostUSBDeviceSubState_AwaitingDetach
1466 && i_hasAsyncOperationTimedOut())
1467 {
1468 LogRel(("USB: timeout in %s for {%RTuuid} / {%s}\n", i_getStateName(), mId.raw(), mName));
1469 *aRunFilters = i_failTransition(kHostUSBDeviceState_Invalid);
1470 fIsImportant = true;
1471 }
1472 else
1473 {
1474 LogFlowThisFunc(("%p {%s} %s - no change %d\n", this, mName, i_getStateName(), enmOldState));
1475 /** @todo might have to handle some stuff here too if we cannot make the release/capture
1476 * handling deal with that above ... */
1477 }
1478
1479 return fIsImportant;
1480}
1481
1482
1483/**
1484 * Updates the state of the device, checking for cases which we fake.
1485 *
1486 * See HostUSBDevice::updateState() for details.
1487 *
1488 * @param[in] aDev See HostUSBDevice::updateState().
1489 * @param[out] aRunFilters See HostUSBDevice::updateState()
1490 * @param[out] aIgnoreMachine See HostUSBDevice::updateState()
1491 *
1492 * @returns See HostUSBDevice::updateState()
1493 */
1494bool HostUSBDevice::i_updateStateFake(PCUSBDEVICE aDev, bool *aRunFilters, SessionMachine **aIgnoreMachine)
1495{
1496 Assert(!isWriteLockOnCurrentThread());
1497 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1498 const HostUSBDeviceState enmState = mUniState;
1499 switch (enmState)
1500 {
1501 case kHostUSBDeviceState_Capturing:
1502 case kHostUSBDeviceState_ReleasingToHost:
1503 {
1504 *aIgnoreMachine = mUniState == kHostUSBDeviceState_ReleasingToHost ? mMachine : NULL;
1505 *aRunFilters = i_advanceTransition();
1506 LogThisFunc(("{%s} %s\n", mName, i_getStateName()));
1507
1508 if (mUsb != aDev)
1509 {
1510 aDev->pNext = mUsb->pNext;
1511 aDev->pPrev = mUsb->pPrev;
1512 USBProxyService::freeDevice(mUsb);
1513 mUsb = aDev;
1514 }
1515
1516 /* call the completion method */
1517 if (enmState == kHostUSBDeviceState_Capturing)
1518 mUSBProxyService->captureDeviceCompleted(this, true /* aSuccess */);
1519 else
1520 mUSBProxyService->releaseDeviceCompleted(this, true /* aSuccess */);
1521
1522 /* Take action if we're supposed to attach it to a VM. */
1523 if (mUniState == kHostUSBDeviceState_AttachingToVM)
1524 {
1525 alock.release();
1526 i_attachToVM(mMachine, mCaptureFilename, mMaskedIfs);
1527 }
1528 return true;
1529 }
1530
1531 default:
1532 alock.release();
1533 return i_updateState(aDev, aRunFilters, aIgnoreMachine);
1534 }
1535}
1536
1537
1538/**
1539 * Checks if there is a pending asynchronous operation and whether
1540 * it has timed out or not.
1541 *
1542 * @returns true on timeout, false if not.
1543 *
1544 * @note Caller must have read or write locked the object before calling.
1545 */
1546bool HostUSBDevice::i_hasAsyncOperationTimedOut() const
1547{
1548 switch (mUniSubState)
1549 {
1550#ifndef RT_OS_WINDOWS /* no timeouts on windows yet since I don't have all the details here... */
1551 case kHostUSBDeviceSubState_AwaitingDetach:
1552 case kHostUSBDeviceSubState_AwaitingReAttach:
1553 {
1554 uint64_t elapsedNanoseconds = RTTimeNanoTS() - mLastStateChangeTS;
1555 return elapsedNanoseconds > UINT64_C(60000000000); /* 60 seconds */ /* PORTME */
1556 }
1557#endif
1558 default:
1559 return false;
1560 }
1561}
1562
1563
1564/**
1565 * Translate the state into
1566 *
1567 * @returns
1568 * @param aState
1569 * @param aSubState
1570 * @param aPendingState
1571 */
1572/*static*/ const char *HostUSBDevice::i_stateName(HostUSBDeviceState aState,
1573 HostUSBDeviceState aPendingState /*= kHostUSBDeviceState_Invalid*/,
1574 HostUSBDeviceSubState aSubState /*= kHostUSBDeviceSubState_Default*/)
1575{
1576 switch (aState)
1577 {
1578 case kHostUSBDeviceState_Unsupported:
1579 AssertReturn(aPendingState == kHostUSBDeviceState_Invalid, "Unsupported{bad}");
1580 AssertReturn(aSubState == kHostUSBDeviceSubState_Default, "Unsupported[bad]");
1581 return "Unsupported";
1582
1583 case kHostUSBDeviceState_UsedByHost:
1584 AssertReturn(aPendingState == kHostUSBDeviceState_Invalid, "UsedByHost{bad}");
1585 AssertReturn(aSubState == kHostUSBDeviceSubState_Default, "UsedByHost[bad]");
1586 return "UsedByHost";
1587
1588 case kHostUSBDeviceState_Capturable:
1589 AssertReturn(aPendingState == kHostUSBDeviceState_Invalid, "Capturable{bad}");
1590 AssertReturn(aSubState == kHostUSBDeviceSubState_Default, "Capturable[bad]");
1591 return "Capturable";
1592
1593 case kHostUSBDeviceState_Unused:
1594 AssertReturn(aPendingState == kHostUSBDeviceState_Invalid, "Unused{bad}");
1595 AssertReturn(aSubState == kHostUSBDeviceSubState_Default, "Unused[bad]");
1596 return "Unused";
1597
1598 case kHostUSBDeviceState_HeldByProxy:
1599 AssertReturn(aPendingState == kHostUSBDeviceState_Invalid, "HeldByProxy{bad}");
1600 AssertReturn(aSubState == kHostUSBDeviceSubState_Default, "HeldByProxy[bad]");
1601 return "HeldByProxy";
1602
1603 case kHostUSBDeviceState_UsedByVM:
1604 AssertReturn(aPendingState == kHostUSBDeviceState_Invalid, "UsedByVM{bad}");
1605 AssertReturn(aSubState == kHostUSBDeviceSubState_Default, "UsedByVM[bad]");
1606 return "UsedByVM";
1607
1608 case kHostUSBDeviceState_PhysDetached:
1609 AssertReturn(aPendingState == kHostUSBDeviceState_Invalid, "PhysDetached{bad}");
1610 AssertReturn(aSubState == kHostUSBDeviceSubState_Default, "PhysDetached[bad]");
1611 return "PhysDetached";
1612
1613 case kHostUSBDeviceState_Capturing:
1614 switch (aPendingState)
1615 {
1616 case kHostUSBDeviceState_UsedByVM:
1617 switch (aSubState)
1618 {
1619 case kHostUSBDeviceSubState_Default:
1620 return "CapturingForVM";
1621 case kHostUSBDeviceSubState_AwaitingDetach:
1622 return "CapturingForVM[Detach]";
1623 case kHostUSBDeviceSubState_AwaitingReAttach:
1624 return "CapturingForVM[Attach]";
1625 default:
1626 AssertFailedReturn("CapturingForVM[bad]");
1627 }
1628 break;
1629
1630 case kHostUSBDeviceState_HeldByProxy:
1631 switch (aSubState)
1632 {
1633 case kHostUSBDeviceSubState_Default:
1634 return "CapturingForProxy";
1635 case kHostUSBDeviceSubState_AwaitingDetach:
1636 return "CapturingForProxy[Detach]";
1637 case kHostUSBDeviceSubState_AwaitingReAttach:
1638 return "CapturingForProxy[Attach]";
1639 default:
1640 AssertFailedReturn("CapturingForProxy[bad]");
1641 }
1642 break;
1643
1644 default:
1645 AssertFailedReturn("Capturing{bad}");
1646 }
1647 break;
1648
1649 case kHostUSBDeviceState_ReleasingToHost:
1650 switch (aPendingState)
1651 {
1652 case kHostUSBDeviceState_Unused:
1653 switch (aSubState)
1654 {
1655 case kHostUSBDeviceSubState_Default:
1656 return "ReleasingToHost";
1657 case kHostUSBDeviceSubState_AwaitingDetach:
1658 return "ReleasingToHost[Detach]";
1659 case kHostUSBDeviceSubState_AwaitingReAttach:
1660 return "ReleasingToHost[Attach]";
1661 default:
1662 AssertFailedReturn("ReleasingToHost[bad]");
1663 }
1664 break;
1665 default:
1666 AssertFailedReturn("ReleasingToHost{bad}");
1667 }
1668 break;
1669
1670 case kHostUSBDeviceState_DetachingFromVM:
1671 switch (aPendingState)
1672 {
1673 case kHostUSBDeviceState_HeldByProxy:
1674 switch (aSubState)
1675 {
1676 case kHostUSBDeviceSubState_Default:
1677 return "DetatchingFromVM>Proxy";
1678 case kHostUSBDeviceSubState_AwaitingDetach:
1679 return "DetatchingFromVM>Proxy[Detach]";
1680 case kHostUSBDeviceSubState_AwaitingReAttach:
1681 return "DetatchingFromVM>Proxy[Attach]";
1682 default:
1683 AssertFailedReturn("DetatchingFromVM>Proxy[bad]");
1684 }
1685 break;
1686
1687 case kHostUSBDeviceState_Unused:
1688 switch (aSubState)
1689 {
1690 case kHostUSBDeviceSubState_Default:
1691 return "DetachingFromVM>Host";
1692 case kHostUSBDeviceSubState_AwaitingDetach:
1693 return "DetachingFromVM>Host[Detach]";
1694 case kHostUSBDeviceSubState_AwaitingReAttach:
1695 return "DetachingFromVM>Host[Attach]";
1696 default:
1697 AssertFailedReturn("DetachingFromVM>Host[bad]");
1698 }
1699 break;
1700
1701 default:
1702 AssertFailedReturn("DetachingFromVM{bad}");
1703 }
1704 break;
1705
1706 case kHostUSBDeviceState_AttachingToVM:
1707 switch (aPendingState)
1708 {
1709 case kHostUSBDeviceState_UsedByVM:
1710 switch (aSubState)
1711 {
1712 case kHostUSBDeviceSubState_Default:
1713 return "AttachingToVM";
1714 case kHostUSBDeviceSubState_AwaitingDetach:
1715 return "AttachingToVM[Detach]";
1716 case kHostUSBDeviceSubState_AwaitingReAttach:
1717 return "AttachingToVM[Attach]";
1718 default:
1719 AssertFailedReturn("AttachingToVM[bad]");
1720 }
1721 break;
1722
1723 default:
1724 AssertFailedReturn("AttachingToVM{bad}");
1725 }
1726 break;
1727
1728
1729 case kHostUSBDeviceState_PhysDetachingFromVM:
1730 switch (aPendingState)
1731 {
1732 case kHostUSBDeviceState_PhysDetached:
1733 switch (aSubState)
1734 {
1735 case kHostUSBDeviceSubState_Default:
1736 return "PhysDetachingFromVM";
1737 default:
1738 AssertFailedReturn("AttachingToVM[bad]");
1739 }
1740 break;
1741
1742 default:
1743 AssertFailedReturn("AttachingToVM{bad}");
1744 }
1745 break;
1746
1747 default:
1748 AssertFailedReturn("BadState");
1749
1750 }
1751
1752 AssertFailedReturn("shouldn't get here");
1753}
1754
1755/**
1756 * Set the device state.
1757 *
1758 * This method will verify that the state transition is a legal one
1759 * according to the statemachine. It will also take care of the
1760 * associated house keeping and determine if filters needs to be applied.
1761 *
1762 * @param aNewState The new state.
1763 * @param aNewPendingState The final state of a transition when applicable.
1764 * @param aNewSubState The new sub-state when applicable.
1765 *
1766 * @returns true if filters should be applied to the device, false if not.
1767 *
1768 * @note The caller must own the write lock for this object.
1769 */
1770bool HostUSBDevice::i_setState(HostUSBDeviceState aNewState,
1771 HostUSBDeviceState aNewPendingState /*= kHostUSBDeviceState_Invalid*/,
1772 HostUSBDeviceSubState aNewSubState /*= kHostUSBDeviceSubState_Default*/)
1773{
1774 Assert(isWriteLockOnCurrentThread());
1775 Assert( aNewSubState == kHostUSBDeviceSubState_Default
1776 || aNewSubState == kHostUSBDeviceSubState_AwaitingDetach
1777 || aNewSubState == kHostUSBDeviceSubState_AwaitingReAttach);
1778
1779 /*
1780 * If the state is unchanged, then don't bother going
1781 * thru the validation and setting. This saves a bit of code.
1782 */
1783 if ( aNewState == mUniState
1784 && aNewPendingState == mPendingUniState
1785 && aNewSubState == mUniSubState)
1786 return false;
1787
1788 /*
1789 * Welcome to the switch orgies!
1790 * You're welcome to check out the ones in startTransition(),
1791 * advanceTransition(), failTransition() and i_getStateName() too. Enjoy!
1792 */
1793
1794 bool fFilters = false;
1795 HostUSBDeviceState NewPrevState = mUniState;
1796 switch (mUniState)
1797 {
1798 /*
1799 * Not much can be done with a device in this state.
1800 */
1801 case kHostUSBDeviceState_Unsupported:
1802 switch (aNewState)
1803 {
1804 case kHostUSBDeviceState_PhysDetached:
1805 Assert(aNewPendingState == kHostUSBDeviceState_Invalid);
1806 Assert(aNewSubState == kHostUSBDeviceSubState_Default);
1807 break;
1808 default:
1809 AssertLogRelMsgFailedReturn(("this=%p %s -X-> %s\n", this, i_getStateName(),
1810 i_stateName(aNewState, aNewPendingState, aNewSubState)), false);
1811 }
1812 break;
1813
1814 /*
1815 * Only the host OS (or the user) can make changes
1816 * that'll make a device get out of this state.
1817 */
1818 case kHostUSBDeviceState_UsedByHost:
1819 switch (aNewState)
1820 {
1821 case kHostUSBDeviceState_Capturable:
1822 case kHostUSBDeviceState_Unused:
1823 fFilters = true;
1824 case kHostUSBDeviceState_PhysDetached:
1825 Assert(aNewPendingState == kHostUSBDeviceState_Invalid);
1826 Assert(aNewSubState == kHostUSBDeviceSubState_Default);
1827 break;
1828 default:
1829 AssertLogRelMsgFailedReturn(("this=%p %s -X-> %s\n", this, i_getStateName(),
1830 i_stateName(aNewState, aNewPendingState, aNewSubState)), false);
1831 }
1832 break;
1833
1834 /*
1835 * Now it gets interesting.
1836 */
1837 case kHostUSBDeviceState_Capturable:
1838 switch (aNewState)
1839 {
1840 /* Host changes. */
1841 case kHostUSBDeviceState_Unused:
1842 fFilters = true; /* Wildcard only... */
1843 case kHostUSBDeviceState_UsedByHost:
1844 case kHostUSBDeviceState_PhysDetached:
1845 Assert(aNewPendingState == kHostUSBDeviceState_Invalid);
1846 Assert(aNewSubState == kHostUSBDeviceSubState_Default);
1847 break;
1848
1849 /* VBox actions */
1850 case kHostUSBDeviceState_Capturing:
1851 switch (aNewPendingState)
1852 {
1853 case kHostUSBDeviceState_HeldByProxy:
1854 case kHostUSBDeviceState_UsedByVM:
1855 break;
1856 default:
1857 AssertLogRelMsgFailedReturn(("this=%p %s -X-> %s\n", this, i_getStateName(),
1858 i_stateName(aNewState, aNewPendingState, aNewSubState)), false);
1859 }
1860 break;
1861 default:
1862 AssertLogRelMsgFailedReturn(("this=%p %s -X-> %s\n", this, i_getStateName(),
1863 i_stateName(aNewState, aNewPendingState, aNewSubState)), false);
1864 }
1865 break;
1866
1867 case kHostUSBDeviceState_Unused:
1868 switch (aNewState)
1869 {
1870 /* Host changes. */
1871 case kHostUSBDeviceState_PhysDetached:
1872 case kHostUSBDeviceState_UsedByHost:
1873 case kHostUSBDeviceState_Capturable:
1874 Assert(aNewPendingState == kHostUSBDeviceState_Invalid);
1875 Assert(aNewSubState == kHostUSBDeviceSubState_Default);
1876 break;
1877
1878 /* VBox actions */
1879 case kHostUSBDeviceState_Capturing:
1880 switch (aNewPendingState)
1881 {
1882 case kHostUSBDeviceState_HeldByProxy:
1883 case kHostUSBDeviceState_UsedByVM:
1884 break;
1885 default:
1886 AssertLogRelMsgFailedReturn(("this=%p %s -X-> %s\n", this, i_getStateName(),
1887 i_stateName(aNewState, aNewPendingState, aNewSubState)), false);
1888 }
1889 break;
1890 default:
1891 AssertLogRelMsgFailedReturn(("this=%p %s -X-> %s\n", this, i_getStateName(),
1892 i_stateName(aNewState, aNewPendingState, aNewSubState)), false);
1893 }
1894 break;
1895
1896 /*
1897 * VBox owns this device now, what's next...
1898 */
1899 case kHostUSBDeviceState_HeldByProxy:
1900 switch (aNewState)
1901 {
1902 /* Host changes. */
1903 case kHostUSBDeviceState_PhysDetached:
1904 Assert(aNewPendingState == kHostUSBDeviceState_Invalid);
1905 Assert(aNewSubState == kHostUSBDeviceSubState_Default);
1906 break;
1907
1908 /* VBox actions */
1909 case kHostUSBDeviceState_AttachingToVM:
1910 switch (aNewPendingState)
1911 {
1912 case kHostUSBDeviceState_UsedByVM:
1913 break;
1914 default:
1915 AssertLogRelMsgFailedReturn(("this=%p %s -X-> %s\n", this, i_getStateName(),
1916 i_stateName(aNewState, aNewPendingState, aNewSubState)), false);
1917 }
1918 break;
1919 case kHostUSBDeviceState_ReleasingToHost:
1920 switch (aNewPendingState)
1921 {
1922 case kHostUSBDeviceState_Unused: /* Only this! */
1923 break;
1924 default:
1925 AssertLogRelMsgFailedReturn(("this=%p %s -X-> %s\n", this, i_getStateName(),
1926 i_stateName(aNewState, aNewPendingState, aNewSubState)), false);
1927 }
1928 break;
1929 default:
1930 AssertLogRelMsgFailedReturn(("this=%p %s -X-> %s\n", this, i_getStateName(),
1931 i_stateName(aNewState, aNewPendingState, aNewSubState)), false);
1932 }
1933 break;
1934
1935
1936 case kHostUSBDeviceState_UsedByVM:
1937 switch (aNewState)
1938 {
1939 /* Host changes. */
1940 case kHostUSBDeviceState_PhysDetachingFromVM:
1941 Assert(aNewSubState == kHostUSBDeviceSubState_Default);
1942 Assert(aNewPendingState == kHostUSBDeviceState_PhysDetached);
1943 break;
1944
1945 /* VBox actions */
1946 case kHostUSBDeviceState_DetachingFromVM:
1947 switch (aNewPendingState)
1948 {
1949 case kHostUSBDeviceState_HeldByProxy:
1950 case kHostUSBDeviceState_Unused: /* Only this! */
1951 break;
1952 default:
1953 AssertLogRelMsgFailedReturn(("this=%p %s -X-> %s\n", this, i_getStateName(),
1954 i_stateName(aNewState, aNewPendingState, aNewSubState)), false);
1955 }
1956 break;
1957 default:
1958 AssertLogRelMsgFailedReturn(("this=%p %s -X-> %s\n", this, i_getStateName(),
1959 i_stateName(aNewState, aNewPendingState, aNewSubState)), false);
1960 }
1961 break;
1962
1963 /*
1964 * The final state.
1965 */
1966 case kHostUSBDeviceState_PhysDetached:
1967 switch (mUniState)
1968 {
1969 case kHostUSBDeviceState_Unsupported:
1970 case kHostUSBDeviceState_UsedByHost:
1971 case kHostUSBDeviceState_Capturable:
1972 case kHostUSBDeviceState_Unused:
1973 case kHostUSBDeviceState_HeldByProxy:
1974 case kHostUSBDeviceState_PhysDetachingFromVM:
1975 case kHostUSBDeviceState_DetachingFromVM: // ??
1976 case kHostUSBDeviceState_Capturing:
1977 case kHostUSBDeviceState_ReleasingToHost:
1978 break;
1979
1980 case kHostUSBDeviceState_AttachingToVM: // ??
1981 case kHostUSBDeviceState_UsedByVM:
1982 default:
1983 AssertLogRelMsgFailedReturn(("this=%p %s -X-> %s\n", this, i_getStateName(),
1984 i_stateName(aNewState, aNewPendingState, aNewSubState)), false);
1985 }
1986 break;
1987
1988
1989 /*
1990 * The transitional states.
1991 */
1992 case kHostUSBDeviceState_Capturing:
1993 NewPrevState = mPrevUniState;
1994 switch (aNewState)
1995 {
1996 /* Sub state advance. */
1997 case kHostUSBDeviceState_Capturing:
1998 switch (aNewSubState)
1999 {
2000 case kHostUSBDeviceSubState_AwaitingReAttach:
2001 Assert(mUniSubState == kHostUSBDeviceSubState_AwaitingDetach);
2002 Assert(aNewPendingState == mPendingUniState);
2003 break;
2004 default:
2005 AssertReleaseMsgFailedReturn(("this=%p mUniState=%d\n", this, mUniState), false);
2006 }
2007 break;
2008
2009 /* Host/User/Failure. */
2010 case kHostUSBDeviceState_PhysDetached:
2011 Assert(aNewPendingState == kHostUSBDeviceState_Invalid);
2012 Assert(aNewSubState == kHostUSBDeviceSubState_Default);
2013 break;
2014 case kHostUSBDeviceState_UsedByHost:
2015 case kHostUSBDeviceState_Capturable:
2016 case kHostUSBDeviceState_Unused:
2017 Assert(aNewState == mPrevUniState); /** @todo This is kind of wrong, see i_failTransition. */
2018 Assert(aNewPendingState == kHostUSBDeviceState_Invalid);
2019 Assert(aNewSubState == kHostUSBDeviceSubState_Default);
2020 break;
2021
2022 /* VBox */
2023 case kHostUSBDeviceState_HeldByProxy:
2024 Assert(aNewPendingState == kHostUSBDeviceState_Invalid);
2025 Assert(aNewSubState == kHostUSBDeviceSubState_Default);
2026 Assert( mPendingUniState == kHostUSBDeviceState_HeldByProxy
2027 || mPendingUniState == kHostUSBDeviceState_UsedByVM /* <- failure */ );
2028 break;
2029 case kHostUSBDeviceState_AttachingToVM:
2030 Assert(aNewPendingState == kHostUSBDeviceState_UsedByVM);
2031 NewPrevState = kHostUSBDeviceState_HeldByProxy;
2032 break;
2033
2034 default:
2035 AssertLogRelMsgFailedReturn(("this=%p %s -X-> %s\n", this, i_getStateName(),
2036 i_stateName(aNewState, aNewPendingState, aNewSubState)), false);
2037 }
2038 break;
2039
2040 case kHostUSBDeviceState_ReleasingToHost:
2041 Assert(mPrevUniState == kHostUSBDeviceState_HeldByProxy);
2042 NewPrevState = mPrevUniState;
2043 switch (aNewState)
2044 {
2045 /* Sub state advance. */
2046 case kHostUSBDeviceState_ReleasingToHost:
2047 switch (aNewSubState)
2048 {
2049 case kHostUSBDeviceSubState_AwaitingReAttach:
2050 Assert(mUniSubState == kHostUSBDeviceSubState_AwaitingDetach);
2051 Assert(aNewPendingState == mPendingUniState);
2052 break;
2053 default:
2054 AssertReleaseMsgFailedReturn(("this=%p mUniState=%d\n", this, mUniState), false);
2055 }
2056 break;
2057
2058 /* Host/Failure. */
2059 case kHostUSBDeviceState_PhysDetached:
2060 Assert(aNewPendingState == kHostUSBDeviceState_Invalid);
2061 Assert(aNewSubState == kHostUSBDeviceSubState_Default);
2062 break;
2063 case kHostUSBDeviceState_HeldByProxy:
2064 Assert(aNewPendingState == kHostUSBDeviceState_Invalid);
2065 Assert(aNewSubState == kHostUSBDeviceSubState_Default);
2066 Assert(mPendingUniState == kHostUSBDeviceState_Unused);
2067 break;
2068
2069 /* Success */
2070 case kHostUSBDeviceState_UsedByHost:
2071 case kHostUSBDeviceState_Capturable:
2072 case kHostUSBDeviceState_Unused:
2073 Assert(aNewPendingState == kHostUSBDeviceState_Invalid);
2074 Assert(aNewSubState == kHostUSBDeviceSubState_Default);
2075 Assert(mPendingUniState == kHostUSBDeviceState_Unused);
2076 break;
2077
2078 default:
2079 AssertLogRelMsgFailedReturn(("this=%p %s -X-> %s\n", this, i_getStateName(),
2080 i_stateName(aNewState, aNewPendingState, aNewSubState)), false);
2081 }
2082 break;
2083
2084 case kHostUSBDeviceState_AttachingToVM:
2085 Assert(mPrevUniState == kHostUSBDeviceState_HeldByProxy);
2086 NewPrevState = mPrevUniState;
2087 switch (aNewState)
2088 {
2089 /* Host/Failure. */
2090 case kHostUSBDeviceState_PhysDetachingFromVM:
2091 Assert(aNewPendingState == kHostUSBDeviceState_PhysDetached);
2092 Assert(aNewSubState == kHostUSBDeviceSubState_Default);
2093 break;
2094 case kHostUSBDeviceState_HeldByProxy:
2095 Assert(aNewPendingState == kHostUSBDeviceState_Invalid);
2096 Assert(aNewSubState == kHostUSBDeviceSubState_Default);
2097 Assert(mPendingUniState == kHostUSBDeviceState_Unused);
2098 break;
2099
2100 /* Success */
2101 case kHostUSBDeviceState_UsedByVM:
2102 Assert(aNewPendingState == kHostUSBDeviceState_Invalid);
2103 Assert(aNewSubState == kHostUSBDeviceSubState_Default);
2104 Assert(mPendingUniState == kHostUSBDeviceState_UsedByVM);
2105 break;
2106
2107 default:
2108 AssertLogRelMsgFailedReturn(("this=%p %s -X-> %s\n", this, i_getStateName(),
2109 i_stateName(aNewState, aNewPendingState, aNewSubState)), false);
2110 }
2111 break;
2112
2113 case kHostUSBDeviceState_DetachingFromVM:
2114 Assert(mPrevUniState == kHostUSBDeviceState_UsedByVM);
2115 NewPrevState = mPrevUniState;
2116 switch (aNewState)
2117 {
2118 /* Host/Failure. */
2119 case kHostUSBDeviceState_PhysDetached: //??
2120 Assert(aNewPendingState == kHostUSBDeviceState_Invalid);
2121 Assert(aNewSubState == kHostUSBDeviceSubState_Default);
2122 break;
2123 case kHostUSBDeviceState_PhysDetachingFromVM:
2124 Assert(aNewPendingState == kHostUSBDeviceState_PhysDetached);
2125 Assert(aNewSubState == kHostUSBDeviceSubState_Default);
2126 break;
2127
2128 /* Success */
2129 case kHostUSBDeviceState_HeldByProxy:
2130 Assert(aNewPendingState == kHostUSBDeviceState_Invalid);
2131 Assert(aNewSubState == kHostUSBDeviceSubState_Default);
2132 Assert(mPendingUniState == kHostUSBDeviceState_HeldByProxy);
2133 fFilters = true;
2134 break;
2135
2136 case kHostUSBDeviceState_ReleasingToHost:
2137 Assert(aNewPendingState == kHostUSBDeviceState_Invalid);
2138 Assert(aNewSubState == kHostUSBDeviceSubState_Default);
2139 Assert(mPendingUniState == kHostUSBDeviceState_Unused);
2140 NewPrevState = kHostUSBDeviceState_HeldByProxy;
2141 break;
2142
2143 default:
2144 AssertLogRelMsgFailedReturn(("this=%p %s -X-> %s\n", this, i_getStateName(),
2145 i_stateName(aNewState, aNewPendingState, aNewSubState)), false);
2146 }
2147 break;
2148
2149 case kHostUSBDeviceState_PhysDetachingFromVM:
2150 Assert( mPrevUniState == kHostUSBDeviceState_DetachingFromVM
2151 || mPrevUniState == kHostUSBDeviceState_AttachingToVM
2152 || mPrevUniState == kHostUSBDeviceState_UsedByVM);
2153 NewPrevState = mPrevUniState; /* preserving it is more useful. */
2154 switch (aNewState)
2155 {
2156 case kHostUSBDeviceState_PhysDetached:
2157 Assert(aNewPendingState == kHostUSBDeviceState_Invalid);
2158 Assert(aNewSubState == kHostUSBDeviceSubState_Default);
2159 break;
2160 default:
2161 AssertLogRelMsgFailedReturn(("this=%p %s -X-> %s\n", this, i_getStateName(),
2162 i_stateName(aNewState, aNewPendingState, aNewSubState)), false);
2163 }
2164 break;
2165
2166 default:
2167 AssertReleaseMsgFailedReturn(("this=%p mUniState=%d\n", this, mUniState), false);
2168 }
2169
2170 /*
2171 * Make the state change.
2172 */
2173 if (NewPrevState != mPrevUniState)
2174 LogFlowThisFunc(("%s -> %s (prev: %s -> %s) [%s]\n",
2175 i_getStateName(), i_stateName(aNewState, aNewPendingState, aNewSubState),
2176 i_stateName(mPrevUniState), i_stateName(NewPrevState), mName));
2177 else
2178 LogFlowThisFunc(("%s -> %s (prev: %s) [%s]\n",
2179 i_getStateName(), i_stateName(aNewState, aNewPendingState, aNewSubState),
2180 i_stateName(NewPrevState), mName));
2181 mPrevUniState = NewPrevState;
2182 mUniState = aNewState;
2183 mUniSubState = aNewSubState;
2184 mPendingUniState = aNewPendingState;
2185 mLastStateChangeTS = RTTimeNanoTS();
2186
2187 return fFilters;
2188}
2189
2190
2191/**
2192 * A convenience for entering a transitional state.
2193
2194 * @param aNewState The new state (transitional).
2195 * @param aFinalSubState The final state of the transition (non-transitional).
2196 * @param aNewSubState The new sub-state when applicable.
2197 *
2198 * @returns Always false because filters are never applied for the start of a transition.
2199 *
2200 * @note The caller must own the write lock for this object.
2201 */
2202bool HostUSBDevice::i_startTransition(HostUSBDeviceState aNewState, HostUSBDeviceState aFinalState,
2203 HostUSBDeviceSubState aNewSubState /*= kHostUSBDeviceSubState_Default*/)
2204{
2205 AssertReturn(isWriteLockOnCurrentThread(), false);
2206 /*
2207 * A quick prevalidation thing. Not really necessary since setState
2208 * verifies this too, but it's very easy here.
2209 */
2210 switch (mUniState)
2211 {
2212 case kHostUSBDeviceState_Unsupported:
2213 case kHostUSBDeviceState_UsedByHost:
2214 case kHostUSBDeviceState_Capturable:
2215 case kHostUSBDeviceState_Unused:
2216 case kHostUSBDeviceState_HeldByProxy:
2217 case kHostUSBDeviceState_UsedByVM:
2218 break;
2219
2220 case kHostUSBDeviceState_DetachingFromVM:
2221 case kHostUSBDeviceState_Capturing:
2222 case kHostUSBDeviceState_ReleasingToHost:
2223 case kHostUSBDeviceState_AttachingToVM:
2224 case kHostUSBDeviceState_PhysDetachingFromVM:
2225 AssertMsgFailedReturn(("this=%p %s is a transitional state.\n", this, i_getStateName()), false);
2226
2227 case kHostUSBDeviceState_PhysDetached:
2228 default:
2229 AssertReleaseMsgFailedReturn(("this=%p mUniState=%d\n", this, mUniState), false);
2230 }
2231
2232 return i_setState(aNewState, aFinalState, aNewSubState);
2233}
2234
2235
2236/**
2237 * A convenience for advancing a transitional state forward.
2238 *
2239 * @param aSkipReAttach Fast forwards thru the re-attach substate if
2240 * applicable.
2241 *
2242 * @returns true if filters should be applied to the device, false if not.
2243 *
2244 * @note The caller must own the write lock for this object.
2245 */
2246bool HostUSBDevice::i_advanceTransition(bool aSkipReAttach /* = false */)
2247{
2248 AssertReturn(isWriteLockOnCurrentThread(), false);
2249 HostUSBDeviceState enmPending = mPendingUniState;
2250 HostUSBDeviceSubState enmSub = mUniSubState;
2251 HostUSBDeviceState enmState = mUniState;
2252 switch (enmState)
2253 {
2254 case kHostUSBDeviceState_Capturing:
2255 switch (enmSub)
2256 {
2257 case kHostUSBDeviceSubState_AwaitingDetach:
2258 enmSub = kHostUSBDeviceSubState_AwaitingReAttach;
2259 break;
2260 case kHostUSBDeviceSubState_AwaitingReAttach:
2261 enmSub = kHostUSBDeviceSubState_Default;
2262 /* fall thru */
2263 case kHostUSBDeviceSubState_Default:
2264 switch (enmPending)
2265 {
2266 case kHostUSBDeviceState_UsedByVM:
2267 enmState = kHostUSBDeviceState_AttachingToVM;
2268 break;
2269 case kHostUSBDeviceState_HeldByProxy:
2270 enmState = enmPending;
2271 enmPending = kHostUSBDeviceState_Invalid;
2272 break;
2273 default:
2274 AssertMsgFailedReturn(("this=%p invalid pending state %d: %s\n",
2275 this, enmPending, i_getStateName()), false);
2276 }
2277 break;
2278 default:
2279 AssertReleaseMsgFailedReturn(("this=%p mUniState=%d\n", this, mUniState), false);
2280 }
2281 break;
2282
2283 case kHostUSBDeviceState_ReleasingToHost:
2284 switch (enmSub)
2285 {
2286 case kHostUSBDeviceSubState_AwaitingDetach:
2287 enmSub = kHostUSBDeviceSubState_AwaitingReAttach;
2288 break;
2289 case kHostUSBDeviceSubState_AwaitingReAttach:
2290 enmSub = kHostUSBDeviceSubState_Default;
2291 /* fall thru */
2292 case kHostUSBDeviceSubState_Default:
2293 switch (enmPending)
2294 {
2295 /* Use Unused here since it implies that filters has been applied
2296 and will make sure they aren't applied if the final state really
2297 is Capturable. */
2298 case kHostUSBDeviceState_Unused:
2299 enmState = enmPending;
2300 enmPending = kHostUSBDeviceState_Invalid;
2301 break;
2302 default:
2303 AssertMsgFailedReturn(("this=%p invalid pending state %d: %s\n",
2304 this, enmPending, i_getStateName()), false);
2305 }
2306 break;
2307 default:
2308 AssertReleaseMsgFailedReturn(("this=%p mUniState=%d\n", this, mUniState), false);
2309 }
2310 break;
2311
2312 case kHostUSBDeviceState_AttachingToVM:
2313 switch (enmSub)
2314 {
2315 case kHostUSBDeviceSubState_AwaitingDetach:
2316 enmSub = kHostUSBDeviceSubState_AwaitingReAttach;
2317 break;
2318 case kHostUSBDeviceSubState_AwaitingReAttach:
2319 enmSub = kHostUSBDeviceSubState_Default;
2320 /* fall thru */
2321 case kHostUSBDeviceSubState_Default:
2322 switch (enmPending)
2323 {
2324 case kHostUSBDeviceState_UsedByVM:
2325 enmState = enmPending;
2326 enmPending = kHostUSBDeviceState_Invalid;
2327 break;
2328 default:
2329 AssertMsgFailedReturn(("this=%p invalid pending state %d: %s\n",
2330 this, enmPending, i_getStateName()), false);
2331 }
2332 break;
2333 default:
2334 AssertReleaseMsgFailedReturn(("this=%p mUniState=%d\n", this, mUniState), false);
2335 }
2336 break;
2337
2338 case kHostUSBDeviceState_DetachingFromVM:
2339 switch (enmSub)
2340 {
2341 case kHostUSBDeviceSubState_AwaitingDetach:
2342 enmSub = kHostUSBDeviceSubState_AwaitingReAttach;
2343 break;
2344 case kHostUSBDeviceSubState_AwaitingReAttach:
2345 enmSub = kHostUSBDeviceSubState_Default;
2346 /* fall thru */
2347 case kHostUSBDeviceSubState_Default:
2348 switch (enmPending)
2349 {
2350 case kHostUSBDeviceState_HeldByProxy:
2351 enmState = enmPending;
2352 enmPending = kHostUSBDeviceState_Invalid;
2353 break;
2354 case kHostUSBDeviceState_Unused:
2355 enmState = kHostUSBDeviceState_ReleasingToHost;
2356 break;
2357 default:
2358 AssertMsgFailedReturn(("this=%p invalid pending state %d: %s\n",
2359 this, enmPending, i_getStateName()), false);
2360 }
2361 break;
2362 default:
2363 AssertReleaseMsgFailedReturn(("this=%p mUniState=%d\n", this, mUniState), false);
2364 }
2365 break;
2366
2367 case kHostUSBDeviceState_PhysDetachingFromVM:
2368 switch (enmSub)
2369 {
2370 case kHostUSBDeviceSubState_Default:
2371 switch (enmPending)
2372 {
2373 case kHostUSBDeviceState_PhysDetached:
2374 enmState = enmPending;
2375 enmPending = kHostUSBDeviceState_Invalid;
2376 break;
2377 default:
2378 AssertMsgFailedReturn(("this=%p invalid pending state %d: %s\n",
2379 this, enmPending, i_getStateName()), false);
2380 }
2381 break;
2382 default:
2383 AssertReleaseMsgFailedReturn(("this=%p mUniState=%d\n", this, mUniState), false);
2384 }
2385 break;
2386
2387 case kHostUSBDeviceState_Unsupported:
2388 case kHostUSBDeviceState_UsedByHost:
2389 case kHostUSBDeviceState_Capturable:
2390 case kHostUSBDeviceState_Unused:
2391 case kHostUSBDeviceState_HeldByProxy:
2392 case kHostUSBDeviceState_UsedByVM:
2393 AssertMsgFailedReturn(("this=%p %s is not transitional\n", this, i_getStateName()), false);
2394 case kHostUSBDeviceState_PhysDetached:
2395 default:
2396 AssertReleaseMsgFailedReturn(("this=%p mUniState=%d\n", this, enmState), false);
2397
2398 }
2399
2400 bool fRc = i_setState(enmState, enmPending, enmSub);
2401 if (aSkipReAttach && mUniSubState == kHostUSBDeviceSubState_AwaitingReAttach)
2402 fRc |= i_advanceTransition(false /* don't fast forward re-attach */);
2403 return fRc;
2404}
2405
2406/**
2407 * A convenience for failing a transitional state.
2408 *
2409 * @return true if filters should be applied to the device, false if not.
2410 * @param a_enmStateHint USB device state hint. kHostUSBDeviceState_Invalid
2411 * if the caller doesn't have a clue to give.
2412 *
2413 * @note The caller must own the write lock for this object.
2414 */
2415bool HostUSBDevice::i_failTransition(HostUSBDeviceState a_enmStateHint)
2416{
2417 AssertReturn(isWriteLockOnCurrentThread(), false);
2418 HostUSBDeviceSubState enmSub = mUniSubState;
2419 HostUSBDeviceState enmState = mUniState;
2420 switch (enmState)
2421 {
2422 /*
2423 * There are just two cases, either we got back to the
2424 * previous state (assumes Capture+Attach-To-VM updates it)
2425 * or we assume the device has been unplugged (physically).
2426 */
2427 case kHostUSBDeviceState_DetachingFromVM:
2428 case kHostUSBDeviceState_Capturing:
2429 case kHostUSBDeviceState_ReleasingToHost:
2430 case kHostUSBDeviceState_AttachingToVM:
2431 switch (enmSub)
2432 {
2433 case kHostUSBDeviceSubState_AwaitingDetach:
2434 enmSub = kHostUSBDeviceSubState_Default;
2435 /* fall thru */
2436 case kHostUSBDeviceSubState_Default:
2437 enmState = mPrevUniState;
2438 break;
2439 case kHostUSBDeviceSubState_AwaitingReAttach:
2440 enmSub = kHostUSBDeviceSubState_Default;
2441 if (a_enmStateHint != kHostUSBDeviceState_Invalid)
2442 enmState = mPrevUniState; /** @todo enmState = a_enmStateHint is more correct, but i_setState doesn't like it. It will usually correct itself shortly. */
2443 else
2444 enmState = kHostUSBDeviceState_PhysDetached;
2445 break;
2446 default:
2447 AssertReleaseMsgFailedReturn(("this=%p mUniState=%d\n", this, mUniState), false);
2448 }
2449 break;
2450
2451 case kHostUSBDeviceState_PhysDetachingFromVM:
2452 AssertMsgFailedReturn(("this=%p %s shall not fail\n", this, i_getStateName()), false);
2453
2454 case kHostUSBDeviceState_Unsupported:
2455 case kHostUSBDeviceState_UsedByHost:
2456 case kHostUSBDeviceState_Capturable:
2457 case kHostUSBDeviceState_Unused:
2458 case kHostUSBDeviceState_HeldByProxy:
2459 case kHostUSBDeviceState_UsedByVM:
2460 AssertMsgFailedReturn(("this=%p %s is not transitional\n", this, i_getStateName()), false);
2461 case kHostUSBDeviceState_PhysDetached:
2462 default:
2463 AssertReleaseMsgFailedReturn(("this=%p mUniState=%d\n", this, mUniState), false);
2464
2465 }
2466
2467 return i_setState(enmState, kHostUSBDeviceState_Invalid, enmSub);
2468}
2469
2470
2471/**
2472 * Determines the canonical state of the device.
2473 *
2474 * @returns canonical state.
2475 *
2476 * @note The caller must own the read (or write) lock for this object.
2477 */
2478USBDeviceState_T HostUSBDevice::i_canonicalState() const
2479{
2480 switch (mUniState)
2481 {
2482 /*
2483 * Straight forward.
2484 */
2485 case kHostUSBDeviceState_Unsupported:
2486 return USBDeviceState_NotSupported;
2487
2488 case kHostUSBDeviceState_UsedByHost:
2489 return USBDeviceState_Unavailable;
2490
2491 case kHostUSBDeviceState_Capturable:
2492 return USBDeviceState_Busy;
2493
2494 case kHostUSBDeviceState_Unused:
2495 return USBDeviceState_Available;
2496
2497 case kHostUSBDeviceState_HeldByProxy:
2498 return USBDeviceState_Held;
2499
2500 case kHostUSBDeviceState_UsedByVM:
2501 return USBDeviceState_Captured;
2502
2503 /*
2504 * Pretend we've reached the final state.
2505 */
2506 case kHostUSBDeviceState_Capturing:
2507 Assert( mPendingUniState == kHostUSBDeviceState_UsedByVM
2508 || mPendingUniState == kHostUSBDeviceState_HeldByProxy);
2509 return mPendingUniState == kHostUSBDeviceState_UsedByVM
2510 ? (USBDeviceState_T)USBDeviceState_Captured
2511 : (USBDeviceState_T)USBDeviceState_Held;
2512 /* The cast ^^^^ is because xidl is using different enums for
2513 each of the values. *Very* nice idea... :-) */
2514
2515 case kHostUSBDeviceState_AttachingToVM:
2516 return USBDeviceState_Captured;
2517
2518 /*
2519 * Return the previous state.
2520 */
2521 case kHostUSBDeviceState_ReleasingToHost:
2522 Assert( mPrevUniState == kHostUSBDeviceState_UsedByVM
2523 || mPrevUniState == kHostUSBDeviceState_HeldByProxy);
2524 return mPrevUniState == kHostUSBDeviceState_UsedByVM
2525 ? (USBDeviceState_T)USBDeviceState_Captured
2526 : (USBDeviceState_T)USBDeviceState_Held;
2527 /* The cast ^^^^ is because xidl is using different enums for
2528 each of the values. *Very* nice idea... :-) */
2529
2530 case kHostUSBDeviceState_DetachingFromVM:
2531 return USBDeviceState_Captured;
2532 case kHostUSBDeviceState_PhysDetachingFromVM:
2533 return USBDeviceState_Captured;
2534
2535 case kHostUSBDeviceState_PhysDetached:
2536 default:
2537 AssertReleaseMsgFailedReturn(("this=%p mUniState=%d\n", this, mUniState), USBDeviceState_NotSupported);
2538 }
2539 /* won't ever get here. */
2540}
2541/* vi: set tabstop=4 shiftwidth=4 expandtab: */
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