VirtualBox

source: vbox/trunk/src/VBox/Main/src-client/EmulatedUSBImpl.cpp@ 48915

Last change on this file since 48915 was 48631, checked in by vboxsync, 11 years ago

Main: EmulatedUSB callback, detach device.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 13.7 KB
Line 
1/* $Id: EmulatedUSBImpl.cpp 48631 2013-09-23 10:46:00Z vboxsync $ */
2/** @file
3 *
4 * Emulated USB manager implementation.
5 */
6
7/*
8 * Copyright (C) 2013 Oracle Corporation
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
19#define LOG_GROUP_MAIN_OVERRIDE LOG_GROUP_MAIN_EMULATEDUSB
20
21#include "EmulatedUSBImpl.h"
22#include "ConsoleImpl.h"
23#include "Logging.h"
24
25#include <VBox/vmm/pdmusb.h>
26
27
28/*
29 * Emulated USB webcam device instance.
30 */
31typedef std::map <Utf8Str, Utf8Str> EUSBSettingsMap;
32
33typedef enum EUSBDEVICESTATUS
34{
35 EUSBDEVICE_CREATED,
36 EUSBDEVICE_ATTACHING,
37 EUSBDEVICE_ATTACHED
38} EUSBDEVICESTATUS;
39
40class EUSBWEBCAM /* : public EUSBDEVICE */
41{
42 private:
43 int32_t volatile mcRefs;
44
45 EmulatedUSB *mpEmulatedUSB;
46
47 RTUUID mUuid;
48 char mszUuid[RTUUID_STR_LENGTH];
49
50 Utf8Str mPath;
51 Utf8Str mSettings;
52
53 EUSBSettingsMap mDevSettings;
54 EUSBSettingsMap mDrvSettings;
55
56 static DECLCALLBACK(int) emulatedWebcamAttach(PUVM pUVM, EUSBWEBCAM *pThis);
57 static DECLCALLBACK(int) emulatedWebcamDetach(PUVM pUVM, EUSBWEBCAM *pThis);
58
59 HRESULT settingsParse(void);
60
61 ~EUSBWEBCAM()
62 {
63 }
64
65 public:
66 EUSBWEBCAM()
67 :
68 mcRefs(1),
69 mpEmulatedUSB(NULL),
70 enmStatus(EUSBDEVICE_CREATED)
71 {
72 RT_ZERO(mUuid);
73 RT_ZERO(mszUuid);
74 }
75
76 int32_t AddRef(void)
77 {
78 return ASMAtomicIncS32(&mcRefs);
79 }
80
81 void Release(void)
82 {
83 int32_t c = ASMAtomicDecS32(&mcRefs);
84 if (c == 0)
85 {
86 delete this;
87 }
88 }
89
90 HRESULT Initialize(Console *pConsole,
91 EmulatedUSB *pEmulatedUSB,
92 const com::Utf8Str *aPath,
93 const com::Utf8Str *aSettings);
94 HRESULT Attach(Console *pConsole,
95 PUVM pUVM);
96 HRESULT Detach(Console *pConsole,
97 PUVM pUVM);
98
99 bool HasId(const char *pszId) { return RTStrCmp(pszId, mszUuid) == 0;}
100
101 EUSBDEVICESTATUS enmStatus;
102};
103
104
105/* static */ DECLCALLBACK(int) EUSBWEBCAM::emulatedWebcamAttach(PUVM pUVM, EUSBWEBCAM *pThis)
106{
107 EUSBSettingsMap::const_iterator it;
108
109 PCFGMNODE pInstance = CFGMR3CreateTree(pUVM);
110 PCFGMNODE pConfig;
111 CFGMR3InsertNode(pInstance, "Config", &pConfig);
112 for (it = pThis->mDevSettings.begin(); it != pThis->mDevSettings.end(); ++it)
113 CFGMR3InsertString(pConfig, it->first.c_str(), it->second.c_str());
114 PCFGMNODE pEUSB;
115 CFGMR3InsertNode(pConfig, "EmulatedUSB", &pEUSB);
116 CFGMR3InsertString(pEUSB, "Id", pThis->mszUuid);
117 CFGMR3InsertInteger(pEUSB, "pfnCallback", (uintptr_t)EmulatedUSB::eusbCallback);
118 CFGMR3InsertInteger(pEUSB, "pvCallback", (uintptr_t)pThis->mpEmulatedUSB);
119
120 PCFGMNODE pLunL0;
121 CFGMR3InsertNode(pInstance, "LUN#0", &pLunL0);
122 CFGMR3InsertString(pLunL0, "Driver", "HostWebcam");
123 CFGMR3InsertNode(pLunL0, "Config", &pConfig);
124 CFGMR3InsertString(pConfig, "DevicePath", pThis->mPath.c_str());
125 for (it = pThis->mDrvSettings.begin(); it != pThis->mDrvSettings.end(); ++it)
126 CFGMR3InsertString(pConfig, it->first.c_str(), it->second.c_str());
127
128 int rc = PDMR3UsbCreateEmulatedDevice(pUVM, "Webcam", pInstance, &pThis->mUuid);
129 LogRelFlowFunc(("PDMR3UsbCreateEmulatedDevice %Rrc\n", rc));
130 if (RT_FAILURE(rc) && pInstance)
131 CFGMR3RemoveNode(pInstance);
132 return rc;
133}
134
135/* static */ DECLCALLBACK(int) EUSBWEBCAM::emulatedWebcamDetach(PUVM pUVM, EUSBWEBCAM *pThis)
136{
137 return PDMR3UsbDetachDevice(pUVM, &pThis->mUuid);
138}
139
140HRESULT EUSBWEBCAM::Initialize(Console *pConsole,
141 EmulatedUSB *pEmulatedUSB,
142 const com::Utf8Str *aPath,
143 const com::Utf8Str *aSettings)
144{
145 HRESULT hrc = S_OK;
146
147 int vrc = RTUuidCreate(&mUuid);
148 if (RT_SUCCESS(vrc))
149 {
150 RTStrPrintf(mszUuid, sizeof(mszUuid), "%RTuuid", &mUuid);
151 hrc = mPath.assignEx(*aPath);
152 if (SUCCEEDED(hrc))
153 {
154 hrc = mSettings.assignEx(*aSettings);
155 }
156
157 if (SUCCEEDED(hrc))
158 {
159 hrc = settingsParse();
160
161 if (SUCCEEDED(hrc))
162 {
163 mpEmulatedUSB = pEmulatedUSB;
164 }
165 }
166 }
167
168 if (SUCCEEDED(hrc) && RT_FAILURE(vrc))
169 {
170 LogFlowThisFunc(("%Rrc\n", vrc));
171 hrc = pConsole->setError(VBOX_E_IPRT_ERROR,
172 "Init emulated USB webcam (%Rrc)", vrc);
173 }
174
175 return hrc;
176}
177
178HRESULT EUSBWEBCAM::settingsParse(void)
179{
180 HRESULT hrc = S_OK;
181
182 return hrc;
183}
184
185HRESULT EUSBWEBCAM::Attach(Console *pConsole,
186 PUVM pUVM)
187{
188 HRESULT hrc = S_OK;
189
190 int vrc = VMR3ReqCallWaitU(pUVM, 0 /* idDstCpu (saved state, see #6232) */,
191 (PFNRT)emulatedWebcamAttach, 2,
192 pUVM, this);
193
194 if (SUCCEEDED(hrc) && RT_FAILURE(vrc))
195 {
196 LogFlowThisFunc(("%Rrc\n", vrc));
197 hrc = pConsole->setError(VBOX_E_IPRT_ERROR,
198 "Attach emulated USB webcam (%Rrc)", vrc);
199 }
200
201 return hrc;
202}
203
204HRESULT EUSBWEBCAM::Detach(Console *pConsole,
205 PUVM pUVM)
206{
207 HRESULT hrc = S_OK;
208
209 int vrc = VMR3ReqCallWaitU(pUVM, 0 /* idDstCpu (saved state, see #6232) */,
210 (PFNRT)emulatedWebcamDetach, 2,
211 pUVM, this);
212
213 if (SUCCEEDED(hrc) && RT_FAILURE(vrc))
214 {
215 LogFlowThisFunc(("%Rrc\n", vrc));
216 hrc = pConsole->setError(VBOX_E_IPRT_ERROR,
217 "Detach emulated USB webcam (%Rrc)", vrc);
218 }
219
220 return hrc;
221}
222
223
224/*
225 * EmulatedUSB implementation.
226 */
227DEFINE_EMPTY_CTOR_DTOR(EmulatedUSB)
228
229HRESULT EmulatedUSB::FinalConstruct()
230{
231 return BaseFinalConstruct();
232}
233
234void EmulatedUSB::FinalRelease()
235{
236 uninit();
237
238 BaseFinalRelease();
239}
240
241/*
242 * Initializes the instance.
243 *
244 * @param pConsole The owner.
245 */
246HRESULT EmulatedUSB::init(ComObjPtr<Console> pConsole)
247{
248 LogFlowThisFunc(("\n"));
249
250 ComAssertRet(!pConsole.isNull(), E_INVALIDARG);
251
252 /* Enclose the state transition NotReady->InInit->Ready */
253 AutoInitSpan autoInitSpan(this);
254 AssertReturn(autoInitSpan.isOk(), E_FAIL);
255
256 m.pConsole = pConsole;
257
258 /* Confirm a successful initialization */
259 autoInitSpan.setSucceeded();
260
261 return S_OK;
262}
263
264/*
265 * Uninitializes the instance.
266 * Called either from FinalRelease() or by the parent when it gets destroyed.
267 */
268void EmulatedUSB::uninit()
269{
270 LogFlowThisFunc(("\n"));
271
272 m.pConsole.setNull();
273
274 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
275 WebcamsMap::iterator it = m.webcams.begin();
276 while (it != m.webcams.end())
277 {
278 EUSBWEBCAM *p = it->second;
279 m.webcams.erase(it++);
280 p->Release();
281 }
282 alock.release();
283
284 /* Enclose the state transition Ready->InUninit->NotReady */
285 AutoUninitSpan autoUninitSpan(this);
286 if (autoUninitSpan.uninitDone())
287 return;
288}
289
290HRESULT EmulatedUSB::getWebcams(std::vector<com::Utf8Str> &aWebcams)
291{
292 HRESULT hrc = S_OK;
293
294 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
295
296 try
297 {
298 aWebcams.resize(m.webcams.size());
299 }
300 catch (std::bad_alloc &)
301 {
302 hrc = E_OUTOFMEMORY;
303 }
304 catch (...)
305 {
306 hrc = E_FAIL;
307 }
308
309 if (SUCCEEDED(hrc))
310 {
311 size_t i;
312 WebcamsMap::const_iterator it;
313 for (i = 0, it = m.webcams.begin(); it != m.webcams.end(); ++it)
314 aWebcams[i++] = it->first;
315 }
316
317 return hrc;
318}
319
320static const Utf8Str s_pathDefault(".0");
321
322HRESULT EmulatedUSB::webcamAttach(const com::Utf8Str &aPath,
323 const com::Utf8Str &aSettings)
324{
325 HRESULT hrc = S_OK;
326
327 const Utf8Str &path = aPath.isEmpty() || aPath == "."? s_pathDefault: aPath;
328
329 Console::SafeVMPtr ptrVM(m.pConsole);
330 if (ptrVM.isOk())
331 {
332 EUSBWEBCAM *p = new EUSBWEBCAM();
333 if (p)
334 {
335 hrc = p->Initialize(m.pConsole, this, &path, &aSettings);
336 if (SUCCEEDED(hrc))
337 {
338 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
339 WebcamsMap::const_iterator it = m.webcams.find(path);
340 if (it == m.webcams.end())
341 {
342 p->AddRef();
343 try
344 {
345 m.webcams[path] = p;
346 }
347 catch (std::bad_alloc &)
348 {
349 hrc = E_OUTOFMEMORY;
350 }
351 catch (...)
352 {
353 hrc = E_FAIL;
354 }
355 p->enmStatus = EUSBDEVICE_ATTACHING;
356 }
357 else
358 {
359 hrc = E_FAIL;
360 }
361 }
362
363 if (SUCCEEDED(hrc))
364 {
365 hrc = p->Attach(m.pConsole, ptrVM.rawUVM());
366 }
367
368 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
369 if (SUCCEEDED(hrc))
370 {
371 p->enmStatus = EUSBDEVICE_ATTACHED;
372 }
373 else
374 {
375 if (p->enmStatus != EUSBDEVICE_CREATED)
376 {
377 m.webcams.erase(path);
378 }
379 }
380 alock.release();
381
382 p->Release();
383 }
384 else
385 {
386 hrc = E_OUTOFMEMORY;
387 }
388 }
389 else
390 {
391 hrc = VBOX_E_INVALID_VM_STATE;
392 }
393
394 return hrc;
395}
396
397HRESULT EmulatedUSB::webcamDetach(const com::Utf8Str &aPath)
398{
399 HRESULT hrc = S_OK;
400
401 const Utf8Str &path = aPath.isEmpty() || aPath == "."? s_pathDefault: aPath;
402
403 Console::SafeVMPtr ptrVM(m.pConsole);
404 if (ptrVM.isOk())
405 {
406 EUSBWEBCAM *p = NULL;
407
408 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
409 WebcamsMap::iterator it = m.webcams.find(path);
410 if (it != m.webcams.end())
411 {
412 if (it->second->enmStatus == EUSBDEVICE_ATTACHED)
413 {
414 p = it->second;
415 m.webcams.erase(it);
416 }
417 }
418 alock.release();
419
420 if (p)
421 {
422 hrc = p->Detach(m.pConsole, ptrVM.rawUVM());
423 p->Release();
424 }
425 else
426 {
427 hrc = E_INVALIDARG;
428 }
429 }
430 else
431 {
432 hrc = VBOX_E_INVALID_VM_STATE;
433 }
434
435 return hrc;
436}
437
438/* static */ DECLCALLBACK(int) EmulatedUSB::eusbCallbackEMT(EmulatedUSB *pThis, char *pszId, uint32_t iEvent,
439 void *pvData, uint32_t cbData)
440{
441 LogRelFlowFunc(("id %s event %d, data %p %d\n", pszId, iEvent, pvData, cbData));
442
443 NOREF(cbData);
444
445 int rc = VINF_SUCCESS;
446 if (iEvent == 0)
447 {
448 com::Utf8Str path;
449 HRESULT hr = pThis->webcamPathFromId(&path, pszId);
450 if (SUCCEEDED(hr))
451 {
452 hr = pThis->webcamDetach(path);
453 if (FAILED(hr))
454 {
455 rc = VERR_INVALID_STATE;
456 }
457 }
458 else
459 {
460 rc = VERR_NOT_FOUND;
461 }
462 }
463 else
464 {
465 rc = VERR_INVALID_PARAMETER;
466 }
467
468 RTMemFree(pszId);
469 RTMemFree(pvData);
470
471 LogRelFlowFunc(("rc %Rrc\n", rc));
472 return rc;
473}
474
475/* static */ DECLCALLBACK(int) EmulatedUSB::eusbCallback(void *pv, const char *pszId, uint32_t iEvent,
476 const void *pvData, uint32_t cbData)
477{
478 /* Make a copy of parameters, forward to EMT and leave the callback to not hold any lock in the device. */
479 int rc = VINF_SUCCESS;
480
481 void *pvIdCopy = NULL;
482 void *pvDataCopy = NULL;
483 if (cbData > 0)
484 {
485 pvDataCopy = RTMemDup(pvData, cbData);
486 if (!pvDataCopy)
487 {
488 rc = VERR_NO_MEMORY;
489 }
490 }
491
492 if (RT_SUCCESS(rc))
493 {
494 pvIdCopy = RTMemDup(pszId, strlen(pszId) + 1);
495 if (!pvIdCopy)
496 {
497 rc = VERR_NO_MEMORY;
498 }
499 }
500
501 if (RT_SUCCESS(rc))
502 {
503 EmulatedUSB *pThis = (EmulatedUSB *)pv;
504 Console::SafeVMPtr ptrVM(pThis->m.pConsole);
505 if (ptrVM.isOk())
506 {
507 /* No wait. */
508 rc = VMR3ReqCallNoWaitU(ptrVM.rawUVM(), 0 /* idDstCpu */,
509 (PFNRT)EmulatedUSB::eusbCallbackEMT, 5,
510 pThis, pvIdCopy, iEvent, pvDataCopy, cbData);
511 }
512 else
513 {
514 rc = VERR_INVALID_STATE;
515 }
516 }
517
518 if (RT_FAILURE(rc))
519 {
520 RTMemFree(pvIdCopy);
521 RTMemFree(pvDataCopy);
522 }
523
524 return rc;
525}
526
527HRESULT EmulatedUSB::webcamPathFromId(com::Utf8Str *pPath, const char *pszId)
528{
529 HRESULT hr = S_OK;
530
531 Console::SafeVMPtr ptrVM(m.pConsole);
532 if (ptrVM.isOk())
533 {
534 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
535 WebcamsMap::const_iterator it;
536 for (it = m.webcams.begin(); it != m.webcams.end(); ++it)
537 {
538 EUSBWEBCAM *p = it->second;
539 if (p->HasId(pszId))
540 {
541 *pPath = it->first;
542 break;
543 }
544 }
545
546 if (it == m.webcams.end())
547 {
548 hr = E_FAIL;
549 }
550 alock.release();
551 }
552 else
553 {
554 hr = VBOX_E_INVALID_VM_STATE;
555 }
556
557 return hr;
558}
559
560/* vi: set tabstop=4 shiftwidth=4 expandtab: */
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