VirtualBox

source: vbox/trunk/src/VBox/Main/include/HostUSBDeviceImpl.h@ 94249

Last change on this file since 94249 was 93115, checked in by vboxsync, 3 years ago

scm --update-copyright-year

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 12.3 KB
Line 
1/* $Id: HostUSBDeviceImpl.h 93115 2022-01-01 11:31:46Z vboxsync $ */
2/** @file
3 * VirtualBox IHostUSBDevice COM interface implementation.
4 */
5
6/*
7 * Copyright (C) 2006-2022 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#ifndef MAIN_INCLUDED_HostUSBDeviceImpl_h
19#define MAIN_INCLUDED_HostUSBDeviceImpl_h
20#ifndef RT_WITHOUT_PRAGMA_ONCE
21# pragma once
22#endif
23
24#include "VirtualBoxBase.h"
25#include "USBDeviceFilterImpl.h"
26#include <VBox/usb.h>
27#include "HostUSBDeviceWrap.h"
28
29class SessionMachine;
30class USBProxyBackend;
31
32/**
33 * The unified state machine of HostUSBDevice.
34 *
35 * This is a super set of USBDEVICESTATE / USBDeviceState_T that
36 * includes additional states for tracking state transitions.
37 *
38 * @remarks
39 * The CapturingForVM and CapturingForProxy states have been merged
40 * into Capturing with a destination state (AttachingToVM or HeldByProxy).
41 *
42 * The DetachingFromVM state is a merge of DetachingFromVMToProxy and
43 * DetachingFromVMToHost and uses the destination state (HeldByProxy
44 * or ReleasingToHost) like Capturing.
45 *
46 * The *AwaitingDetach and *AwaitingReattach substates (optionally used
47 * in Capturing, AttachingToVM, DetachingFromVM and ReleasingToHost) are
48 * implemented via a substate kHostUSBDeviceSubState.
49 */
50typedef enum
51{
52 /** The device is unsupported (HUB).
53 * Next Host: PhysDetached.
54 * Next VBox: No change permitted.
55 */
56 kHostUSBDeviceState_Unsupported = USBDEVICESTATE_UNSUPPORTED,
57 /** The device is used exclusivly by the host or is inaccessible for some other reason.
58 * Next Host: Capturable, Unused, PhysDetached.
59 * Run filters.
60 * Next VBox: No change permitted.
61 */
62 kHostUSBDeviceState_UsedByHost = USBDEVICESTATE_USED_BY_HOST,
63 /** The device is used by the host but can be captured.
64 * Next Host: Unsupported, UsedByHost, Unused, PhysDetached.
65 * Run filters if Unused (for wildcard filters).
66 * Next VBox: CapturingForVM, CapturingForProxy.
67 */
68 kHostUSBDeviceState_Capturable = USBDEVICESTATE_USED_BY_HOST_CAPTURABLE,
69 /** The device is not used by the host and can be captured.
70 * Next Host: UsedByHost, Capturable, PhysDetached
71 * Don't run any filters (done on state entry).
72 * Next VBox: CapturingForVM, CapturingForProxy.
73 */
74 kHostUSBDeviceState_Unused = USBDEVICESTATE_UNUSED,
75 /** The device is held captive by the proxy.
76 * Next Host: PhysDetached
77 * Next VBox: ReleasingHeld, AttachingToVM
78 */
79 kHostUSBDeviceState_HeldByProxy = USBDEVICESTATE_HELD_BY_PROXY,
80 /** The device is in use by a VM.
81 * Next Host: PhysDetachingFromVM
82 * Next VBox: DetachingFromVM
83 */
84 kHostUSBDeviceState_UsedByVM = USBDEVICESTATE_USED_BY_GUEST,
85 /** The device has been detach from both the host and VMs.
86 * This is the final state. */
87 kHostUSBDeviceState_PhysDetached = 9,
88
89
90 /** The start of the transitional states. */
91 kHostUSBDeviceState_FirstTransitional,
92
93 /** The device is being seized from the host, either for HeldByProxy or for AttachToVM.
94 *
95 * On some hosts we will need to re-enumerate the in which case the sub-state
96 * is employed to track this progress. On others, this is synchronous or faked, and
97 * will will then leave the device in this state and poke the service thread to do
98 * the completion state change.
99 *
100 * Next Host: PhysDetached.
101 * Next VBox: HeldByProxy or AttachingToVM on success,
102 * previous state (Unused or Capturable) or UsedByHost on failure.
103 */
104 kHostUSBDeviceState_Capturing = kHostUSBDeviceState_FirstTransitional,
105
106 /** The device is being released back to the host, following VM or Proxy usage.
107 * Most hosts needs to re-enumerate the device and will therefore employ the
108 * sub-state as during capturing. On the others we'll just leave it to the usb
109 * service thread to advance the device state.
110 *
111 * Next Host: Unused, UsedByHost, Capturable.
112 * No filters.
113 * Next VBox: PhysDetached (timeout), HeldByProxy (failure).
114 */
115 kHostUSBDeviceState_ReleasingToHost,
116
117 /** The device is being attached to a VM.
118 *
119 * This requires IPC to the VM and we will not advance the state until
120 * that completes.
121 *
122 * Next Host: PhysDetachingFromVM.
123 * Next VBox: UsedByGuest, HeldByProxy (failure).
124 */
125 kHostUSBDeviceState_AttachingToVM,
126
127 /** The device is being detached from a VM and will be returned to the proxy or host.
128 *
129 * This involves IPC and may or may not also require re-enumeration of the
130 * device. Which means that it might transition directly into the ReleasingToHost state
131 * because the client (VM) will do the actual re-enumeration.
132 *
133 * Next Host: PhysDetachingFromVM (?) or just PhysDetached.
134 * Next VBox: ReleasingToHost, HeldByProxy.
135 */
136 kHostUSBDeviceState_DetachingFromVM,
137
138 /** The device has been physically removed while a VM used it.
139 *
140 * This is the device state while VBoxSVC is doing IPC to the client (VM) telling it
141 * to detach it.
142 *
143 * Next Host: None.
144 * Next VBox: PhysDetached
145 */
146 kHostUSBDeviceState_PhysDetachingFromVM,
147
148 /** Just an invalid state value for use as default for some methods. */
149 kHostUSBDeviceState_Invalid = 0x7fff
150} HostUSBDeviceState;
151
152
153/**
154 * Sub-state for dealing with device re-enumeration.
155 */
156typedef enum
157{
158 /** Not in any sub-state. */
159 kHostUSBDeviceSubState_Default = 0,
160 /** Awaiting a logical device detach following a device re-enumeration. */
161 kHostUSBDeviceSubState_AwaitingDetach,
162 /** Awaiting a logical device re-attach following a device re-enumeration. */
163 kHostUSBDeviceSubState_AwaitingReAttach
164} HostUSBDeviceSubState;
165
166
167/**
168 * Object class used to hold Host USB Device properties.
169 */
170class ATL_NO_VTABLE HostUSBDevice :
171 public HostUSBDeviceWrap
172{
173public:
174 DECLARE_COMMON_CLASS_METHODS(HostUSBDevice)
175
176 HRESULT FinalConstruct();
177 void FinalRelease();
178
179 // public initializer/uninitializer for internal purposes only
180 HRESULT init(PUSBDEVICE aUsb, USBProxyBackend *aUSBProxyBackend);
181 void uninit();
182
183 // public methods only for internal purposes
184
185 /** @note Must be called from under the object read lock. */
186 const Guid& i_getId() const { return mId; }
187
188 /** @note Must be called from under the object read lock. */
189 HostUSBDeviceState i_getUnistate() const { return mUniState; }
190
191 /** @note Must be called from under the object read lock. */
192 const char *i_getStateName() { return i_stateName (mUniState, mPendingUniState, mUniSubState); }
193
194 /** @note Must be called from under the object read lock. */
195 bool i_isCapturableOrHeld()
196 {
197 return mUniState == kHostUSBDeviceState_Unused
198 || mUniState == kHostUSBDeviceState_Capturable
199 || mUniState == kHostUSBDeviceState_HeldByProxy;
200 }
201
202 /** @note Must be called from under the object read lock. */
203 ComObjPtr<SessionMachine> &i_getMachine() { return mMachine; }
204
205 /** @note Must be called from under the object read lock. */
206 PCUSBDEVICE i_getUsbData() const { return mUsb; }
207
208 USBProxyBackend *i_getUsbProxyBackend() const { return mUSBProxyBackend; }
209
210 com::Utf8Str i_getName();
211
212 HRESULT i_requestCaptureForVM(SessionMachine *aMachine, bool aSetError,
213 const com::Utf8Str &aCaptureFilename, ULONG aMaskedIfs = 0);
214 HRESULT i_onDetachFromVM(SessionMachine *aMachine, bool aDone, bool *aRunFilters, bool aAbnormal = false);
215 HRESULT i_requestReleaseToHost();
216 HRESULT i_requestHold();
217 bool i_wasActuallyDetached();
218 void i_onPhysicalDetached();
219
220 bool i_isMatch(const USBDeviceFilter::BackupableUSBDeviceFilterData &aData);
221 int i_compare(PCUSBDEVICE aDev2);
222 static int i_compare(PCUSBDEVICE aDev1, PCUSBDEVICE aDev2, bool aIsAwaitingReAttach = false);
223
224 bool i_updateState(PCUSBDEVICE aDev, bool *aRunFilters, SessionMachine **aIgnoreMachine);
225 bool i_updateStateFake(PCUSBDEVICE aDev, bool *aRunFilters, SessionMachine **aIgnoreMachine);
226
227 static const char *i_stateName(HostUSBDeviceState aState,
228 HostUSBDeviceState aPendingState = kHostUSBDeviceState_Invalid,
229 HostUSBDeviceSubState aSubState = kHostUSBDeviceSubState_Default);
230
231 void *i_getBackendUserData() { return m_pvBackendUser; }
232 void i_setBackendUserData(void *pvBackendUser) { m_pvBackendUser = pvBackendUser; }
233
234protected:
235
236 HRESULT i_attachToVM(SessionMachine *aMachine, const com::Utf8Str &aCaptureFilename, ULONG aMaskedIfs = 0);
237 void i_detachFromVM(HostUSBDeviceState aFinalState);
238 void i_onPhysicalDetachedInternal();
239 bool i_hasAsyncOperationTimedOut() const;
240
241 bool i_setState (HostUSBDeviceState aNewState, HostUSBDeviceState aNewPendingState = kHostUSBDeviceState_Invalid,
242 HostUSBDeviceSubState aNewSubState = kHostUSBDeviceSubState_Default);
243 bool i_startTransition (HostUSBDeviceState aNewState, HostUSBDeviceState aFinalState,
244 HostUSBDeviceSubState aNewSubState = kHostUSBDeviceSubState_Default);
245 bool i_advanceTransition(bool aSkipReAttach = false);
246 bool i_failTransition(HostUSBDeviceState a_enmStateHint);
247 USBDeviceState_T i_canonicalState() const;
248
249private:
250
251 // wrapped IUSBDevice properties
252 HRESULT getId(com::Guid &aId);
253 HRESULT getVendorId(USHORT *aVendorId);
254 HRESULT getProductId(USHORT *aProductId);
255 HRESULT getRevision(USHORT *aRevision);
256 HRESULT getManufacturer(com::Utf8Str &aManufacturer);
257 HRESULT getProduct(com::Utf8Str &aProduct);
258 HRESULT getSerialNumber(com::Utf8Str &aSerialNumber);
259 HRESULT getAddress(com::Utf8Str &aAddress);
260 HRESULT getPort(USHORT *aPort);
261 HRESULT getPortPath(com::Utf8Str &aPortPath);
262 HRESULT getVersion(USHORT *aVersion);
263 HRESULT getPortVersion(USHORT *aPortVersion);
264 HRESULT getSpeed(USBConnectionSpeed_T *aSpeed);
265 HRESULT getRemote(BOOL *aRemote);
266 HRESULT getState(USBDeviceState_T *aState);
267 HRESULT getBackend(com::Utf8Str &aBackend);
268 HRESULT getDeviceInfo(std::vector<com::Utf8Str> &aInfo);
269
270
271 const Guid mId;
272
273 /** @name The state machine variables
274 * Only setState(), init() and uninit() will modify these members!
275 * @{ */
276 /** The RTTimeNanoTS() corresponding to the last state change.
277 *
278 * Old state machine: RTTimeNanoTS() of when mIsStatePending was set or mDetaching changed
279 * from kNotDetaching. For operations that cannot be canceled it's 0. */
280 uint64_t mLastStateChangeTS;
281 /** Current state. */
282 HostUSBDeviceState mUniState;
283 /** Sub-state for tracking re-enumeration. */
284 HostUSBDeviceSubState mUniSubState;
285 /** The final state of an pending transition.
286 * This is mainly a measure to reduce the number of HostUSBDeviceState values. */
287 HostUSBDeviceState mPendingUniState;
288 /** Previous state.
289 * This is used for bailing out when a transition like capture fails. */
290 HostUSBDeviceState mPrevUniState;
291 /** Indicator set by onDetachedPhys and check when advancing a transitional state. */
292 bool mIsPhysicallyDetached;
293 /** @} */
294
295 /** The machine the usb device is (being) attached to. */
296 ComObjPtr<SessionMachine> mMachine;
297 /** Pointer to the USB Proxy Backend instance. */
298 USBProxyBackend *mUSBProxyBackend;
299 /** Pointer to the USB Device structure owned by this device.
300 * Only used for host devices. */
301 PUSBDEVICE mUsb;
302 /** The interface mask to be used in the pending capture.
303 * This is a filter property. */
304 ULONG mMaskedIfs;
305 /** The name of this device. */
306 Utf8Str mNameObj;
307 /** The name of this device (for logging purposes).
308 * This points to the string in mNameObj. */
309 const char *mName;
310 /** The filename to capture the USB traffic to. */
311 com::Utf8Str mCaptureFilename;
312 /** Optional opaque user data assigned by the USB proxy backend owning the device. */
313 void *m_pvBackendUser;
314};
315
316#endif /* !MAIN_INCLUDED_HostUSBDeviceImpl_h */
317/* 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