VirtualBox

source: vbox/trunk/src/VBox/Main/include/USBProxyBackend.h@ 60744

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

Main/USBProxyBackend: Add flag for each backend indicating whether we have to fake the device state update or Linux and USB/IP backends don't work

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 12.8 KB
Line 
1/* $Id: USBProxyBackend.h 60744 2016-04-28 15:30:02Z vboxsync $ */
2/** @file
3 * VirtualBox USB Proxy Backend (base) class.
4 */
5
6/*
7 * Copyright (C) 2005-2015 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 */
17
18
19#ifndef ____H_USBPROXYBACKEND
20#define ____H_USBPROXYBACKEND
21
22#include <VBox/usb.h>
23#include <VBox/usbfilter.h>
24
25#include <iprt/socket.h>
26#include <iprt/poll.h>
27#include <iprt/semaphore.h>
28#include <iprt/cpp/utils.h>
29
30#include "VirtualBoxBase.h"
31#include "VirtualBoxImpl.h"
32#include "HostUSBDeviceImpl.h"
33#include "USBProxyBackendWrap.h"
34class USBProxyService;
35
36/**
37 * Base class for the USB Proxy Backend.
38 */
39class ATL_NO_VTABLE USBProxyBackend
40 : public USBProxyBackendWrap
41{
42public:
43
44 DECLARE_EMPTY_CTOR_DTOR (USBProxyBackend)
45
46 HRESULT FinalConstruct();
47 void FinalRelease();
48
49 // public initializer/uninitializer for internal purposes only
50 virtual int init(USBProxyService *pUsbProxyService, const com::Utf8Str &strId, const com::Utf8Str &strAddress);
51 virtual void uninit();
52
53 bool isActive(void);
54 const com::Utf8Str &i_getId();
55 const com::Utf8Str &i_getAddress();
56 virtual const com::Utf8Str &i_getBackend();
57 uint32_t i_getRefCount();
58
59 /** @name Interface for the USBController and the Host object.
60 * @{ */
61 virtual void *insertFilter(PCUSBFILTER aFilter);
62 virtual void removeFilter(void *aId);
63 /** @} */
64
65 /** @name Interfaces for the HostUSBDevice
66 * @{ */
67 virtual int captureDevice(HostUSBDevice *aDevice);
68 virtual void captureDeviceCompleted(HostUSBDevice *aDevice, bool aSuccess);
69 virtual int releaseDevice(HostUSBDevice *aDevice);
70 virtual void releaseDeviceCompleted(HostUSBDevice *aDevice, bool aSuccess);
71 /** @} */
72
73 static void freeDevice(PUSBDEVICE pDevice);
74
75protected:
76 int start(void);
77 int stop(void);
78 virtual void serviceThreadInit(void);
79 virtual void serviceThreadTerm(void);
80
81 virtual int wait(RTMSINTERVAL aMillies);
82 virtual int interruptWait(void);
83 virtual PUSBDEVICE getDevices(void);
84 uint32_t incRef();
85 uint32_t decRef();
86
87 static HRESULT setError(HRESULT aResultCode, const char *aText, ...);
88
89 static void initFilterFromDevice(PUSBFILTER aFilter, HostUSBDevice *aDevice);
90 static void freeDeviceMembers(PUSBDEVICE pDevice);
91
92 /**
93 * Backend specific callback when a device was added.
94 * (Currently only Linux uses it to adjust the udev polling).
95 */
96 virtual void deviceAdded(ComObjPtr<HostUSBDevice> &aDevice, PUSBDEVICE pDev);
97 virtual bool isFakeUpdateRequired();
98
99private:
100
101 // wrapped IUSBProxyBackend properties
102 HRESULT getName(com::Utf8Str &aName);
103 HRESULT getType(com::Utf8Str &aType);
104
105 static DECLCALLBACK(int) serviceThread(RTTHREAD Thread, void *pvUser);
106
107 void updateDeviceList(PUSBDEVICE pDevices);
108
109protected:
110 /** Pointer to the owning USB Proxy Service object. */
111 USBProxyService *m_pUsbProxyService;
112 /** Thread handle of the service thread. */
113 RTTHREAD mThread;
114 /** Flag which stop() sets to cause serviceThread to return. */
115 bool volatile mTerminate;
116 /** Id of the instance. */
117 const com::Utf8Str m_strId;
118 /** Address of the instance. */
119 const com::Utf8Str m_strAddress;
120 /** Backend identifier as used in the settings. */
121 const com::Utf8Str m_strBackend;
122 /** Reference counter which prevents the backend instance from being removed. */
123 uint32_t m_cRefs;
124 /** List of smart HostUSBDevice pointers. */
125 typedef std::list<ComObjPtr<HostUSBDevice> > HostUSBDeviceList;
126 /** List of the known USB devices for this backend. */
127 HostUSBDeviceList m_llDevices;
128};
129
130
131# ifdef RT_OS_DARWIN
132# include <VBox/param.h>
133# undef PAGE_SHIFT
134# undef PAGE_SIZE
135# define OSType Carbon_OSType
136# include <Carbon/Carbon.h>
137# undef OSType
138
139/**
140 * The Darwin hosted USB Proxy Backend.
141 */
142class USBProxyBackendDarwin : public USBProxyBackend
143{
144public:
145 DECLARE_EMPTY_CTOR_DTOR (USBProxyBackendDarwin)
146
147 int init(USBProxyService *pUsbProxyService, const com::Utf8Str &strId, const com::Utf8Str &strAddress);
148 void uninit();
149
150 virtual void *insertFilter(PCUSBFILTER aFilter);
151 virtual void removeFilter(void *aId);
152
153 virtual int captureDevice(HostUSBDevice *aDevice);
154 virtual void captureDeviceCompleted(HostUSBDevice *aDevice, bool aSuccess);
155 virtual int releaseDevice(HostUSBDevice *aDevice);
156 virtual void releaseDeviceCompleted(HostUSBDevice *aDevice, bool aSuccess);
157
158protected:
159 virtual int wait(RTMSINTERVAL aMillies);
160 virtual int interruptWait (void);
161 virtual PUSBDEVICE getDevices (void);
162 virtual void serviceThreadInit (void);
163 virtual void serviceThreadTerm (void);
164
165private:
166 /** Reference to the runloop of the service thread.
167 * This is NULL if the service thread isn't running. */
168 CFRunLoopRef mServiceRunLoopRef;
169 /** The opaque value returned by DarwinSubscribeUSBNotifications. */
170 void *mNotifyOpaque;
171 /** A hack to work around the problem with the usb device enumeration
172 * not including newly attached devices. */
173 bool mWaitABitNextTime;
174 /** Whether we've successfully initialized the USBLib and should call USBLibTerm in the destructor. */
175 bool mUSBLibInitialized;
176};
177# endif /* RT_OS_DARWIN */
178
179
180# ifdef RT_OS_LINUX
181# include <stdio.h>
182# ifdef VBOX_USB_WITH_SYSFS
183# include <HostHardwareLinux.h>
184# endif
185
186/**
187 * The Linux hosted USB Proxy Backend.
188 */
189class USBProxyBackendLinux: public USBProxyBackend
190{
191public:
192 DECLARE_EMPTY_CTOR_DTOR (USBProxyBackendLinux)
193
194 int init(USBProxyService *pUsbProxyService, const com::Utf8Str &strId, const com::Utf8Str &strAddress);
195 void uninit();
196
197 virtual int captureDevice(HostUSBDevice *aDevice);
198 virtual int releaseDevice(HostUSBDevice *aDevice);
199
200protected:
201 int initUsbfs(void);
202 int initSysfs(void);
203 void doUsbfsCleanupAsNeeded(void);
204 virtual int wait(RTMSINTERVAL aMillies);
205 virtual int interruptWait(void);
206 virtual PUSBDEVICE getDevices(void);
207 virtual void deviceAdded(ComObjPtr<HostUSBDevice> &aDevice, PUSBDEVICE aUSBDevice);
208 virtual bool isFakeUpdateRequired();
209
210private:
211 int waitUsbfs(RTMSINTERVAL aMillies);
212 int waitSysfs(RTMSINTERVAL aMillies);
213
214private:
215 /** File handle to the '/proc/bus/usb/devices' file. */
216 RTFILE mhFile;
217 /** Pipe used to interrupt wait(), the read end. */
218 RTPIPE mhWakeupPipeR;
219 /** Pipe used to interrupt wait(), the write end. */
220 RTPIPE mhWakeupPipeW;
221 /** The root of usbfs. */
222 Utf8Str mDevicesRoot;
223 /** Whether we're using <mUsbfsRoot>/devices or /sys/whatever. */
224 bool mUsingUsbfsDevices;
225 /** Number of 500ms polls left to do. See usbDeterminState for details. */
226 unsigned mUdevPolls;
227# ifdef VBOX_USB_WITH_SYSFS
228 /** Object used for polling for hotplug events from hal. */
229 VBoxMainHotplugWaiter *mpWaiter;
230# endif
231};
232# endif /* RT_OS_LINUX */
233
234
235# ifdef RT_OS_OS2
236# include <usbcalls.h>
237
238/**
239 * The Linux hosted USB Proxy Backend.
240 */
241class USBProxyBackendOs2 : public USBProxyBackend
242{
243public:
244 DECLARE_EMPTY_CTOR_DTOR (USBProxyBackend)
245
246 virtual int captureDevice (HostUSBDevice *aDevice);
247 virtual int releaseDevice (HostUSBDevice *aDevice);
248
249protected:
250 virtual int wait(RTMSINTERVAL aMillies);
251 virtual int interruptWait(void);
252 virtual PUSBDEVICE getDevices(void);
253 int addDeviceToChain(PUSBDEVICE pDev, PUSBDEVICE *ppFirst, PUSBDEVICE **pppNext, int rc);
254
255private:
256 /** The notification event semaphore */
257 HEV mhev;
258 /** The notification id. */
259 USBNOTIFY mNotifyId;
260 /** The usbcalls.dll handle. */
261 HMODULE mhmod;
262 /** UsbRegisterChangeNotification */
263 APIRET (APIENTRY *mpfnUsbRegisterChangeNotification)(PUSBNOTIFY, HEV, HEV);
264 /** UsbDeregisterNotification */
265 APIRET (APIENTRY *mpfnUsbDeregisterNotification)(USBNOTIFY);
266 /** UsbQueryNumberDevices */
267 APIRET (APIENTRY *mpfnUsbQueryNumberDevices)(PULONG);
268 /** UsbQueryDeviceReport */
269 APIRET (APIENTRY *mpfnUsbQueryDeviceReport)(ULONG, PULONG, PVOID);
270};
271# endif /* RT_OS_LINUX */
272
273
274# ifdef RT_OS_SOLARIS
275# include <libdevinfo.h>
276
277/**
278 * The Solaris hosted USB Proxy Backend.
279 */
280class USBProxyBackendSolaris : public USBProxyBackend
281{
282public:
283 DECLARE_EMPTY_CTOR_DTOR (USBProxyBackendSolaris)
284
285 int init(USBProxyService *pUsbProxyService, const com::Utf8Str &strId, const com::Utf8Str &strAddress);
286 void uninit();
287
288 virtual void *insertFilter (PCUSBFILTER aFilter);
289 virtual void removeFilter (void *aID);
290
291 virtual int captureDevice (HostUSBDevice *aDevice);
292 virtual int releaseDevice (HostUSBDevice *aDevice);
293 virtual void captureDeviceCompleted(HostUSBDevice *aDevice, bool aSuccess);
294 virtual void releaseDeviceCompleted(HostUSBDevice *aDevice, bool aSuccess);
295
296protected:
297 virtual int wait(RTMSINTERVAL aMillies);
298 virtual int interruptWait(void);
299 virtual PUSBDEVICE getDevices(void);
300
301private:
302 RTSEMEVENT mNotifyEventSem;
303 /** Whether we've successfully initialized the USBLib and should call USBLibTerm in the destructor. */
304 bool mUSBLibInitialized;
305};
306#endif /* RT_OS_SOLARIS */
307
308
309# ifdef RT_OS_WINDOWS
310/**
311 * The Windows hosted USB Proxy Backend.
312 */
313class USBProxyBackendWindows : public USBProxyBackend
314{
315public:
316 DECLARE_EMPTY_CTOR_DTOR (USBProxyBackendWindows)
317
318 int init(USBProxyService *pUsbProxyService, const com::Utf8Str &strId, const com::Utf8Str &strAddress);
319 void uninit();
320
321 virtual void *insertFilter (PCUSBFILTER aFilter);
322 virtual void removeFilter (void *aID);
323
324 virtual int captureDevice (HostUSBDevice *aDevice);
325 virtual int releaseDevice (HostUSBDevice *aDevice);
326
327protected:
328 virtual int wait(RTMSINTERVAL aMillies);
329 virtual int interruptWait(void);
330 virtual PUSBDEVICE getDevices(void);
331
332private:
333
334 HANDLE mhEventInterrupt;
335};
336# endif /* RT_OS_WINDOWS */
337
338# ifdef RT_OS_FREEBSD
339/**
340 * The FreeBSD hosted USB Proxy Backend.
341 */
342class USBProxyBackendFreeBSD : public USBProxyBackend
343{
344public:
345 DECLARE_EMPTY_CTOR_DTOR (USBProxyBackendFreeBSD)
346
347 int init(USBProxyService *pUsbProxyService, const com::Utf8Str &strId, const com::Utf8Str &strAddress);
348 void uninit();
349
350 virtual int captureDevice(HostUSBDevice *aDevice);
351 virtual int releaseDevice(HostUSBDevice *aDevice);
352
353protected:
354 int initUsbfs(void);
355 int initSysfs(void);
356 virtual int wait(RTMSINTERVAL aMillies);
357 virtual int interruptWait(void);
358 virtual PUSBDEVICE getDevices(void);
359 int addDeviceToChain(PUSBDEVICE pDev, PUSBDEVICE *ppFirst, PUSBDEVICE **pppNext, int rc);
360 virtual void deviceAdded(ComObjPtr<HostUSBDevice> &aDevice, SessionMachinesList &llOpenedMachines, PUSBDEVICE aUSBDevice);
361
362private:
363 RTSEMEVENT mNotifyEventSem;
364};
365# endif /* RT_OS_FREEBSD */
366
367/**
368 * USB/IP Proxy receive state.
369 */
370typedef enum USBIPRECVSTATE
371{
372 /** Invalid state. */
373 kUsbIpRecvState_Invalid = 0,
374 /** There is no request waiting for an answer. */
375 kUsbIpRecvState_None,
376 /** Waiting for the complete reception of UsbIpRetDevList. */
377 kUsbIpRecvState_Hdr,
378 /** Waiting for the complete reception of a UsbIpExportedDevice structure. */
379 kUsbIpRecvState_ExportedDevice,
380 /** Waiting for a complete reception of a UsbIpDeviceInterface structure to skip. */
381 kUsbIpRecvState_DeviceInterface,
382 /** 32bit hack. */
383 kUsbIpRecvState_32Bit_Hack = 0x7fffffff
384} USBIPRECVSTATE;
385/** Pointer to a USB/IP receive state enum. */
386typedef USBIPRECVSTATE *PUSBIPRECVSTATE;
387
388struct UsbIpExportedDevice;
389
390/**
391 * The USB/IP Proxy Backend.
392 */
393class USBProxyBackendUsbIp: public USBProxyBackend
394{
395public:
396 DECLARE_EMPTY_CTOR_DTOR (USBProxyBackendUsbIp)
397
398 int init(USBProxyService *pUsbProxyService, const com::Utf8Str &strId, const com::Utf8Str &strAddress);
399 void uninit();
400
401 virtual int captureDevice(HostUSBDevice *aDevice);
402 virtual int releaseDevice(HostUSBDevice *aDevice);
403
404protected:
405 virtual int wait(RTMSINTERVAL aMillies);
406 virtual int interruptWait(void);
407 virtual PUSBDEVICE getDevices(void);
408 virtual bool isFakeUpdateRequired();
409
410private:
411 int updateDeviceList(bool *pfDeviceListChanged);
412 bool hasDevListChanged(PUSBDEVICE pDevices);
413 void freeDeviceList(PUSBDEVICE pHead);
414 void resetRecvState();
415 int reconnect();
416 void disconnect();
417 int startListExportedDevicesReq();
418 void advanceState(USBIPRECVSTATE enmRecvState);
419 int receiveData();
420 int processData();
421 int addDeviceToList(UsbIpExportedDevice *pDev);
422
423 struct Data; // opaque data struct, defined in USBProxyBackendUsbIp.cpp
424 Data *m;
425};
426
427#endif /* !____H_USBPROXYBACKEND */
428
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