VirtualBox

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

Last change on this file since 49069 was 48969, checked in by vboxsync, 11 years ago

Main/EmulatedUSB: update.

  • 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 48969 2013-10-08 10:53:53Z 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 /* pInstance will be used by PDM and deallocated on error. */
129 int rc = PDMR3UsbCreateEmulatedDevice(pUVM, "Webcam", pInstance, &pThis->mUuid);
130 LogRelFlowFunc(("PDMR3UsbCreateEmulatedDevice %Rrc\n", rc));
131 return rc;
132}
133
134/* static */ DECLCALLBACK(int) EUSBWEBCAM::emulatedWebcamDetach(PUVM pUVM, EUSBWEBCAM *pThis)
135{
136 return PDMR3UsbDetachDevice(pUVM, &pThis->mUuid);
137}
138
139HRESULT EUSBWEBCAM::Initialize(Console *pConsole,
140 EmulatedUSB *pEmulatedUSB,
141 const com::Utf8Str *aPath,
142 const com::Utf8Str *aSettings)
143{
144 HRESULT hrc = S_OK;
145
146 int vrc = RTUuidCreate(&mUuid);
147 if (RT_SUCCESS(vrc))
148 {
149 RTStrPrintf(mszUuid, sizeof(mszUuid), "%RTuuid", &mUuid);
150 hrc = mPath.assignEx(*aPath);
151 if (SUCCEEDED(hrc))
152 {
153 hrc = mSettings.assignEx(*aSettings);
154 }
155
156 if (SUCCEEDED(hrc))
157 {
158 hrc = settingsParse();
159
160 if (SUCCEEDED(hrc))
161 {
162 mpEmulatedUSB = pEmulatedUSB;
163 }
164 }
165 }
166
167 if (SUCCEEDED(hrc) && RT_FAILURE(vrc))
168 {
169 LogFlowThisFunc(("%Rrc\n", vrc));
170 hrc = pConsole->setError(VBOX_E_IPRT_ERROR,
171 "Init emulated USB webcam (%Rrc)", vrc);
172 }
173
174 return hrc;
175}
176
177HRESULT EUSBWEBCAM::settingsParse(void)
178{
179 HRESULT hrc = S_OK;
180
181 return hrc;
182}
183
184HRESULT EUSBWEBCAM::Attach(Console *pConsole,
185 PUVM pUVM)
186{
187 HRESULT hrc = S_OK;
188
189 int vrc = VMR3ReqCallWaitU(pUVM, 0 /* idDstCpu (saved state, see #6232) */,
190 (PFNRT)emulatedWebcamAttach, 2,
191 pUVM, this);
192
193 if (SUCCEEDED(hrc) && RT_FAILURE(vrc))
194 {
195 LogFlowThisFunc(("%Rrc\n", vrc));
196 hrc = pConsole->setError(VBOX_E_IPRT_ERROR,
197 "Attach emulated USB webcam (%Rrc)", vrc);
198 }
199
200 return hrc;
201}
202
203HRESULT EUSBWEBCAM::Detach(Console *pConsole,
204 PUVM pUVM)
205{
206 HRESULT hrc = S_OK;
207
208 int vrc = VMR3ReqCallWaitU(pUVM, 0 /* idDstCpu (saved state, see #6232) */,
209 (PFNRT)emulatedWebcamDetach, 2,
210 pUVM, this);
211
212 if (SUCCEEDED(hrc) && RT_FAILURE(vrc))
213 {
214 LogFlowThisFunc(("%Rrc\n", vrc));
215 hrc = pConsole->setError(VBOX_E_IPRT_ERROR,
216 "Detach emulated USB webcam (%Rrc)", vrc);
217 }
218
219 return hrc;
220}
221
222
223/*
224 * EmulatedUSB implementation.
225 */
226DEFINE_EMPTY_CTOR_DTOR(EmulatedUSB)
227
228HRESULT EmulatedUSB::FinalConstruct()
229{
230 return BaseFinalConstruct();
231}
232
233void EmulatedUSB::FinalRelease()
234{
235 uninit();
236
237 BaseFinalRelease();
238}
239
240/*
241 * Initializes the instance.
242 *
243 * @param pConsole The owner.
244 */
245HRESULT EmulatedUSB::init(ComObjPtr<Console> pConsole)
246{
247 LogFlowThisFunc(("\n"));
248
249 ComAssertRet(!pConsole.isNull(), E_INVALIDARG);
250
251 /* Enclose the state transition NotReady->InInit->Ready */
252 AutoInitSpan autoInitSpan(this);
253 AssertReturn(autoInitSpan.isOk(), E_FAIL);
254
255 m.pConsole = pConsole;
256
257 /* Confirm a successful initialization */
258 autoInitSpan.setSucceeded();
259
260 return S_OK;
261}
262
263/*
264 * Uninitializes the instance.
265 * Called either from FinalRelease() or by the parent when it gets destroyed.
266 */
267void EmulatedUSB::uninit()
268{
269 LogFlowThisFunc(("\n"));
270
271 m.pConsole.setNull();
272
273 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
274 WebcamsMap::iterator it = m.webcams.begin();
275 while (it != m.webcams.end())
276 {
277 EUSBWEBCAM *p = it->second;
278 m.webcams.erase(it++);
279 p->Release();
280 }
281 alock.release();
282
283 /* Enclose the state transition Ready->InUninit->NotReady */
284 AutoUninitSpan autoUninitSpan(this);
285 if (autoUninitSpan.uninitDone())
286 return;
287}
288
289HRESULT EmulatedUSB::getWebcams(std::vector<com::Utf8Str> &aWebcams)
290{
291 HRESULT hrc = S_OK;
292
293 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
294
295 try
296 {
297 aWebcams.resize(m.webcams.size());
298 }
299 catch (std::bad_alloc &)
300 {
301 hrc = E_OUTOFMEMORY;
302 }
303 catch (...)
304 {
305 hrc = E_FAIL;
306 }
307
308 if (SUCCEEDED(hrc))
309 {
310 size_t i;
311 WebcamsMap::const_iterator it;
312 for (i = 0, it = m.webcams.begin(); it != m.webcams.end(); ++it)
313 aWebcams[i++] = it->first;
314 }
315
316 return hrc;
317}
318
319static const Utf8Str s_pathDefault(".0");
320
321HRESULT EmulatedUSB::webcamAttach(const com::Utf8Str &aPath,
322 const com::Utf8Str &aSettings)
323{
324 HRESULT hrc = S_OK;
325
326 const Utf8Str &path = aPath.isEmpty() || aPath == "."? s_pathDefault: aPath;
327
328 Console::SafeVMPtr ptrVM(m.pConsole);
329 if (ptrVM.isOk())
330 {
331 EUSBWEBCAM *p = new EUSBWEBCAM();
332 if (p)
333 {
334 hrc = p->Initialize(m.pConsole, this, &path, &aSettings);
335 if (SUCCEEDED(hrc))
336 {
337 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
338 WebcamsMap::const_iterator it = m.webcams.find(path);
339 if (it == m.webcams.end())
340 {
341 p->AddRef();
342 try
343 {
344 m.webcams[path] = p;
345 }
346 catch (std::bad_alloc &)
347 {
348 hrc = E_OUTOFMEMORY;
349 }
350 catch (...)
351 {
352 hrc = E_FAIL;
353 }
354 p->enmStatus = EUSBDEVICE_ATTACHING;
355 }
356 else
357 {
358 hrc = E_FAIL;
359 }
360 }
361
362 if (SUCCEEDED(hrc))
363 {
364 hrc = p->Attach(m.pConsole, ptrVM.rawUVM());
365 }
366
367 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
368 if (SUCCEEDED(hrc))
369 {
370 p->enmStatus = EUSBDEVICE_ATTACHED;
371 }
372 else
373 {
374 if (p->enmStatus != EUSBDEVICE_CREATED)
375 {
376 m.webcams.erase(path);
377 }
378 }
379 alock.release();
380
381 p->Release();
382 }
383 else
384 {
385 hrc = E_OUTOFMEMORY;
386 }
387 }
388 else
389 {
390 hrc = VBOX_E_INVALID_VM_STATE;
391 }
392
393 return hrc;
394}
395
396HRESULT EmulatedUSB::webcamDetach(const com::Utf8Str &aPath)
397{
398 HRESULT hrc = S_OK;
399
400 const Utf8Str &path = aPath.isEmpty() || aPath == "."? s_pathDefault: aPath;
401
402 Console::SafeVMPtr ptrVM(m.pConsole);
403 if (ptrVM.isOk())
404 {
405 EUSBWEBCAM *p = NULL;
406
407 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
408 WebcamsMap::iterator it = m.webcams.find(path);
409 if (it != m.webcams.end())
410 {
411 if (it->second->enmStatus == EUSBDEVICE_ATTACHED)
412 {
413 p = it->second;
414 m.webcams.erase(it);
415 }
416 }
417 alock.release();
418
419 if (p)
420 {
421 hrc = p->Detach(m.pConsole, ptrVM.rawUVM());
422 p->Release();
423 }
424 else
425 {
426 hrc = E_INVALIDARG;
427 }
428 }
429 else
430 {
431 hrc = VBOX_E_INVALID_VM_STATE;
432 }
433
434 return hrc;
435}
436
437/* static */ DECLCALLBACK(int) EmulatedUSB::eusbCallbackEMT(EmulatedUSB *pThis, char *pszId, uint32_t iEvent,
438 void *pvData, uint32_t cbData)
439{
440 LogRelFlowFunc(("id %s event %d, data %p %d\n", pszId, iEvent, pvData, cbData));
441
442 NOREF(cbData);
443
444 int rc = VINF_SUCCESS;
445 if (iEvent == 0)
446 {
447 com::Utf8Str path;
448 HRESULT hr = pThis->webcamPathFromId(&path, pszId);
449 if (SUCCEEDED(hr))
450 {
451 hr = pThis->webcamDetach(path);
452 if (FAILED(hr))
453 {
454 rc = VERR_INVALID_STATE;
455 }
456 }
457 else
458 {
459 rc = VERR_NOT_FOUND;
460 }
461 }
462 else
463 {
464 rc = VERR_INVALID_PARAMETER;
465 }
466
467 RTMemFree(pszId);
468 RTMemFree(pvData);
469
470 LogRelFlowFunc(("rc %Rrc\n", rc));
471 return rc;
472}
473
474/* static */ DECLCALLBACK(int) EmulatedUSB::eusbCallback(void *pv, const char *pszId, uint32_t iEvent,
475 const void *pvData, uint32_t cbData)
476{
477 /* Make a copy of parameters, forward to EMT and leave the callback to not hold any lock in the device. */
478 int rc = VINF_SUCCESS;
479
480 void *pvIdCopy = NULL;
481 void *pvDataCopy = NULL;
482 if (cbData > 0)
483 {
484 pvDataCopy = RTMemDup(pvData, cbData);
485 if (!pvDataCopy)
486 {
487 rc = VERR_NO_MEMORY;
488 }
489 }
490
491 if (RT_SUCCESS(rc))
492 {
493 pvIdCopy = RTMemDup(pszId, strlen(pszId) + 1);
494 if (!pvIdCopy)
495 {
496 rc = VERR_NO_MEMORY;
497 }
498 }
499
500 if (RT_SUCCESS(rc))
501 {
502 EmulatedUSB *pThis = (EmulatedUSB *)pv;
503 Console::SafeVMPtr ptrVM(pThis->m.pConsole);
504 if (ptrVM.isOk())
505 {
506 /* No wait. */
507 rc = VMR3ReqCallNoWaitU(ptrVM.rawUVM(), 0 /* idDstCpu */,
508 (PFNRT)EmulatedUSB::eusbCallbackEMT, 5,
509 pThis, pvIdCopy, iEvent, pvDataCopy, cbData);
510 }
511 else
512 {
513 rc = VERR_INVALID_STATE;
514 }
515 }
516
517 if (RT_FAILURE(rc))
518 {
519 RTMemFree(pvIdCopy);
520 RTMemFree(pvDataCopy);
521 }
522
523 return rc;
524}
525
526HRESULT EmulatedUSB::webcamPathFromId(com::Utf8Str *pPath, const char *pszId)
527{
528 HRESULT hr = S_OK;
529
530 Console::SafeVMPtr ptrVM(m.pConsole);
531 if (ptrVM.isOk())
532 {
533 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
534 WebcamsMap::const_iterator it;
535 for (it = m.webcams.begin(); it != m.webcams.end(); ++it)
536 {
537 EUSBWEBCAM *p = it->second;
538 if (p->HasId(pszId))
539 {
540 *pPath = it->first;
541 break;
542 }
543 }
544
545 if (it == m.webcams.end())
546 {
547 hr = E_FAIL;
548 }
549 alock.release();
550 }
551 else
552 {
553 hr = VBOX_E_INVALID_VM_STATE;
554 }
555
556 return hr;
557}
558
559/* 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