VirtualBox

source: vbox/trunk/src/VBox/Frontends/VBoxBFE/HostUSBImpl.cpp@ 21248

Last change on this file since 21248 was 19300, checked in by vboxsync, 16 years ago

VMReq,*: Replaced VMREQDEST with VMCPUID because it's a pain to have to cast CPU IDs all the time.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 9.8 KB
Line 
1/** @file
2 *
3 * VBox frontends: Basic Frontend (BFE):
4 * Implementation of HostUSB
5 */
6
7/*
8 * Copyright (C) 2006-2007 Sun Microsystems, Inc.
9 *
10 * This file is part of VirtualBox Open Source Edition (OSE), as
11 * available from http://www.virtualbox.org. This file is free software;
12 * you can redistribute it and/or modify it under the terms of the GNU
13 * General Public License (GPL) as published by the Free Software
14 * Foundation, in version 2 as it comes in the "COPYING" file of the
15 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
16 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
17 *
18 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
19 * Clara, CA 95054 USA or visit http://www.sun.com if you need
20 * additional information or have any questions.
21 */
22
23/* r=michael: I have removed almost all functionality from the Main
24 * version of this file, but left the structure as similar
25 * as I could in case we do need some of it some day. */
26
27#include <string>
28
29#ifdef RT_OS_LINUX
30#include <sys/types.h>
31#include <sys/stat.h>
32#include <unistd.h>
33#include <sys/ioctl.h>
34#include <fcntl.h>
35#include <mntent.h>
36/* bird: This is a hack to work around conflicts between these linux kernel headers
37 * and the GLIBC tcpip headers. They have different declarations of the 4
38 * standard byte order functions. */
39#define _LINUX_BYTEORDER_GENERIC_H
40#include <linux/cdrom.h>
41#include <errno.h>
42#endif /* RT_OS_LINUX */
43
44#include "HostUSBImpl.h"
45#include "HostUSBDeviceImpl.h"
46#include "USBProxyService.h"
47#include "Logging.h"
48
49#include <VBox/pdm.h>
50#include <VBox/vusb.h>
51#include <VBox/usb.h>
52#include <VBox/err.h>
53#include <iprt/string.h>
54#include <iprt/time.h>
55#include <stdio.h>
56
57#include <algorithm>
58
59// destructor
60/////////////////////////////////////////////////////////////////////////////
61
62HostUSB::~HostUSB()
63{
64 if (isReady())
65 uninit();
66}
67
68// public initializer/uninitializer for internal purposes only
69/////////////////////////////////////////////////////////////////////////////
70
71/**
72 * Initializes the host USB object.
73 *
74 * @returns COM result indicator
75 * @param parent handle of our parent object
76 */
77HRESULT HostUSB::init(PVM pVM)
78{
79 LogFlowMember(("HostUSB::init(): isReady=%d\n", isReady()));
80
81 ComAssertRet (!isReady(), E_UNEXPECTED);
82
83 /* Save pointer to the VM */
84 mpVM = pVM;
85
86/*
87#ifdef RT_OS_LINUX
88 mUSBProxyService = new USBProxyServiceLinux (this);
89#elif defined RT_OS_WINDOWS
90 mUSBProxyService = new USBProxyServiceWin32 (this);
91*/
92#ifdef RT_OS_L4
93 mUSBProxyService = new USBProxyServiceLinux (this);
94#else
95 mUSBProxyService = new USBProxyService (this);
96#endif
97 /** @todo handle !mUSBProxySerivce->isActive() and mUSBProxyService->getLastError()
98 * and somehow report or whatever that the proxy failed to startup.
99 * Also, there might be init order issues... */
100
101 setReady(true);
102 return S_OK;
103}
104
105/**
106 * Uninitializes the host object and sets the ready flag to FALSE.
107 * Called either from FinalRelease() or by the parent when it gets destroyed.
108 */
109void HostUSB::uninit()
110{
111 LogFlowMember(("HostUSB::uninit(): isReady=%d\n", isReady()));
112
113 AssertReturn (isReady(), (void) 0);
114
115 delete mUSBProxyService;
116 mUSBProxyService = NULL;
117
118 mUSBDevices.clear();
119
120 setReady (FALSE);
121}
122
123// private methods
124////////////////////////////////////////////////////////////////////////////////
125
126/**
127 * Called by USB proxy service when a new device is physically attached
128 * to the host.
129 *
130 * @param aDevice Pointer to the device which has been attached.
131 */
132void HostUSB::onUSBDeviceAttached (HostUSBDevice *aDevice)
133{
134 LogFlowMember (("HostUSB::onUSBDeviceAttached: aDevice=%p\n",
135 aDevice));
136 HostUSB::AutoLock alock (this);
137
138 // add to the collecion
139 mUSBDevices.push_back (aDevice);
140
141 // apply all filters (no need to lock the device, nobody can access it yet)
142 HRESULT rc = AttachUSBDevice(aDevice);
143 AssertComRC (rc);
144}
145
146/**
147 * Called by USB proxy service (?) when the device is physically detached
148 * from the host.
149 *
150 * @param aDevice Pointer to the device which has been detached.
151 */
152void HostUSB::onUSBDeviceDetached (HostUSBDevice *aDevice)
153{
154 LogFlowMember (("HostUSB::onUSBDeviceDetached: aDevice=%p\n",
155 &aDevice));
156 HostUSB::AutoLock alock (this);
157
158 RTUUID id = aDevice->id();
159
160 HostUSBDevice *device = 0;
161 HostUSB::USBDeviceList::iterator it = mUSBDevices.begin();
162 while (it != mUSBDevices.end())
163 {
164 if (RTUuidCompare(&(*it)->id(), &id) == 0)
165 {
166 device = (*it);
167 break;
168 }
169 ++ it;
170 }
171
172 AssertReturn (!!device, (void) 0);
173
174 // remove from the collecion
175 mUSBDevices.erase (it);
176
177 if (device->isCaptured())
178 {
179 // the device is captured, release it
180 alock.unlock();
181 HRESULT rc = DetachUSBDevice (device);
182 AssertComRC (rc);
183 }
184}
185
186/**
187 * Called by USB proxy service when the state of the host-driven device
188 * has changed because of non proxy interaction.
189 *
190 * @param aDevice The device in question.
191 */
192void HostUSB::onUSBDeviceStateChanged (HostUSBDevice *aDevice)
193{
194 LogFlowMember (("HostUSB::onUSBDeviceStateChanged: \n"));
195 HostUSB::AutoLock alock (this);
196
197 /** @todo dmik, is there anything we should do here? For instance if the device now is available? */
198}
199
200STDMETHODIMP HostUSB::AttachUSBDevice (HostUSBDevice *hostDevice)
201{
202 AutoLock alock (this);
203 /* This originally checked that the console object was ready.
204 * Unfortunately, this method now belongs to HostUSB, and can get
205 * called during construction - so before we are "ready". */
206// CHECK_READY();
207
208 /*
209 * Don't proceed unless we've found the usb controller.
210 */
211 if (!mpVM)
212 return setError (E_FAIL, tr ("VM is not powered up"));
213 PPDMIBASE pBase;
214 int vrc = PDMR3QueryLun (mpVM, "usb-ohci", 0, 0, &pBase);
215 if (RT_FAILURE (vrc))
216 return setError (E_FAIL, tr ("VM doesn't have a USB controller"));
217 /*
218 * Make sure that the device is in a captureable state
219 */
220 USBDeviceState_T eState = hostDevice->state();
221 if (eState != USBDeviceState_Busy &&
222 eState != USBDeviceState_Available &&
223 eState != USBDeviceState_Held)
224 return setError (E_FAIL,
225 tr ("Device is not in a capturable state"));
226 PVUSBIRHCONFIG pRhConfig = (PVUSBIRHCONFIG)pBase->pfnQueryInterface (pBase, PDMINTERFACE_VUSB_RH_CONFIG);
227 AssertReturn (pRhConfig, E_FAIL);
228
229 /*
230 * Get the address and the Uuid, and call the pfnCreateProxyDevice roothub method in EMT.
231 */
232 std::string Address;
233 HRESULT hrc = hostDevice->COMGETTER (Address) (&Address);
234 AssertComRC (hrc);
235
236 RTUUID Uuid;
237 hrc = hostDevice->COMGETTER (Id) (Uuid);
238 AssertComRC (hrc);
239
240 /* No remote devices for now */
241 BOOL fRemote = FALSE;
242 void *pvRemote = NULL;
243
244 LogFlowMember (("Console::AttachUSBDevice: Proxying USB device '%s' %RTuuid...\n", Address.c_str(), &Uuid));
245 PVMREQ pReq;
246 vrc = VMR3ReqCall (mpVM, VMCPUID_ANY, &pReq, RT_INDEFINITE_WAIT,
247 (PFNRT)pRhConfig->pfnCreateProxyDevice,
248 5, pRhConfig, &Uuid, fRemote,
249 Address.c_str(), pvRemote);
250 if (RT_SUCCESS (vrc))
251 vrc = pReq->iStatus;
252 VMR3ReqFree (pReq);
253 if (RT_SUCCESS (vrc))
254 hostDevice->setCaptured();
255 else
256 {
257 Log (("Console::AttachUSBDevice: Failed to create proxy device for '%s' %RTuuid, vrc=%Rrc\n", Address.c_str(),
258 &Uuid, vrc));
259 AssertRC (vrc);
260 /* michael: I presume this is not needed. */
261/* hrc = mControl->ReleaseUSBDevice (Uuid);
262 AssertComRC (hrc); */
263 switch (vrc)
264 {
265 case VERR_VUSB_NO_PORTS:
266 hrc = setError (E_FAIL, tr ("No available ports on the USB controller"));
267 break;
268 case VERR_VUSB_USBFS_PERMISSION:
269 hrc = setError (E_FAIL, tr ("Not permitted to open the USB device, check usbfs options"));
270 break;
271 default:
272 hrc = setError (E_FAIL, tr ("Failed to create USB proxy device: %Rrc"), vrc);
273 break;
274 }
275 return hrc;
276 }
277
278 return S_OK;
279}
280
281STDMETHODIMP HostUSB::DetachUSBDevice (HostUSBDevice *aDevice)
282{
283 AutoLock alock (this);
284 /* This originally checked that the console object was ready.
285 * Unfortunately, this method now belongs to HostUSB, and can get
286 * called during construction - so before we are "ready". */
287// CHECK_READY();
288
289 /*
290 * Detach the device from the VM
291 */
292 int vrc = VERR_PDM_DEVICE_NOT_FOUND;
293 if (mpVM)
294 {
295 PPDMIBASE pBase;
296 vrc = PDMR3QueryLun (mpVM, "usb-ohci", 0, 0, &pBase);
297 if (RT_SUCCESS (vrc))
298 {
299 PVUSBIRHCONFIG pRhConfig = (PVUSBIRHCONFIG)pBase->pfnQueryInterface (pBase, PDMINTERFACE_VUSB_RH_CONFIG);
300 Assert (pRhConfig);
301
302 RTUUID Uuid = aDevice->id();
303 LogFlowMember (("Console::DetachUSBDevice: Detaching USB proxy device %RTuuid...\n", &Uuid));
304 PVMREQ pReq;
305 vrc = VMR3ReqCall (mpVM, VMCPUID_ANY, &pReq, RT_INDEFINITE_WAIT, (PFNRT)pRhConfig->pfnDestroyProxyDevice,
306 2, pRhConfig, &Uuid);
307 if (RT_SUCCESS (vrc))
308 vrc = pReq->iStatus;
309 VMR3ReqFree (pReq);
310 }
311 }
312 if ( vrc == VERR_PDM_DEVICE_NOT_FOUND
313 || vrc == VERR_PDM_DEVICE_INSTANCE_NOT_FOUND)
314 {
315 Log (("Console::DetachUSBDevice: USB isn't enabled.\n"));
316 vrc = VINF_SUCCESS;
317 }
318 if (RT_SUCCESS (vrc))
319 return S_OK;
320 Log (("Console::AttachUSBDevice: Failed to detach the device from the USB controller, vrc=%Rrc.\n", vrc));
321 return(setError (E_UNEXPECTED, tr ("Failed to destroy the USB proxy device: %Rrc"), vrc));
322}
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