VirtualBox

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

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

build fix

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 14.2 KB
Line 
1/* $Id: USBProxyBackend.h 60108 2016-03-19 10:43:52Z 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 /** @todo unused */
70 virtual void detachingDevice(HostUSBDevice *aDevice);
71 virtual int releaseDevice(HostUSBDevice *aDevice);
72 virtual void releaseDeviceCompleted(HostUSBDevice *aDevice, bool aSuccess);
73 /** @} */
74
75 static void freeDevice(PUSBDEVICE pDevice);
76
77 HRESULT runAllFiltersOnDevice(ComObjPtr<HostUSBDevice> &aDevice,
78 SessionMachinesList &llOpenedMachines,
79 SessionMachine *aIgnoreMachine);
80 bool runMachineFilters(SessionMachine *aMachine, ComObjPtr<HostUSBDevice> &aDevice);
81
82 virtual void deviceAdded(ComObjPtr<HostUSBDevice> &aDevice, SessionMachinesList &llOpenedMachines, PUSBDEVICE aUSBDevice);
83 virtual void deviceRemoved(ComObjPtr<HostUSBDevice> &aDevice);
84 virtual void deviceChanged(ComObjPtr<HostUSBDevice> &aDevice, SessionMachinesList *pllOpenedMachines, SessionMachine *aIgnoreMachine);
85 virtual bool updateDeviceState(HostUSBDevice *aDevice, PUSBDEVICE aUSBDevice, bool *aRunFilters, SessionMachine **aIgnoreMachine);
86
87protected:
88 int start(void);
89 int stop(void);
90 virtual void serviceThreadInit(void);
91 virtual void serviceThreadTerm(void);
92
93 virtual int wait(RTMSINTERVAL aMillies);
94 virtual int interruptWait(void);
95 virtual PUSBDEVICE getDevices(void);
96 bool updateDeviceStateFake(HostUSBDevice *aDevice, PUSBDEVICE aUSBDevice, bool *aRunFilters, SessionMachine **aIgnoreMachine);
97 uint32_t incRef();
98 uint32_t decRef();
99
100 static HRESULT setError(HRESULT aResultCode, const char *aText, ...);
101
102 static void initFilterFromDevice(PUSBFILTER aFilter, HostUSBDevice *aDevice);
103 static void freeDeviceMembers(PUSBDEVICE pDevice);
104
105private:
106
107 // wrapped IUSBProxyBackend properties
108 HRESULT getName(com::Utf8Str &aName);
109 HRESULT getType(com::Utf8Str &aType);
110
111 static DECLCALLBACK(int) serviceThread(RTTHREAD Thread, void *pvUser);
112
113protected:
114 /** Pointer to the owning USB Proxy Service object. */
115 USBProxyService *m_pUsbProxyService;
116 /** Thread handle of the service thread. */
117 RTTHREAD mThread;
118 /** Flag which stop() sets to cause serviceThread to return. */
119 bool volatile mTerminate;
120 /** Id of the instance. */
121 const com::Utf8Str m_strId;
122 /** Address of the instance. */
123 const com::Utf8Str m_strAddress;
124 /** Backend identifier as used in the settings. */
125 const com::Utf8Str m_strBackend;
126 /** Reference counter which prevents the backend instance from being removed. */
127 uint32_t m_cRefs;
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 /** @todo unused */
156 virtual void detachingDevice(HostUSBDevice *aDevice);
157 virtual int releaseDevice(HostUSBDevice *aDevice);
158 virtual void releaseDeviceCompleted(HostUSBDevice *aDevice, bool aSuccess);
159
160protected:
161 virtual int wait(RTMSINTERVAL aMillies);
162 virtual int interruptWait (void);
163 virtual PUSBDEVICE getDevices (void);
164 virtual void serviceThreadInit (void);
165 virtual void serviceThreadTerm (void);
166 virtual bool updateDeviceState (HostUSBDevice *aDevice, PUSBDEVICE aUSBDevice, bool *aRunFilters, SessionMachine **aIgnoreMachine);
167
168private:
169 /** Reference to the runloop of the service thread.
170 * This is NULL if the service thread isn't running. */
171 CFRunLoopRef mServiceRunLoopRef;
172 /** The opaque value returned by DarwinSubscribeUSBNotifications. */
173 void *mNotifyOpaque;
174 /** A hack to work around the problem with the usb device enumeration
175 * not including newly attached devices. */
176 bool mWaitABitNextTime;
177 /** Whether we've successfully initialized the USBLib and should call USBLibTerm in the destructor. */
178 bool mUSBLibInitialized;
179};
180# endif /* RT_OS_DARWIN */
181
182
183# ifdef RT_OS_LINUX
184# include <stdio.h>
185# ifdef VBOX_USB_WITH_SYSFS
186# include <HostHardwareLinux.h>
187# endif
188
189/**
190 * The Linux hosted USB Proxy Backend.
191 */
192class USBProxyBackendLinux: public USBProxyBackend
193{
194public:
195 DECLARE_EMPTY_CTOR_DTOR (USBProxyBackendLinux)
196
197 int init(USBProxyService *pUsbProxyService, const com::Utf8Str &strId, const com::Utf8Str &strAddress);
198 void uninit();
199
200 virtual int captureDevice(HostUSBDevice *aDevice);
201 virtual int releaseDevice(HostUSBDevice *aDevice);
202
203protected:
204 int initUsbfs(void);
205 int initSysfs(void);
206 void doUsbfsCleanupAsNeeded(void);
207 virtual int wait(RTMSINTERVAL aMillies);
208 virtual int interruptWait(void);
209 virtual PUSBDEVICE getDevices(void);
210 virtual void deviceAdded(ComObjPtr<HostUSBDevice> &aDevice, SessionMachinesList &llOpenedMachines, PUSBDEVICE aUSBDevice);
211 virtual bool updateDeviceState(HostUSBDevice *aDevice, PUSBDEVICE aUSBDevice, bool *aRunFilters, SessionMachine **aIgnoreMachine);
212
213private:
214 int waitUsbfs(RTMSINTERVAL aMillies);
215 int waitSysfs(RTMSINTERVAL aMillies);
216
217private:
218 /** File handle to the '/proc/bus/usb/devices' file. */
219 RTFILE mhFile;
220 /** Pipe used to interrupt wait(), the read end. */
221 RTPIPE mhWakeupPipeR;
222 /** Pipe used to interrupt wait(), the write end. */
223 RTPIPE mhWakeupPipeW;
224 /** The root of usbfs. */
225 Utf8Str mDevicesRoot;
226 /** Whether we're using <mUsbfsRoot>/devices or /sys/whatever. */
227 bool mUsingUsbfsDevices;
228 /** Number of 500ms polls left to do. See usbDeterminState for details. */
229 unsigned mUdevPolls;
230# ifdef VBOX_USB_WITH_SYSFS
231 /** Object used for polling for hotplug events from hal. */
232 VBoxMainHotplugWaiter *mpWaiter;
233# endif
234};
235# endif /* RT_OS_LINUX */
236
237
238# ifdef RT_OS_OS2
239# include <usbcalls.h>
240
241/**
242 * The Linux hosted USB Proxy Backend.
243 */
244class USBProxyBackendOs2 : public USBProxyBackend
245{
246public:
247 DECLARE_EMPTY_CTOR_DTOR (USBProxyBackend)
248
249 virtual int captureDevice (HostUSBDevice *aDevice);
250 virtual int releaseDevice (HostUSBDevice *aDevice);
251
252protected:
253 virtual int wait(RTMSINTERVAL aMillies);
254 virtual int interruptWait(void);
255 virtual PUSBDEVICE getDevices(void);
256 int addDeviceToChain(PUSBDEVICE pDev, PUSBDEVICE *ppFirst, PUSBDEVICE **pppNext, int rc);
257 virtual bool updateDeviceState(HostUSBDevice *aDevice, PUSBDEVICE aUSBDevice, bool *aRunFilters, SessionMachine **aIgnoreMachine);
258
259private:
260 /** The notification event semaphore */
261 HEV mhev;
262 /** The notification id. */
263 USBNOTIFY mNotifyId;
264 /** The usbcalls.dll handle. */
265 HMODULE mhmod;
266 /** UsbRegisterChangeNotification */
267 APIRET (APIENTRY *mpfnUsbRegisterChangeNotification)(PUSBNOTIFY, HEV, HEV);
268 /** UsbDeregisterNotification */
269 APIRET (APIENTRY *mpfnUsbDeregisterNotification)(USBNOTIFY);
270 /** UsbQueryNumberDevices */
271 APIRET (APIENTRY *mpfnUsbQueryNumberDevices)(PULONG);
272 /** UsbQueryDeviceReport */
273 APIRET (APIENTRY *mpfnUsbQueryDeviceReport)(ULONG, PULONG, PVOID);
274};
275# endif /* RT_OS_LINUX */
276
277
278# ifdef RT_OS_SOLARIS
279# include <libdevinfo.h>
280
281/**
282 * The Solaris hosted USB Proxy Backend.
283 */
284class USBProxyBackendSolaris : public USBProxyBackend
285{
286public:
287 DECLARE_EMPTY_CTOR_DTOR (USBProxyBackendSolaris)
288
289 int init(USBProxyService *pUsbProxyService, const com::Utf8Str &strId, const com::Utf8Str &strAddress);
290 void uninit();
291
292 virtual void *insertFilter (PCUSBFILTER aFilter);
293 virtual void removeFilter (void *aID);
294
295 virtual int captureDevice (HostUSBDevice *aDevice);
296 virtual int releaseDevice (HostUSBDevice *aDevice);
297 virtual void captureDeviceCompleted(HostUSBDevice *aDevice, bool aSuccess);
298 virtual void releaseDeviceCompleted(HostUSBDevice *aDevice, bool aSuccess);
299
300protected:
301 virtual int wait(RTMSINTERVAL aMillies);
302 virtual int interruptWait(void);
303 virtual PUSBDEVICE getDevices(void);
304 virtual bool updateDeviceState(HostUSBDevice *aDevice, PUSBDEVICE aUSBDevice, bool *aRunFilters, SessionMachine **aIgnoreMachine);
305
306private:
307 RTSEMEVENT mNotifyEventSem;
308 /** Whether we've successfully initialized the USBLib and should call USBLibTerm in the destructor. */
309 bool mUSBLibInitialized;
310};
311#endif /* RT_OS_SOLARIS */
312
313
314# ifdef RT_OS_WINDOWS
315/**
316 * The Windows hosted USB Proxy Backend.
317 */
318class USBProxyBackendWindows : public USBProxyBackend
319{
320public:
321 DECLARE_EMPTY_CTOR_DTOR (USBProxyBackendWindows)
322
323 int init(USBProxyService *pUsbProxyService, const com::Utf8Str &strId, const com::Utf8Str &strAddress);
324 void uninit();
325
326 virtual void *insertFilter (PCUSBFILTER aFilter);
327 virtual void removeFilter (void *aID);
328
329 virtual int captureDevice (HostUSBDevice *aDevice);
330 virtual int releaseDevice (HostUSBDevice *aDevice);
331
332protected:
333 virtual int wait(RTMSINTERVAL aMillies);
334 virtual int interruptWait(void);
335 virtual PUSBDEVICE getDevices(void);
336 virtual bool updateDeviceState(HostUSBDevice *aDevice, PUSBDEVICE aUSBDevice, bool *aRunFilters, SessionMachine **aIgnoreMachine);
337
338private:
339
340 HANDLE mhEventInterrupt;
341};
342# endif /* RT_OS_WINDOWS */
343
344# ifdef RT_OS_FREEBSD
345/**
346 * The FreeBSD hosted USB Proxy Backend.
347 */
348class USBProxyBackendFreeBSD : public USBProxyBackend
349{
350public:
351 DECLARE_EMPTY_CTOR_DTOR (USBProxyBackendFreeBSD)
352
353 int init(USBProxyService *pUsbProxyService, const com::Utf8Str &strId, const com::Utf8Str &strAddress);
354 void uninit();
355
356 virtual int captureDevice(HostUSBDevice *aDevice);
357 virtual int releaseDevice(HostUSBDevice *aDevice);
358
359protected:
360 int initUsbfs(void);
361 int initSysfs(void);
362 virtual int wait(RTMSINTERVAL aMillies);
363 virtual int interruptWait(void);
364 virtual PUSBDEVICE getDevices(void);
365 int addDeviceToChain(PUSBDEVICE pDev, PUSBDEVICE *ppFirst, PUSBDEVICE **pppNext, int rc);
366 virtual void deviceAdded(ComObjPtr<HostUSBDevice> &aDevice, SessionMachinesList &llOpenedMachines, PUSBDEVICE aUSBDevice);
367 virtual bool updateDeviceState(HostUSBDevice *aDevice, PUSBDEVICE aUSBDevice, bool *aRunFilters, SessionMachine **aIgnoreMachine);
368
369private:
370 RTSEMEVENT mNotifyEventSem;
371};
372# endif /* RT_OS_FREEBSD */
373
374/**
375 * USB/IP Proxy receive state.
376 */
377typedef enum USBIPRECVSTATE
378{
379 /** Invalid state. */
380 kUsbIpRecvState_Invalid = 0,
381 /** There is no request waiting for an answer. */
382 kUsbIpRecvState_None,
383 /** Waiting for the complete reception of UsbIpRetDevList. */
384 kUsbIpRecvState_Hdr,
385 /** Waiting for the complete reception of a UsbIpExportedDevice structure. */
386 kUsbIpRecvState_ExportedDevice,
387 /** Waiting for a complete reception of a UsbIpDeviceInterface structure to skip. */
388 kUsbIpRecvState_DeviceInterface,
389 /** 32bit hack. */
390 kUsbIpRecvState_32Bit_Hack = 0x7fffffff
391} USBIPRECVSTATE;
392/** Pointer to a USB/IP receive state enum. */
393typedef USBIPRECVSTATE *PUSBIPRECVSTATE;
394
395struct UsbIpExportedDevice;
396
397/**
398 * The USB/IP Proxy Backend.
399 */
400class USBProxyBackendUsbIp: public USBProxyBackend
401{
402public:
403 DECLARE_EMPTY_CTOR_DTOR (USBProxyBackendUsbIp)
404
405 int init(USBProxyService *pUsbProxyService, const com::Utf8Str &strId, const com::Utf8Str &strAddress);
406 void uninit();
407
408 virtual int captureDevice(HostUSBDevice *aDevice);
409 virtual int releaseDevice(HostUSBDevice *aDevice);
410
411protected:
412 virtual int wait(RTMSINTERVAL aMillies);
413 virtual int interruptWait(void);
414 virtual PUSBDEVICE getDevices(void);
415 virtual bool updateDeviceState(HostUSBDevice *aDevice, PUSBDEVICE aUSBDevice, bool *aRunFilters, SessionMachine **aIgnoreMachine);
416
417private:
418 int updateDeviceList(bool *pfDeviceListChanged);
419 bool hasDevListChanged(PUSBDEVICE pDevices);
420 void freeDeviceList(PUSBDEVICE pHead);
421 void resetRecvState();
422 int reconnect();
423 void disconnect();
424 int startListExportedDevicesReq();
425 void advanceState(USBIPRECVSTATE enmRecvState);
426 int receiveData();
427 int processData();
428 int addDeviceToList(UsbIpExportedDevice *pDev);
429
430 struct Data; // opaque data struct, defined in USBProxyBackendUsbIp.cpp
431 Data *m;
432};
433
434#endif /* !____H_USBPROXYBACKEND */
435
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