Changeset 29220 in vbox for trunk/src/VBox/HostServices
- Timestamp:
- May 7, 2010 3:09:53 PM (15 years ago)
- svn:sync-xref-src-repo-rev:
- 61300
- File:
-
- 1 edited
Legend:
- Unmodified
- Added
- Removed
-
trunk/src/VBox/HostServices/GuestControl/service.cpp
r29003 r29220 28 28 29 29 #include <VBox/log.h> 30 #include <iprt/asm.h> 30 #include <iprt/asm.h> /* For ASMBreakpoint(). */ 31 31 #include <iprt/assert.h> 32 32 #include <iprt/cpp/autores.h> … … 65 65 struct GuestCall 66 66 { 67 /** Client ID; a client can have multiple handles! */ 68 uint32_t mClientID; 67 69 /** The call handle */ 68 70 VBOXHGCMCALLHANDLE mHandle; … … 73 75 74 76 /** The standard constructor */ 75 GuestCall() : m Handle(0), mParms(NULL), mNumParms(0) {}77 GuestCall() : mClientID(0), mHandle(0), mParms(NULL), mNumParms(0) {} 76 78 /** The normal contructor */ 77 GuestCall(VBOXHGCMCALLHANDLE aHandle, VBOXHGCMSVCPARM aParms[], int cParms) 78 : mHandle(aHandle), mParms(aParms), mNumParms(cParms) {} 79 GuestCall(uint32_t aClientID, VBOXHGCMCALLHANDLE aHandle, 80 VBOXHGCMSVCPARM aParms[], uint32_t cParms) 81 : mClientID(aClientID), mHandle(aHandle), mParms(aParms), mNumParms(cParms) {} 79 82 }; 80 83 /** The guest call list type */ 81 typedef std::list <GuestCall> CallList; 84 typedef std::list< GuestCall > CallList; 85 typedef std::list< GuestCall >::iterator CallListIter; 86 typedef std::list< GuestCall >::const_iterator CallListIterConst; 82 87 83 88 /** … … 91 96 /** HGCM helper functions. */ 92 97 PVBOXHGCMSVCHELPERS mpHelpers; 93 /** @todo we should have classes for thread and request handler thread */94 /** Queue of outstanding property change notifications */95 RTREQQUEUE *mReqQueue;96 /** Request that we've left pending in a call to flushNotifications. */97 PRTREQ mPendingDummyReq;98 /** Thread for processing the request queue */99 RTTHREAD mReqThread;100 /** Tell the thread that it should exit */101 bool volatile mfExitThread;102 98 /** Callback function supplied by the host for notification of updates 103 99 * to properties */ … … 113 109 explicit Service(PVBOXHGCMSVCHELPERS pHelpers) 114 110 : mpHelpers(pHelpers) 115 , mPendingDummyReq(NULL)116 , mfExitThread(false)117 111 , mpfnHostCallback(NULL) 118 112 , mpvHostData(NULL) 119 113 { 120 int rc = RTReqCreateQueue(&mReqQueue);121 #ifndef VBOX_GUEST_CTRL_TEST_NOTHREAD122 if (RT_SUCCESS(rc))123 rc = RTThreadCreate(&mReqThread, reqThreadFn, this, 0,124 RTTHREADTYPE_MSG_PUMP, RTTHREADFLAGS_WAITABLE,125 "GuestCtrlReq");126 #endif127 if (RT_FAILURE(rc))128 throw rc;129 114 } 130 115 … … 231 216 int paramBufferAssign(PVBOXGUESTCTRPARAMBUFFER pBuf, uint32_t cParms, VBOXHGCMSVCPARM paParms[]); 232 217 int prepareExecute(uint32_t cParms, VBOXHGCMSVCPARM paParms[]); 233 static DECLCALLBACK(int) reqThreadFn(RTTHREAD ThreadSelf, void *pvUser);234 218 int clientConnect(uint32_t u32ClientID, void *pvClient); 235 219 int clientDisconnect(uint32_t u32ClientID, void *pvClient); 236 220 int sendHostCmdToGuest(HostCmd *pCmd, VBOXHGCMCALLHANDLE callHandle, uint32_t cParms, VBOXHGCMSVCPARM paParms[]); 237 int retrieveNextHostCmd( VBOXHGCMCALLHANDLE callHandle, uint32_t cParms, VBOXHGCMSVCPARM paParms[]);221 int retrieveNextHostCmd(uint32_t u32ClientID, VBOXHGCMCALLHANDLE callHandle, uint32_t cParms, VBOXHGCMSVCPARM paParms[]); 238 222 int notifyHost(uint32_t eFunction, uint32_t cParms, VBOXHGCMSVCPARM paParms[]); 239 223 int processHostCmd(uint32_t eFunction, uint32_t cParms, VBOXHGCMSVCPARM paParms[]); … … 246 230 247 231 248 /**249 * Thread function for processing the request queue250 * @copydoc FNRTTHREAD251 */252 /* static */253 DECLCALLBACK(int) Service::reqThreadFn(RTTHREAD ThreadSelf, void *pvUser)254 {255 SELF *pSelf = reinterpret_cast<SELF *>(pvUser);256 while (!pSelf->mfExitThread)257 RTReqProcess(pSelf->mReqQueue, RT_INDEFINITE_WAIT);258 return VINF_SUCCESS;259 }260 261 232 /** @todo Write some nice doc headers! */ 262 233 /* Stores a HGCM request in an internal buffer (pEx). Needs to be freed later using execBufferFree(). */ … … 265 236 AssertPtr(pBuf); 266 237 int rc = VINF_SUCCESS; 238 239 /* Paranoia. */ 240 if (cParms > 256) 241 cParms = 256; 267 242 268 243 /* … … 392 367 { 393 368 LogFlowFunc(("Client (%ld) disconnected\n", u32ClientID)); 369 /* 370 * Throw out all stale clients. 371 */ 372 CallListIter it = mClientList.begin(); 373 while (it != mClientList.end()) 374 { 375 if (it->mClientID == u32ClientID) 376 it = mClientList.erase(it); 377 else 378 it++; 379 } 394 380 return VINF_SUCCESS; 395 381 } … … 424 410 * defer the guest call until we have something from the host. 425 411 */ 426 int Service::retrieveNextHostCmd(VBOXHGCMCALLHANDLE callHandle, uint32_t cParms, VBOXHGCMSVCPARM paParms[]) 412 int Service::retrieveNextHostCmd(uint32_t u32ClientID, VBOXHGCMCALLHANDLE callHandle, 413 uint32_t cParms, VBOXHGCMSVCPARM paParms[]) 427 414 { 428 415 int rc = VINF_SUCCESS; … … 435 422 if (mHostCmds.empty()) /* If command list is empty, defer ... */ 436 423 { 437 mClientList.push_back(GuestCall( callHandle, paParms, cParms));424 mClientList.push_back(GuestCall(u32ClientID, callHandle, paParms, cParms)); 438 425 rc = VINF_HGCM_ASYNC_EXECUTE; 439 426 } … … 535 522 if (!fProcessed) 536 523 { 537 mHostCmds.push_back(newCmd); 538 524 mHostCmds.push_back(newCmd); 525 #if 0 539 526 /* Limit list size by deleting oldest element. */ 540 527 if (mHostCmds.size() > 256) /** @todo Use a define! */ 541 528 mHostCmds.pop_front(); 529 #endif 542 530 } 543 531 } … … 568 556 case GUEST_GET_HOST_MSG: 569 557 LogFlowFunc(("GUEST_GET_HOST_MSG\n")); 570 rc = retrieveNextHostCmd( callHandle, cParms, paParms);558 rc = retrieveNextHostCmd(u32ClientID, callHandle, cParms, paParms); 571 559 break; 572 560
Note:
See TracChangeset
for help on using the changeset viewer.