- Timestamp:
- Sep 23, 2013 10:46:00 AM (11 years ago)
- Location:
- trunk/src/VBox/Main
- Files:
-
- 2 edited
Legend:
- Unmodified
- Added
- Removed
-
trunk/src/VBox/Main/include/EmulatedUSBImpl.h
r48406 r48631 42 42 void uninit(); 43 43 44 /* Public method for internal use. */ 45 static DECLCALLBACK(int) eusbCallback(void *pv, const char *pszId, uint32_t iEvent, 46 const void *pvData, uint32_t cbData); 47 44 48 private: 49 50 static DECLCALLBACK(int) eusbCallbackEMT(EmulatedUSB *pThis, char *pszId, uint32_t iEvent, 51 void *pvData, uint32_t cbData); 52 53 HRESULT webcamPathFromId(com::Utf8Str *pPath, const char *pszId); 45 54 46 55 // wrapped IEmulatedUSB properties -
trunk/src/VBox/Main/src-client/EmulatedUSBImpl.cpp
r48423 r48631 17 17 */ 18 18 19 #define LOG_GROUP_MAIN_OVERRIDE LOG_GROUP_MAIN_EMULATEDUSB 20 19 21 #include "EmulatedUSBImpl.h" 20 22 #include "ConsoleImpl.h" … … 41 43 int32_t volatile mcRefs; 42 44 45 EmulatedUSB *mpEmulatedUSB; 46 43 47 RTUUID mUuid; 48 char mszUuid[RTUUID_STR_LENGTH]; 44 49 45 50 Utf8Str mPath; … … 62 67 : 63 68 mcRefs(1), 69 mpEmulatedUSB(NULL), 64 70 enmStatus(EUSBDEVICE_CREATED) 65 71 { 66 72 RT_ZERO(mUuid); 73 RT_ZERO(mszUuid); 67 74 } 68 75 … … 82 89 83 90 HRESULT Initialize(Console *pConsole, 91 EmulatedUSB *pEmulatedUSB, 84 92 const com::Utf8Str *aPath, 85 93 const com::Utf8Str *aSettings); … … 89 97 PUVM pUVM); 90 98 99 bool HasId(const char *pszId) { return RTStrCmp(pszId, mszUuid) == 0;} 100 91 101 EUSBDEVICESTATUS enmStatus; 92 102 }; … … 102 112 for (it = pThis->mDevSettings.begin(); it != pThis->mDevSettings.end(); ++it) 103 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); 104 119 105 120 PCFGMNODE pLunL0; … … 112 127 113 128 int rc = PDMR3UsbCreateEmulatedDevice(pUVM, "Webcam", pInstance, &pThis->mUuid); 114 LogRel (("PDMR3UsbCreateEmulatedDevice %Rrc\n", rc));129 LogRelFlowFunc(("PDMR3UsbCreateEmulatedDevice %Rrc\n", rc)); 115 130 if (RT_FAILURE(rc) && pInstance) 116 131 CFGMR3RemoveNode(pInstance); … … 124 139 125 140 HRESULT EUSBWEBCAM::Initialize(Console *pConsole, 141 EmulatedUSB *pEmulatedUSB, 126 142 const com::Utf8Str *aPath, 127 143 const com::Utf8Str *aSettings) … … 132 148 if (RT_SUCCESS(vrc)) 133 149 { 150 RTStrPrintf(mszUuid, sizeof(mszUuid), "%RTuuid", &mUuid); 134 151 hrc = mPath.assignEx(*aPath); 135 152 if (SUCCEEDED(hrc)) … … 141 158 { 142 159 hrc = settingsParse(); 160 161 if (SUCCEEDED(hrc)) 162 { 163 mpEmulatedUSB = pEmulatedUSB; 164 } 143 165 } 144 166 } … … 311 333 if (p) 312 334 { 313 hrc = p->Initialize(m.pConsole, &path, &aSettings);335 hrc = p->Initialize(m.pConsole, this, &path, &aSettings); 314 336 if (SUCCEEDED(hrc)) 315 337 { … … 414 436 } 415 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 527 HRESULT 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 416 560 /* vi: set tabstop=4 shiftwidth=4 expandtab: */
Note:
See TracChangeset
for help on using the changeset viewer.