VirtualBox

source: vbox/trunk/src/VBox/Main/src-server/darwin/USBProxyBackendDarwin.cpp@ 60067

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

Main: Add API to IHost for adding and removing USB device sources in addition to the default host one (only USB/IP backend supported so far which will be used in the future for automatic USB testing). Add support for it in VBoxManage

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 7.9 KB
Line 
1/* $Id: USBProxyBackendDarwin.cpp 60067 2016-03-16 19:17:22Z vboxsync $ */
2/** @file
3 * VirtualBox USB Proxy Service (in VBoxSVC), Darwin Specialization.
4 */
5
6/*
7 * Copyright (C) 2005-2012 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/*********************************************************************************************************************************
20* Header Files *
21*********************************************************************************************************************************/
22#define LOG_GROUP LOG_GROUP_MAIN
23#include "USBProxyBackend.h"
24#include "Logging.h"
25#include "iokit.h"
26
27#include <VBox/usb.h>
28#include <VBox/usblib.h>
29#include <VBox/err.h>
30
31#include <iprt/string.h>
32#include <iprt/alloc.h>
33#include <iprt/assert.h>
34#include <iprt/file.h>
35#include <iprt/err.h>
36#include <iprt/asm.h>
37
38
39/**
40 * Initialize data members.
41 */
42USBProxyBackendDarwin::USBProxyBackendDarwin(USBProxyService *aUsbProxyService, const com::Utf8Str &strId)
43 : USBProxyBackend(aUsbProxyService, strId), mServiceRunLoopRef(NULL), mNotifyOpaque(NULL), mWaitABitNextTime(false), mUSBLibInitialized(false)
44{
45 LogFlowThisFunc(("aUsbProxyService=%p\n", aUsbProxyService));
46}
47
48
49/**
50 * Initializes the object (called right after construction).
51 *
52 * @returns VBox status code.
53 */
54int USBProxyBackendDarwin::init(const com::Utf8Str &strAddress)
55{
56 NOREF(strAddress);
57
58 /*
59 * Initialize the USB library.
60 */
61 int rc = USBLibInit();
62 if (RT_FAILURE(rc))
63 return rc;
64
65 mUSBLibInitialized = true;
66
67 /*
68 * Start the poller thread.
69 */
70 start();
71 return VINF_SUCCESS;
72}
73
74
75/**
76 * Stop all service threads and free the device chain.
77 */
78USBProxyBackendDarwin::~USBProxyBackendDarwin()
79{
80 LogFlowThisFunc(("\n"));
81
82 /*
83 * Stop the service.
84 */
85 if (isActive())
86 stop();
87
88 /*
89 * Terminate the USB library - it'll
90 */
91 if (mUSBLibInitialized)
92 {
93 USBLibTerm();
94 mUSBLibInitialized = false;
95 }
96}
97
98
99void *USBProxyBackendDarwin::insertFilter(PCUSBFILTER aFilter)
100{
101 return USBLibAddFilter(aFilter);
102}
103
104
105void USBProxyBackendDarwin::removeFilter(void *aId)
106{
107 USBLibRemoveFilter(aId);
108}
109
110
111int USBProxyBackendDarwin::captureDevice(HostUSBDevice *aDevice)
112{
113 /*
114 * Check preconditions.
115 */
116 AssertReturn(aDevice, VERR_GENERAL_FAILURE);
117 AssertReturn(!aDevice->isWriteLockOnCurrentThread(), VERR_GENERAL_FAILURE);
118
119 AutoReadLock devLock(aDevice COMMA_LOCKVAL_SRC_POS);
120 LogFlowThisFunc(("aDevice=%s\n", aDevice->i_getName().c_str()));
121
122 Assert(aDevice->i_getUnistate() == kHostUSBDeviceState_Capturing);
123
124 /*
125 * Create a one-shot capture filter for the device (don't
126 * match on port) and trigger a re-enumeration of it.
127 */
128 USBFILTER Filter;
129 USBFilterInit(&Filter, USBFILTERTYPE_ONESHOT_CAPTURE);
130 initFilterFromDevice(&Filter, aDevice);
131
132 void *pvId = USBLibAddFilter(&Filter);
133 if (!pvId)
134 return VERR_GENERAL_FAILURE;
135
136 int rc = DarwinReEnumerateUSBDevice(aDevice->i_getUsbData());
137 if (RT_SUCCESS(rc))
138 aDevice->i_setBackendUserData(pvId);
139 else
140 {
141 USBLibRemoveFilter(pvId);
142 pvId = NULL;
143 }
144 LogFlowThisFunc(("returns %Rrc pvId=%p\n", rc, pvId));
145 return rc;
146}
147
148
149void USBProxyBackendDarwin::captureDeviceCompleted(HostUSBDevice *aDevice, bool aSuccess)
150{
151 AssertReturnVoid(aDevice->isWriteLockOnCurrentThread());
152
153 /*
154 * Remove the one-shot filter if necessary.
155 */
156 LogFlowThisFunc(("aDevice=%s aSuccess=%RTbool mOneShotId=%p\n", aDevice->i_getName().c_str(), aSuccess, aDevice->i_getBackendUserData()));
157 if (!aSuccess && aDevice->i_getBackendUserData())
158 USBLibRemoveFilter(aDevice->i_getBackendUserData());
159 aDevice->i_setBackendUserData(NULL);
160 USBProxyBackend::captureDeviceCompleted(aDevice, aSuccess);
161}
162
163
164int USBProxyBackendDarwin::releaseDevice(HostUSBDevice *aDevice)
165{
166 /*
167 * Check preconditions.
168 */
169 AssertReturn(aDevice, VERR_GENERAL_FAILURE);
170 AssertReturn(!aDevice->isWriteLockOnCurrentThread(), VERR_GENERAL_FAILURE);
171
172 AutoReadLock devLock(aDevice COMMA_LOCKVAL_SRC_POS);
173 LogFlowThisFunc(("aDevice=%s\n", aDevice->i_getName().c_str()));
174
175 Assert(aDevice->i_getUnistate() == kHostUSBDeviceState_ReleasingToHost);
176
177 /*
178 * Create a one-shot ignore filter for the device
179 * and trigger a re-enumeration of it.
180 */
181 USBFILTER Filter;
182 USBFilterInit(&Filter, USBFILTERTYPE_ONESHOT_IGNORE);
183 initFilterFromDevice(&Filter, aDevice);
184 Log(("USBFILTERIDX_PORT=%#x\n", USBFilterGetNum(&Filter, USBFILTERIDX_PORT)));
185 Log(("USBFILTERIDX_BUS=%#x\n", USBFilterGetNum(&Filter, USBFILTERIDX_BUS)));
186
187 void *pvId = USBLibAddFilter(&Filter);
188 if (!pvId)
189 return VERR_GENERAL_FAILURE;
190
191 int rc = DarwinReEnumerateUSBDevice(aDevice->i_getUsbData());
192 if (RT_SUCCESS(rc))
193 aDevice->i_setBackendUserData(pvId);
194 else
195 {
196 USBLibRemoveFilter(pvId);
197 pvId = NULL;
198 }
199 LogFlowThisFunc(("returns %Rrc pvId=%p\n", rc, pvId));
200 return rc;
201}
202
203
204void USBProxyBackendDarwin::releaseDeviceCompleted(HostUSBDevice *aDevice, bool aSuccess)
205{
206 AssertReturnVoid(aDevice->isWriteLockOnCurrentThread());
207
208 /*
209 * Remove the one-shot filter if necessary.
210 */
211 LogFlowThisFunc(("aDevice=%s aSuccess=%RTbool mOneShotId=%p\n", aDevice->i_getName().c_str(), aSuccess, aDevice->i_getBackendUserData()));
212 if (!aSuccess && aDevice->i_getBackendUserData())
213 USBLibRemoveFilter(aDevice->i_getBackendUserData());
214 aDevice->i_setBackendUserData(NULL);
215 USBProxyBackend::releaseDeviceCompleted(aDevice, aSuccess);
216}
217
218
219/** @todo unused */
220void USBProxyBackendDarwin::detachingDevice(HostUSBDevice *aDevice)
221{
222 NOREF(aDevice);
223}
224
225
226bool USBProxyBackendDarwin::updateDeviceState(HostUSBDevice *aDevice, PUSBDEVICE aUSBDevice, bool *aRunFilters, SessionMachine **aIgnoreMachine)
227{
228 AssertReturn(aDevice, false);
229 AssertReturn(!aDevice->isWriteLockOnCurrentThread(), false);
230 /* Nothing special here so far, so fall back on parent. */
231 return USBProxyBackend::updateDeviceState(aDevice, aUSBDevice, aRunFilters, aIgnoreMachine);
232}
233
234
235int USBProxyBackendDarwin::wait(RTMSINTERVAL aMillies)
236{
237 SInt32 rc = CFRunLoopRunInMode(CFSTR(VBOX_IOKIT_MODE_STRING),
238 mWaitABitNextTime && aMillies >= 1000
239 ? 1.0 /* seconds */
240 : aMillies >= 5000 /* Temporary measure to poll for status changes (MSD). */
241 ? 5.0 /* seconds */
242 : aMillies / 1000.0,
243 true);
244 mWaitABitNextTime = rc != kCFRunLoopRunTimedOut;
245
246 return VINF_SUCCESS;
247}
248
249
250int USBProxyBackendDarwin::interruptWait(void)
251{
252 if (mServiceRunLoopRef)
253 CFRunLoopStop(mServiceRunLoopRef);
254 return 0;
255}
256
257
258PUSBDEVICE USBProxyBackendDarwin::getDevices(void)
259{
260 /* call iokit.cpp */
261 return DarwinGetUSBDevices();
262}
263
264
265void USBProxyBackendDarwin::serviceThreadInit(void)
266{
267 mServiceRunLoopRef = CFRunLoopGetCurrent();
268 mNotifyOpaque = DarwinSubscribeUSBNotifications();
269}
270
271
272void USBProxyBackendDarwin::serviceThreadTerm(void)
273{
274 DarwinUnsubscribeUSBNotifications(mNotifyOpaque);
275 mServiceRunLoopRef = NULL;
276}
277
278
279/**
280 * Wrapper called from iokit.cpp.
281 *
282 * @param pCur The USB device to free.
283 */
284void DarwinFreeUSBDeviceFromIOKit(PUSBDEVICE pCur)
285{
286 USBProxyBackend::freeDevice(pCur);
287}
288
Note: See TracBrowser for help on using the repository browser.

© 2025 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette