VirtualBox

source: vbox/trunk/src/VBox/Main/include/USBProxyService.h@ 7936

Last change on this file since 7936 was 7852, checked in by vboxsync, 17 years ago

manual-cpp -DVBOX_WITH_USBFILTER. One worry, not sure why the Action setter call was omitted from the init. Hopefully just a left over.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 12.4 KB
Line 
1/* $Id: USBProxyService.h 7852 2008-04-09 17:17:24Z vboxsync $ */
2/** @file
3 * VirtualBox USB Proxy Service (base) class.
4 */
5
6/*
7 * Copyright (C) 2006-2007 innotek GmbH
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_USBPROXYSERVICE
20#define ____H_USBPROXYSERVICE
21
22#include "HostImpl.h"
23#include "HostUSBDeviceImpl.h"
24
25/**
26 * Base class for the USB Proxy service.
27 */
28class USBProxyService
29{
30public:
31 USBProxyService (Host *aHost);
32 virtual ~USBProxyService();
33
34 /**
35 * A filter was inserted / loaded.
36 *
37 * @param aFilter Pointer to the inserted filter.
38 * @return ID of the inserted filter
39 */
40 virtual void *insertFilter (PCUSBFILTER aFilter);
41
42 /**
43 * A filter was removed.
44 *
45 * @param aId ID of the filter to remove
46 */
47 virtual void removeFilter (void *aId);
48
49 /**
50 * A VM is trying to capture a device, do necessary preperations.
51 *
52 * @returns VBox status code.
53 * @param aDevice The device in question.
54 */
55 virtual int captureDevice (HostUSBDevice *aDevice);
56
57 /**
58 * The device is going to be detached from a VM.
59 *
60 * @param aDevice The device in question.
61 */
62 virtual void detachingDevice (HostUSBDevice *aDevice);
63
64 /**
65 * A VM is releasing a device back to the host.
66 *
67 * @returns VBox status code.
68 * @param aDevice The device in question.
69 */
70 virtual int releaseDevice (HostUSBDevice *aDevice);
71
72 /**
73 * Updates the device state.
74 * This is responsible for calling HostUSBDevice::updateState() and check for async completion.
75 *
76 * @returns true if there is a state change.
77 * @param aDevice The device in question.
78 * @param aUSBDevice The USB device structure for the last enumeration.
79 */
80 virtual bool updateDeviceState (HostUSBDevice *aDevice, PUSBDEVICE aUSBDevice);
81
82 /**
83 * Add device notification hook for the OS specific code.
84 *
85 * @param aDevice The device in question.
86 * @param aUSBDevice The USB device structure.
87 */
88 virtual void deviceAdded (HostUSBDevice *aDevice, PUSBDEVICE aUSBDevice);
89
90 /**
91 * Remove device notification hook for the OS specific code.
92 *
93 * @param aDevice The device in question.
94 */
95 virtual void deviceRemoved (HostUSBDevice *aDevice);
96
97 /**
98 * Query if the service is active and working.
99 *
100 * @returns true if the service is up running.
101 * @returns false if the service isn't running.
102 */
103 bool isActive (void);
104
105 /**
106 * Get last error.
107 * Can be used to check why the proxy !isActive() upon construction.
108 *
109 * @returns VBox status code.
110 */
111 int getLastError (void);
112
113 /**
114 * Calculate the hash of the serial string.
115 *
116 * 64bit FNV1a, chosen because it is designed to hash in to a power of two
117 * space, and is much quicker and simpler than, say, a half MD4.
118 *
119 * @returns the hash.
120 * @param aSerial The serial string.
121 */
122 static uint64_t calcSerialHash (const char *aSerial);
123
124 /**
125 * Initializes a filter with the data from the specified device.
126 *
127 * @param aFilter The filter to fill.
128 * @param aDevice The device to fill it with.
129 */
130 static void initFilterFromDevice (PUSBFILTER aFilter, HostUSBDevice *aDevice);
131
132protected:
133
134 /**
135 * Starts the service.
136 *
137 * @returns VBox status.
138 */
139 int start (void);
140
141 /**
142 * Stops the service.
143 *
144 * @returns VBox status.
145 */
146 int stop (void);
147
148 /**
149 * Wait for a change in the USB devices attached to the host.
150 *
151 * @returns VBox status (ignored).
152 * @param aMillies Number of milliseconds to wait.
153 */
154 virtual int wait (unsigned aMillies);
155
156 /**
157 * Interrupt any wait() call in progress.
158 *
159 * @returns VBox status.
160 */
161 virtual int interruptWait (void);
162
163 /**
164 * Get a list of USB device currently attached to the host.
165 *
166 * @returns Pointer to a list of USB devices.
167 * The list nodes are freed individually by calling freeDevice().
168 */
169 virtual PUSBDEVICE getDevices (void);
170
171 /**
172 * First call made on the service thread, use it to do
173 * thread initialization.
174 */
175 virtual void serviceThreadInit (void);
176
177 /**
178 * First call made on the service thread, use it to do
179 * thread termination.
180 */
181 virtual void serviceThreadTerm (void);
182
183 /**
184 * Implement fake capture, ++.
185 *
186 * @returns true if there is a state change.
187 * @param pDevice The device in question.
188 * @param pUSBDevice The USB device structure for the last enumeration.
189 */
190 bool updateDeviceStateFake (HostUSBDevice *aDevice, PUSBDEVICE aUSBDevice);
191
192
193public:
194 /**
195 * Free all the members of a USB interface returned by getDevice().
196 *
197 * @param pIf Pointer to the interface.
198 * @param cIfs Number of consecutive interfaces pIf points to
199 */
200 static void freeInterfaceMembers (PUSBINTERFACE pIf, unsigned cIfs);
201
202 /**
203 * Free all the members of a USB device returned by getDevice().
204 *
205 * @param pDevice Pointer to the device.
206 */
207 static void freeDeviceMembers (PUSBDEVICE pDevice);
208
209 /**
210 * Free one USB device returned by getDevice().
211 *
212 * @param pDevice Pointer to the device.
213 */
214 static void freeDevice (PUSBDEVICE pDevice);
215
216private:
217 /**
218 * Process any relevant changes in the attached USB devices.
219 */
220 void processChanges (void);
221
222 /**
223 * The service thread created by start().
224 *
225 * @param Thread The thread handle.
226 * @param pvUser Pointer to the USBProxyService instance.
227 */
228 static DECLCALLBACK (int) serviceThread (RTTHREAD Thread, void *pvUser);
229
230protected:
231 /** Pointer to the Host object. */
232 Host *mHost;
233 /** Thread handle of the service thread. */
234 RTTHREAD mThread;
235 /** Flag which stop() sets to cause serviceThread to return. */
236 bool volatile mTerminate;
237 /** List of smart HostUSBDevice pointers. */
238 typedef std::list <ComObjPtr <HostUSBDevice> > HostUSBDeviceList;
239 /** List of the known USB devices. */
240 HostUSBDeviceList mDevices;
241 /** VBox status code of the last failure.
242 * (Only used by start(), stop() and the child constructors.) */
243 int mLastError;
244};
245
246
247#ifdef VBOX_WITH_USB
248
249# ifdef RT_OS_DARWIN
250# include <VBox/param.h>
251# undef PAGE_SHIFT
252# undef PAGE_SIZE
253# define OSType Carbon_OSType
254# include <Carbon/Carbon.h>
255# undef OSType
256
257/**
258 * The Darwin hosted USB Proxy Service.
259 */
260class USBProxyServiceDarwin : public USBProxyService
261{
262public:
263 USBProxyServiceDarwin (Host *aHost);
264 ~USBProxyServiceDarwin();
265
266#ifdef VBOX_WITH_NEW_USB_CODE_ON_DARWIN
267 virtual void *insertFilter (PCUSBFILTER aFilter);
268 virtual void removeFilter (void *aId);
269#endif
270
271 virtual int captureDevice (HostUSBDevice *aDevice);
272 virtual void detachingDevice (HostUSBDevice *aDevice);
273 virtual int releaseDevice (HostUSBDevice *aDevice);
274 virtual bool updateDeviceState (HostUSBDevice *aDevice, PUSBDEVICE aUSBDevice);
275
276protected:
277 virtual int wait (unsigned aMillies);
278 virtual int interruptWait (void);
279 virtual PUSBDEVICE getDevices (void);
280 virtual void serviceThreadInit (void);
281 virtual void serviceThreadTerm (void);
282
283private:
284 /** Reference to the runloop of the service thread.
285 * This is NULL if the service thread isn't running. */
286 CFRunLoopRef mServiceRunLoopRef;
287 /** The opaque value returned by DarwinSubscribeUSBNotifications. */
288 void *mNotifyOpaque;
289 /** A hack to work around the problem with the usb device enumeration
290 * not including newly attached devices. */
291 bool mWaitABitNextTime;
292#ifndef VBOX_WITH_NEW_USB_CODE_ON_DARWIN
293 /** Whether we've got a fake async event and should return without entering the runloop. */
294 bool volatile mFakeAsync;
295#endif
296 /** Whether we've successfully initialized the USBLib and should call USBLibTerm in the destructor. */
297 bool mUSBLibInitialized;
298};
299# endif /* RT_OS_DARWIN */
300
301
302# ifdef RT_OS_LINUX
303# include <stdio.h>
304
305/**
306 * The Linux hosted USB Proxy Service.
307 */
308class USBProxyServiceLinux : public USBProxyService
309{
310public:
311 USBProxyServiceLinux (Host *aHost, const char *aUsbfsRoot = "/proc/bus/usb");
312 ~USBProxyServiceLinux();
313
314 virtual int captureDevice (HostUSBDevice *aDevice);
315 virtual int releaseDevice (HostUSBDevice *aDevice);
316 virtual bool updateDeviceState (HostUSBDevice *aDevice, PUSBDEVICE aUSBDevice);
317 virtual void deviceAdded (HostUSBDevice *aDevice, PUSBDEVICE aUSBDevice);
318
319protected:
320 virtual int wait (unsigned aMillies);
321 virtual int interruptWait (void);
322 virtual PUSBDEVICE getDevices (void);
323 int addDeviceToChain (PUSBDEVICE pDev, PUSBDEVICE *ppFirst, PUSBDEVICE **pppNext, int rc);
324
325private:
326 /** File handle to the '/proc/bus/usb/devices' file. */
327 RTFILE mFile;
328 /** Stream for mFile. */
329 FILE *mStream;
330 /** Pipe used to interrupt wait(), the read end. */
331 RTFILE mWakeupPipeR;
332 /** Pipe used to interrupt wait(), the write end. */
333 RTFILE mWakeupPipeW;
334 /** The root of usbfs. */
335 Utf8Str mUsbfsRoot;
336 /** Number of 500ms polls left to do. See usbDeterminState for details. */
337 unsigned mUdevPolls;
338};
339# endif /* RT_OS_LINUX */
340
341
342# ifdef RT_OS_OS2
343# include <usbcalls.h>
344
345/**
346 * The Linux hosted USB Proxy Service.
347 */
348class USBProxyServiceOs2 : public USBProxyService
349{
350public:
351 USBProxyServiceOs2 (Host *aHost);
352 ~USBProxyServiceOs2();
353
354 virtual int captureDevice (HostUSBDevice *aDevice);
355 virtual int releaseDevice (HostUSBDevice *aDevice);
356 virtual bool updateDeviceState (HostUSBDevice *aDevice, PUSBDEVICE aUSBDevice);
357
358protected:
359 virtual int wait (unsigned aMillies);
360 virtual int interruptWait (void);
361 virtual PUSBDEVICE getDevices (void);
362 int addDeviceToChain (PUSBDEVICE pDev, PUSBDEVICE *ppFirst, PUSBDEVICE **pppNext, int rc);
363
364private:
365 /** The notification event semaphore */
366 HEV mhev;
367 /** The notification id. */
368 USBNOTIFY mNotifyId;
369 /** The usbcalls.dll handle. */
370 HMODULE mhmod;
371 /** UsbRegisterChangeNotification */
372 APIRET (APIENTRY *mpfnUsbRegisterChangeNotification)(PUSBNOTIFY, HEV, HEV);
373 /** UsbDeregisterNotification */
374 APIRET (APIENTRY *mpfnUsbDeregisterNotification)(USBNOTIFY);
375 /** UsbQueryNumberDevices */
376 APIRET (APIENTRY *mpfnUsbQueryNumberDevices)(PULONG);
377 /** UsbQueryDeviceReport */
378 APIRET (APIENTRY *mpfnUsbQueryDeviceReport)(ULONG, PULONG, PVOID);
379};
380# endif /* RT_OS_LINUX */
381
382
383# ifdef RT_OS_SOLARIS
384# include <libdevinfo.h>
385
386/**
387 * The Solaris hosted USB Proxy Service.
388 */
389class USBProxyServiceSolaris : public USBProxyService
390{
391public:
392 USBProxyServiceSolaris (Host *aHost);
393 ~USBProxyServiceSolaris();
394
395 virtual int captureDevice (HostUSBDevice *aDevice);
396 virtual int releaseDevice (HostUSBDevice *aDevice);
397 virtual bool updateDeviceState (HostUSBDevice *aDevice, PUSBDEVICE aUSBDevice);
398 virtual void deviceAdded (HostUSBDevice *aDevice, PUSBDEVICE aUSBDevice);
399
400protected:
401 virtual int wait (unsigned aMillies);
402 virtual int interruptWait (void);
403 virtual PUSBDEVICE getDevices (void);
404 int addDeviceToChain (PUSBDEVICE pDev, PUSBDEVICE *ppFirst, PUSBDEVICE **pppNext, int rc);
405
406private:
407 RTSEMEVENT mNotifyEventSem;
408};
409#endif /* RT_OS_SOLARIS */
410
411
412# ifdef RT_OS_WINDOWS
413/**
414 * The Win32 hosted USB Proxy Service.
415 */
416class USBProxyServiceWin32 : public USBProxyService
417{
418public:
419 USBProxyServiceWin32 (Host *aHost);
420 ~USBProxyServiceWin32();
421
422 virtual void *insertFilter (IUSBDeviceFilter *aFilter);
423 virtual void removeFilter (void *aID);
424
425 virtual int captureDevice (HostUSBDevice *aDevice);
426 virtual int releaseDevice (HostUSBDevice *aDevice);
427 virtual bool updateDeviceState (HostUSBDevice *aDevice, PUSBDEVICE aUSBDevice);
428
429protected:
430 virtual int wait (unsigned aMillies);
431 virtual int interruptWait (void);
432 virtual PUSBDEVICE getDevices (void);
433
434private:
435
436 HANDLE hEventInterrupt;
437};
438# endif /* RT_OS_WINDOWS */
439
440#endif /* VBOX_WITH_USB */
441
442
443#endif /* !____H_USBPROXYSERVICE */
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