Changeset 45415 in vbox for trunk/src/VBox/Additions/common/VBoxService
- Timestamp:
- Apr 8, 2013 9:40:42 PM (12 years ago)
- svn:sync-xref-src-repo-rev:
- 84843
- Location:
- trunk/src/VBox/Additions/common/VBoxService
- Files:
-
- 4 edited
Legend:
- Unmodified
- Added
- Removed
-
trunk/src/VBox/Additions/common/VBoxService/VBoxServiceControl.cpp
r45109 r45415 274 274 * skip all not wanted messages here. 275 275 */ 276 VBoxServiceVerbose(3, "Skipping msg=%RU32 ...\n", uMsg); 276 rc = VbglR3GuestCtrlMsgSkip(g_uControlSvcClientID); 277 VBoxServiceVerbose(3, "Skipping msg=%RU32, rc=%Rrc\n", uMsg, rc); 277 278 } 278 279 break; … … 323 324 pHostCtx->uClientID, pHostCtx->uProtocol); 324 325 325 rc = GstCntlSessionThread Open(&g_lstControlSessionThreads,326 &ssInfo, NULL /* Session */);326 rc = GstCntlSessionThreadCreate(&g_lstControlSessionThreads, 327 &ssInfo, NULL /* Session */); 327 328 } 328 329 … … 350 351 351 352 uint32_t uSessionID, uFlags; 352 353 353 int rc = VbglR3GuestCtrlSessionGetClose(pHostCtx, &uFlags, &uSessionID); 354 354 if (RT_SUCCESS(rc)) … … 361 361 if (pThread->StartupInfo.uSessionID == uSessionID) 362 362 { 363 rc = GstCntlSessionThread Close(pThread, uFlags);363 rc = GstCntlSessionThreadDestroy(pThread, uFlags); 364 364 break; 365 365 } 366 366 } 367 367 #if 0 368 368 if (RT_FAILURE(rc)) 369 369 { … … 379 379 } 380 380 } 381 } 382 383 VBoxServiceVerbose(2, "Closing guest session %RU32 returned rc=%Rrc\n", 384 uSessionID, rc); 381 #endif 382 VBoxServiceVerbose(2, "Closing guest session %RU32 returned rc=%Rrc\n", 383 uSessionID, rc); 384 } 385 else 386 VBoxServiceError("Closing guest session failed with rc=%Rrc\n", rc); 387 385 388 return rc; 386 389 } … … 420 423 VBoxServiceVerbose(2, "Shutting down ...\n"); 421 424 422 int rc2 = GstCntlSessionThread CloseAll(&g_lstControlSessionThreads,423 0 /* Flags */);425 int rc2 = GstCntlSessionThreadDestroyAll(&g_lstControlSessionThreads, 426 0 /* Flags */); 424 427 if (RT_FAILURE(rc2)) 425 428 VBoxServiceError("Closing session threads failed with rc=%Rrc\n", rc2); -
trunk/src/VBox/Additions/common/VBoxService/VBoxServiceControl.h
r45109 r45415 169 169 /** 170 170 * Structure for a guest session thread to 171 * observe the forked session instance. 171 * observe/control the forked session instance from 172 * the VBoxService main executable. 172 173 */ 173 174 typedef struct VBOXSERVICECTRLSESSIONTHREAD … … 243 244 typedef struct VBOXSERVICECTRLSESSION 244 245 { 246 /* The session's startup information. */ 245 247 VBOXSERVICECTRLSESSIONSTARTUPINFO 246 248 StartupInfo; 247 /** List of active guest control threads (VBOXSERVICECTRLTHREAD). */248 RTLISTANCHOR lst ControlThreadsActive;249 /** List of inactive guest control threads (VBOXSERVICECTRLTHREAD). */249 /** List of active guest process threads (VBOXSERVICECTRLPROCESS). */ 250 RTLISTANCHOR lstProcessesActive; 251 /** List of inactive guest process threads (VBOXSERVICECTRLPROCESS). */ 250 252 /** @todo Still needed? */ 251 RTLISTANCHOR lst ControlThreadsInactive;253 RTLISTANCHOR lstProcessesInactive; 252 254 /** List of guest control files (VBOXSERVICECTRLFILE). */ 253 255 RTLISTANCHOR lstFiles; 254 /** Critical section for protecting the guest process 255 * threading list. */ 256 RTCRITSECT csControlThreads; 256 /** The session's critical section. */ 257 RTCRITSECT CritSect; 257 258 /** Session flags. */ 258 259 uint32_t uFlags; … … 330 331 /** Critical section for thread-safe use. */ 331 332 RTCRITSECT CritSect; 332 /** @todo Document me! */ 333 /** Process startup information. */ 334 VBOXSERVICECTRLPROCSTARTUPINFO 335 StartupInfo; 336 /** The process' PID assigned by the guest OS. */ 333 337 uint32_t uPID; 334 char *pszCmd;335 uint32_t uFlags;336 char **papszArgs;337 uint32_t uNumArgs;338 char **papszEnv;339 uint32_t uNumEnvVars;340 /** Name of specified user account to run the341 * guest process under. */342 char *pszUser;343 /** Password of specified user account. */344 char *pszPassword;345 /** Overall time limit (in ms) that the guest process346 * is allowed to run. 0 for indefinite time. */347 uint32_t uTimeLimitMS;348 338 /** Pointer to the current IPC request being 349 * processed. */ 339 * processed. We only support one request at a 340 * time at the moment. 341 ** @todo Implemenet a request queue. */ 350 342 PVBOXSERVICECTRLREQUEST pRequest; 351 343 /** StdIn pipe for addressing writes to the … … 374 366 375 367 /* Guest session thread handling. */ 376 extern int GstCntlSessionThread Open(PRTLISTANCHOR pList, const PVBOXSERVICECTRLSESSIONSTARTUPINFO pSessionStartupInfo, PVBOXSERVICECTRLSESSIONTHREAD *ppSessionThread);377 extern int GstCntlSessionThread Close(PVBOXSERVICECTRLSESSIONTHREAD pSession, uint32_t uFlags);378 extern int GstCntlSessionThread CloseAll(PRTLISTANCHOR pList, uint32_t uFlags);368 extern int GstCntlSessionThreadCreate(PRTLISTANCHOR pList, const PVBOXSERVICECTRLSESSIONSTARTUPINFO pSessionStartupInfo, PVBOXSERVICECTRLSESSIONTHREAD *ppSessionThread); 369 extern int GstCntlSessionThreadDestroy(PVBOXSERVICECTRLSESSIONTHREAD pSession, uint32_t uFlags); 370 extern int GstCntlSessionThreadDestroyAll(PRTLISTANCHOR pList, uint32_t uFlags); 379 371 extern int GstCntlSessionThreadTerminate(PVBOXSERVICECTRLSESSIONTHREAD pSession); 380 372 extern RTEXITCODE VBoxServiceControlSessionForkInit(int argc, char **argv); … … 392 384 extern int GstCntlProcessPerform(PVBOXSERVICECTRLPROCESS pProcess, PVBOXSERVICECTRLREQUEST pRequest); 393 385 extern int GstCntlProcessStart(const PVBOXSERVICECTRLSESSION pSession, const PVBOXSERVICECTRLPROCSTARTUPINFO pStartupInfo, uint32_t uContext); 394 extern int GstCntlProcessStop( const PVBOXSERVICECTRLPROCESS pThread);395 extern void GstCntlProcessRelease(const PVBOXSERVICECTRLPROCESS p Thread);396 extern int GstCntlProcessWait(const PVBOXSERVICECTRLPROCESS p Thread, RTMSINTERVAL msTimeout, int *prc);397 extern int GstCntlProcessFree(PVBOXSERVICECTRLPROCESS p Thread);386 extern int GstCntlProcessStop(PVBOXSERVICECTRLPROCESS pProcess); 387 extern void GstCntlProcessRelease(const PVBOXSERVICECTRLPROCESS pProcess); 388 extern int GstCntlProcessWait(const PVBOXSERVICECTRLPROCESS pProcess, RTMSINTERVAL msTimeout, int *pRc); 389 extern int GstCntlProcessFree(PVBOXSERVICECTRLPROCESS pProcess); 398 390 /* Process request handling. */ 399 391 extern int GstCntlProcessRequestAlloc(PVBOXSERVICECTRLREQUEST *ppReq, VBOXSERVICECTRLREQUESTTYPE enmType); -
trunk/src/VBox/Additions/common/VBoxService/VBoxServiceControlProcess.cpp
r45109 r45415 87 87 return rc; 88 88 89 pProcess->uPID = 0; /* Don't have a PID yet. */ 90 pProcess->pRequest = NULL; /* No request assigned yet. */ 91 pProcess->uFlags = pProcess->uFlags; 92 pProcess->uTimeLimitMS = ( pProcess->uTimeLimitMS == UINT32_MAX 93 || pProcess->uTimeLimitMS == 0) 94 ? RT_INDEFINITE_WAIT : pProcess->uTimeLimitMS; 95 96 /* Prepare argument list. */ 97 pProcess->uNumArgs = 0; /* Initialize in case of RTGetOptArgvFromString() is failing ... */ 98 rc = RTGetOptArgvFromString(&pProcess->papszArgs, (int*)&pProcess->uNumArgs, 99 (pStartupInfo->uNumArgs > 0) ? pStartupInfo->szArgs : "", NULL); 100 /* Did we get the same result? */ 101 Assert(pStartupInfo->uNumArgs == pProcess->uNumArgs); 102 103 if (RT_SUCCESS(rc)) 104 { 105 /* Prepare environment list. */ 106 pProcess->uNumEnvVars = 0; 107 if (pProcess->uNumEnvVars) 108 { 109 pProcess->papszEnv = (char **)RTMemAlloc(pStartupInfo->uNumEnvVars * sizeof(char*)); 110 AssertPtr(pProcess->papszEnv); 111 pProcess->uNumEnvVars = pProcess->uNumEnvVars; 112 113 const char *pszCur = pStartupInfo->szEnv; 114 uint32_t i = 0; 115 uint32_t cbLen = 0; 116 while (cbLen < pStartupInfo->cbEnv) 117 { 118 /* sanity check */ 119 if (i >= pStartupInfo->uNumEnvVars) 120 { 121 rc = VERR_INVALID_PARAMETER; 122 break; 123 } 124 int cbStr = RTStrAPrintf(&pProcess->papszEnv[i++], "%s", pszCur); 125 if (cbStr < 0) 126 { 127 rc = VERR_NO_STR_MEMORY; 128 break; 129 } 130 pszCur += cbStr + 1; /* Skip terminating '\0' */ 131 cbLen += cbStr + 1; /* Skip terminating '\0' */ 132 } 133 Assert(i == pProcess->uNumEnvVars); 134 } 135 136 /* The actual command to execute. */ 137 pProcess->pszCmd = RTStrDup(pStartupInfo->szCmd); 138 AssertPtr(pProcess->pszCmd); 139 140 /* User management. */ 141 pProcess->pszUser = RTStrDup(pStartupInfo->szUser); 142 AssertPtr(pProcess->pszUser); 143 pProcess->pszPassword = RTStrDup(pStartupInfo->szPassword); 144 AssertPtr(pProcess->pszPassword); 145 } 89 pProcess->uPID = 0; /* Don't have a PID yet. */ 90 pProcess->pRequest = NULL; /* No request assigned yet. */ 91 92 /* Copy over startup info. */ 93 memcpy(&pProcess->StartupInfo, pStartupInfo, sizeof(VBOXSERVICECTRLPROCSTARTUPINFO)); 94 95 /* Adjust timeout value. */ 96 if ( pProcess->StartupInfo.uTimeLimitMS == UINT32_MAX 97 || pProcess->StartupInfo.uTimeLimitMS == 0) 98 pProcess->StartupInfo.uTimeLimitMS = RT_INDEFINITE_WAIT; 146 99 147 100 if (RT_FAILURE(rc)) /* Clean up on failure. */ … … 161 114 AssertPtrReturn(pProcess, VERR_INVALID_POINTER); 162 115 163 VBoxServiceVerbose(3, "[PID % u]: Freeing ...\n",116 VBoxServiceVerbose(3, "[PID %RU32]: Freeing ...\n", 164 117 pProcess->uPID); 165 118 166 int rc = RTCritSectEnter(&pProcess->CritSect);167 if (RT_SUCCESS(rc))168 {169 if (pProcess->uNumEnvVars)170 {171 for (uint32_t i = 0; i < pProcess->uNumEnvVars; i++)172 RTStrFree(pProcess->papszEnv[i]);173 RTMemFree(pProcess->papszEnv);174 }175 RTGetOptArgvFree(pProcess->papszArgs);176 177 RTStrFree(pProcess->pszCmd);178 RTStrFree(pProcess->pszUser);179 RTStrFree(pProcess->pszPassword);180 181 VBoxServiceVerbose(3, "[PID %u]: Setting stopped state\n",182 pProcess->uPID);183 184 rc = RTCritSectLeave(&pProcess->CritSect);185 AssertRC(rc);186 }187 119 188 120 /* … … 198 130 pProcess = NULL; 199 131 200 return rc;132 return VINF_SUCCESS; 201 133 } 202 134 … … 207 139 * 208 140 * @return IPRT status code. 209 * @param pThread Thread to shut down. 210 */ 211 int GstCntlProcessStop(const PVBOXSERVICECTRLPROCESS pThread) 212 { 213 AssertPtrReturn(pThread, VERR_INVALID_POINTER); 214 215 VBoxServiceVerbose(3, "[PID %u]: Stopping ...\n", 216 pThread->uPID); 217 218 int rc = gstcntlProcessRequestCancel(pThread->pRequest); 219 if (RT_FAILURE(rc)) 220 VBoxServiceError("[PID %u]: Signalling request event failed, rc=%Rrc\n", 221 pThread->uPID, rc); 222 223 /* Do *not* set pThread->fShutdown or other stuff here! 224 * The guest thread loop will do that as soon as it processes the quit message. */ 225 226 PVBOXSERVICECTRLREQUEST pRequest; 227 rc = GstCntlProcessRequestAlloc(&pRequest, VBOXSERVICECTRLREQUEST_QUIT); 141 * @param pProcess Process to stop. 142 */ 143 int GstCntlProcessStop(PVBOXSERVICECTRLPROCESS pProcess) 144 { 145 AssertPtrReturn(pProcess, VERR_INVALID_POINTER); 146 147 VBoxServiceVerbose(3, "[PID %RU32]: Stopping ...\n", 148 pProcess->uPID); 149 150 int rc = RTCritSectEnter(&pProcess->CritSect); 228 151 if (RT_SUCCESS(rc)) 229 152 { 230 rc = GstCntlProcessPerform(pThread, pRequest); 231 if (RT_FAILURE(rc)) 232 VBoxServiceVerbose(3, "[PID %u]: Sending quit request failed with rc=%Rrc\n", 233 pThread->uPID, rc); 234 235 GstCntlProcessRequestFree(pRequest); 236 } 153 if (pProcess->pRequest) 154 { 155 rc = gstcntlProcessRequestCancel(pProcess->pRequest); 156 if (RT_FAILURE(rc)) 157 VBoxServiceError("[PID %RU32]: Signalling request event failed, rc=%Rrc\n", 158 pProcess->uPID, rc); 159 } 160 161 /* Do *not* set pThread->fShutdown or other stuff here! 162 * The guest thread loop will do that as soon as it processes the quit message. */ 163 164 PVBOXSERVICECTRLREQUEST pRequest; 165 rc = GstCntlProcessRequestAlloc(&pRequest, VBOXSERVICECTRLREQUEST_QUIT); 166 if (RT_SUCCESS(rc)) 167 { 168 rc = GstCntlProcessPerform(pProcess, pRequest); 169 if (RT_FAILURE(rc)) 170 VBoxServiceVerbose(3, "[PID %RU32]: Sending quit request failed with rc=%Rrc\n", 171 pProcess->uPID, rc); 172 173 GstCntlProcessRequestFree(pRequest); 174 } 175 176 int rc2 = RTCritSectLeave(&pProcess->CritSect); 177 AssertRC(rc2); 178 } 179 237 180 return rc; 238 181 } … … 242 185 * Releases (unlocks) a previously locked guest process. 243 186 * 244 * @param p Thread Threadto unlock.245 */ 246 void GstCntlProcessRelease(const PVBOXSERVICECTRLPROCESS p Thread)247 { 248 AssertPtr (pThread);249 250 int rc = RTCritSectLeave(&p Thread->CritSect);187 * @param pProcess Process to unlock. 188 */ 189 void GstCntlProcessRelease(const PVBOXSERVICECTRLPROCESS pProcess) 190 { 191 AssertPtrReturnVoid(pProcess); 192 193 int rc = RTCritSectLeave(&pProcess->CritSect); 251 194 AssertRC(rc); 252 195 } … … 255 198 /** 256 199 * Wait for a guest process thread to shut down. 257 * 258 * @return IPRT status code. 259 * @param pThread Thread to wait shutting down for. 200 * Note: Caller is responsible for locking! 201 * 202 * @return IPRT status code. 203 * @param pProcess Process to wait shutting down for. 260 204 * @param RTMSINTERVAL Timeout in ms to wait for shutdown. 261 * @param p rc Where to store the thread's return code. Optional.262 */ 263 int GstCntlProcessWait(const PVBOXSERVICECTRLPROCESS p Thread,264 RTMSINTERVAL msTimeout, int *p rc)265 { 266 AssertPtrReturn(p Thread, VERR_INVALID_POINTER);267 /* p rc is optional. */205 * @param pRc Where to store the thread's return code. Optional. 206 */ 207 int GstCntlProcessWait(const PVBOXSERVICECTRLPROCESS pProcess, 208 RTMSINTERVAL msTimeout, int *pRc) 209 { 210 AssertPtrReturn(pProcess, VERR_INVALID_POINTER); 211 /* pRc is optional. */ 268 212 269 213 int rc = VINF_SUCCESS; 270 if ( p Thread->Thread != NIL_RTTHREAD271 && ASMAtomicReadBool(&p Thread->fStarted))272 { 273 VBoxServiceVerbose(2, "[PID % u]: Waiting for shutdown of pThread=0x%p = \"%s\"...\n",274 p Thread->uPID, pThread, pThread->pszCmd);214 if ( pProcess->Thread != NIL_RTTHREAD 215 && ASMAtomicReadBool(&pProcess->fStarted)) 216 { 217 VBoxServiceVerbose(2, "[PID %RU32]: Waiting for shutdown (%RU32ms) ...\n", 218 pProcess->uPID, msTimeout); 275 219 276 220 /* Wait a bit ... */ 277 221 int rcThread; 278 rc = RTThreadWait(p Thread->Thread, msTimeout, &rcThread);222 rc = RTThreadWait(pProcess->Thread, msTimeout, &rcThread); 279 223 if (RT_FAILURE(rc)) 280 224 { 281 VBoxServiceError("[PID % u]: Waiting for shutting down thread returned error rc=%Rrc\n",282 p Thread->uPID, rc);225 VBoxServiceError("[PID %RU32]: Waiting for shutting down thread returned error rc=%Rrc\n", 226 pProcess->uPID, rc); 283 227 } 284 228 else 285 229 { 286 VBoxServiceVerbose(3, "[PID %u]: Thread reported exit code=%Rrc\n", 287 pThread->uPID, rcThread); 288 if (prc) 289 *prc = rcThread; 290 } 291 } 230 VBoxServiceVerbose(3, "[PID %RU32]: Thread reported exit code=%Rrc\n", 231 pProcess->uPID, rcThread); 232 if (pRc) 233 *pRc = rcThread; 234 } 235 } 236 292 237 return rc; 293 238 } … … 368 313 && cbReadable) 369 314 { 370 VBoxServiceVerbose(3, "gstcntlProcessHandleOutputError: idPollHnd=% u has %ldbytes left, vetoing close\n",315 VBoxServiceVerbose(3, "gstcntlProcessHandleOutputError: idPollHnd=%RU32 has %zu bytes left, vetoing close\n", 371 316 idPollHnd, cbReadable); 372 317 … … 377 322 } 378 323 else 379 VBoxServiceVerbose(3, "gstcntlProcessHandleOutputError: idPollHnd=% uwill be closed\n",324 VBoxServiceVerbose(3, "gstcntlProcessHandleOutputError: idPollHnd=%RU32 will be closed\n", 380 325 idPollHnd); 381 326 … … 460 405 pRequest->rc = rc; 461 406 462 #ifdef _DEBUG463 VBoxServiceVerbose(4, "Handled req=% u, CID=%u, rc=%Rrc, cbData=%u, pvData=%p\n",407 #ifdef DEBUG_andy 408 VBoxServiceVerbose(4, "Handled req=%RU32, CID=%RU32, rc=%Rrc, cbData=%RU32, pvData=%p\n", 464 409 pRequest->enmType, pRequest->uCID, pRequest->rc, 465 410 pRequest->cbData, pRequest->pvData); … … 477 422 static int gstcntlProcessHandleRequest(RTPOLLSET hPollSet, uint32_t fPollEvt, 478 423 PRTPIPE phStdInW, PRTPIPE phStdOutR, PRTPIPE phStdErrR, 479 PVBOXSERVICECTRLPROCESS p Thread, PVBOXSERVICECTRLREQUEST pRequest)424 PVBOXSERVICECTRLPROCESS pProcess, PVBOXSERVICECTRLREQUEST pRequest) 480 425 { 481 426 AssertPtrReturn(phStdInW, VERR_INVALID_POINTER); 482 427 AssertPtrReturn(phStdOutR, VERR_INVALID_POINTER); 483 428 AssertPtrReturn(phStdErrR, VERR_INVALID_POINTER); 484 AssertPtrReturn(p Thread, VERR_INVALID_POINTER);429 AssertPtrReturn(pProcess, VERR_INVALID_POINTER); 485 430 AssertPtrReturn(pRequest, VERR_INVALID_POINTER); 431 432 VBoxServiceVerbose(4, "[PID %RU32]: Handling pRequest=%p\n", 433 pProcess->uPID, pRequest); 486 434 487 435 /* Drain the notification pipe. */ 488 436 uint8_t abBuf[8]; 489 437 size_t cbIgnore; 490 int rc = RTPipeRead(p Thread->hNotificationPipeR, abBuf, sizeof(abBuf), &cbIgnore);438 int rc = RTPipeRead(pProcess->hNotificationPipeR, abBuf, sizeof(abBuf), &cbIgnore); 491 439 if (RT_FAILURE(rc)) 492 440 VBoxServiceError("Draining IPC notification pipe failed with rc=%Rrc\n", rc); … … 501 449 /** @todo Check for some conditions to check to 502 450 * veto quitting. */ 503 ASMAtomicXchgBool(&p Thread->fShutdown, true);451 ASMAtomicXchgBool(&pProcess->fShutdown, true); 504 452 rcReq = VERR_CANCELLED; 505 453 break; … … 572 520 573 521 case VBOXSERVICECTRLREQUEST_PROC_TERM: 574 ASMAtomicXchgBool(&p Thread->fShutdown, true);522 ASMAtomicXchgBool(&pProcess->fShutdown, true); 575 523 fDefer = true; 576 524 break; … … 589 537 /* No access to pRequest here anymore -- could be out of scope 590 538 * or modified already! */ 591 p Thread->pRequest = pRequest = NULL;539 pProcess->pRequest = pRequest = NULL; 592 540 } 593 541 else /* Completing the request defered. */ … … 603 551 * 604 552 * @return IPRT status code. 605 * @param p Thread The process' threadhandle.553 * @param pProcess The guest process to handle. 606 554 * @param hProcess The actual process handle. 607 555 * @param cMsTimeout Time limit (in ms) of the process' life time. … … 611 559 * @param hStdErrR Handle to the process' stderr read end. 612 560 */ 613 static int gstcntlProcessProcLoop(PVBOXSERVICECTRLPROCESS p Thread,614 RTPROCESS hProcess, RT MSINTERVAL cMsTimeout, RTPOLLSET hPollSet,561 static int gstcntlProcessProcLoop(PVBOXSERVICECTRLPROCESS pProcess, 562 RTPROCESS hProcess, RTPOLLSET hPollSet, 615 563 PRTPIPE phStdInW, PRTPIPE phStdOutR, PRTPIPE phStdErrR) 616 564 { 617 AssertPtrReturn(p Thread, VERR_INVALID_POINTER);565 AssertPtrReturn(pProcess, VERR_INVALID_POINTER); 618 566 AssertPtrReturn(phStdInW, VERR_INVALID_PARAMETER); 619 567 /* Rest is optional. */ … … 621 569 int rc; 622 570 int rc2; 623 uint64_t const MsStart= RTTimeMilliTS();571 uint64_t const uMsStart = RTTimeMilliTS(); 624 572 RTPROCSTATUS ProcessStatus = { 254, RTPROCEXITREASON_ABEND }; 625 573 bool fProcessAlive = true; … … 636 584 * the first (stale) entry will be found and we get really weird results! 637 585 */ 638 rc = gstcntlProcessAssignPID(p Thread, hProcess);586 rc = gstcntlProcessAssignPID(pProcess, hProcess); 639 587 if (RT_FAILURE(rc)) 640 588 { … … 648 596 * and that it's now OK to send input to the process. 649 597 */ 650 VBoxServiceVerbose(2, "[PID %u]: Process \"%s\" started, CID=%u, User=%s\n", 651 pThread->uPID, pThread->pszCmd, pThread->uContextID, pThread->pszUser); 652 VBGLR3GUESTCTRLCMDCTX ctx = { pThread->uClientID, pThread->uContextID }; 598 VBoxServiceVerbose(2, "[PID %RU32]: Process \"%s\" started, CID=%u, User=%s, cMsTimeout=%RU32\n", 599 pProcess->uPID, pProcess->StartupInfo.szCmd, pProcess->uContextID, 600 pProcess->StartupInfo.szUser, pProcess->StartupInfo.uTimeLimitMS); 601 VBGLR3GUESTCTRLCMDCTX ctx = { pProcess->uClientID, pProcess->uContextID }; 653 602 rc = VbglR3GuestCtrlProcCbStatus(&ctx, 654 p Thread->uPID, PROC_STS_STARTED, 0 /* u32Flags */,603 pProcess->uPID, PROC_STS_STARTED, 0 /* u32Flags */, 655 604 NULL /* pvData */, 0 /* cbData */); 656 605 … … 660 609 PVBOXSERVICECTRLREQUEST pReq = NULL; 661 610 while ( RT_SUCCESS(rc) 662 && RT_UNLIKELY(!p Thread->fShutdown))611 && RT_UNLIKELY(!pProcess->fShutdown)) 663 612 { 664 613 /* … … 668 617 uint32_t fPollEvt; 669 618 rc2 = RTPollNoResume(hPollSet, cMsPollCur, &fPollEvt, &idPollHnd); 670 if (p Thread->fShutdown)619 if (pProcess->fShutdown) 671 620 continue; 672 621 … … 675 624 if (RT_SUCCESS(rc2)) 676 625 { 677 /*VBoxServiceVerbose(4, "[PID %u}: RTPollNoResume idPollHnd=%u\n",678 pThread->uPID, idPollHnd);*/679 626 switch (idPollHnd) 680 627 { … … 693 640 694 641 case VBOXSERVICECTRLPIPEID_IPC_NOTIFY: 695 pReq = p Thread->pRequest; /** @todo Implement request queue. */642 pReq = pProcess->pRequest; /** @todo Implement request queue. */ 696 643 rc = gstcntlProcessHandleRequest(hPollSet, fPollEvt, 697 644 phStdInW, phStdOutR, phStdErrR, 698 p Thread, pReq);645 pProcess, pReq); 699 646 if (rc != VINF_AIO_TASK_PENDING) 700 647 pReq = NULL; … … 708 655 if (RT_FAILURE(rc) || rc == VINF_EOF) 709 656 break; /* Abort command, or client dead or something. */ 710 711 if (RT_UNLIKELY(pThread->fShutdown)) 712 break; /* We were asked to shutdown. */ 713 714 continue; 715 } 716 717 #if 0 718 VBoxServiceVerbose(4, "[PID %u]: Polling done, pollRC=%Rrc, pollCnt=%u, rc=%Rrc, fShutdown=%RTbool\n", 719 pThread->uPID, rc2, RTPollSetGetCount(hPollSet), rc, pThread->fShutdown); 657 } 658 #ifdef DEBUG_andy 659 VBoxServiceVerbose(4, "[PID %RU32]: Polling done, pollRc=%Rrc, pollCnt=%u, idPollHnd=%RU32, rc=%Rrc, fProcessAlive=%RTbool, fShutdown=%RTbool\n", 660 pProcess->uPID, rc2, RTPollSetGetCount(hPollSet), idPollHnd, rc, fProcessAlive, pProcess->fShutdown); 720 661 #endif 662 663 if (RT_UNLIKELY(pProcess->fShutdown)) 664 break; /* We were asked to shutdown. */ 665 721 666 /* 722 667 * Check for process death. … … 725 670 { 726 671 rc2 = RTProcWaitNoResume(hProcess, RTPROCWAIT_FLAGS_NOBLOCK, &ProcessStatus); 672 #ifdef DEBUG_andy 673 VBoxServiceVerbose(4, "[PID %RU32]: RTProcWaitNoResume=%Rrc, stdOut=%s, stdErrR=%s\n", 674 pProcess->uPID, rc2, 675 *phStdOutR == NIL_RTPIPE ? "closed" : "open", 676 *phStdErrR == NIL_RTPIPE ? "closed" : "open"); 677 #endif 727 678 if (RT_SUCCESS_NP(rc2)) 728 679 { … … 756 707 */ 757 708 uint32_t cMilliesLeft = RT_INDEFINITE_WAIT; 758 if ( cMsTimeout!= RT_INDEFINITE_WAIT)709 if (pProcess->StartupInfo.uTimeLimitMS != RT_INDEFINITE_WAIT) 759 710 { 760 711 uint64_t u64Now = RTTimeMilliTS(); 761 uint64_t cMsElapsed = u64Now - MsStart;762 if (cMsElapsed >= cMsTimeout)712 uint64_t cMsElapsed = u64Now - uMsStart; 713 if (cMsElapsed >= pProcess->StartupInfo.uTimeLimitMS) 763 714 { 764 VBoxServiceVerbose(3, "[PID % u]: Timed out (%ums elapsed > %ums timeout), killing ...\n",765 p Thread->uPID, cMsElapsed, cMsTimeout);715 VBoxServiceVerbose(3, "[PID %RU32]: Timed out (%RU32ms elapsed > %RU32ms timeout), killing ...\n", 716 pProcess->uPID, cMsElapsed, pProcess->StartupInfo.uTimeLimitMS); 766 717 767 718 fProcessTimedOut = true; … … 778 729 } 779 730 else 780 cMilliesLeft = cMsTimeout- (uint32_t)cMsElapsed;731 cMilliesLeft = pProcess->StartupInfo.uTimeLimitMS - (uint32_t)cMsElapsed; 781 732 } 782 733 … … 787 738 if (cMilliesLeft < cMsPollCur) 788 739 cMsPollCur = cMilliesLeft; 789 790 /* 791 * Need to exit? 792 */ 793 if (pThread->fShutdown) 794 break; 795 } 796 797 rc2 = RTCritSectEnter(&pThread->CritSect); 740 } 741 742 rc2 = RTCritSectEnter(&pProcess->CritSect); 798 743 if (RT_SUCCESS(rc2)) 799 744 { 800 ASMAtomicXchgBool(&p Thread->fShutdown, true);801 802 rc2 = RTCritSectLeave(&p Thread->CritSect);745 ASMAtomicXchgBool(&pProcess->fShutdown, true); 746 747 rc2 = RTCritSectLeave(&pProcess->CritSect); 803 748 AssertRC(rc2); 804 749 } … … 811 756 if (MsProcessKilled == UINT64_MAX) 812 757 { 813 VBoxServiceVerbose(3, "[PID % u]: Is still alive and not killed yet\n",814 p Thread->uPID);758 VBoxServiceVerbose(3, "[PID %RU32]: Is still alive and not killed yet\n", 759 pProcess->uPID); 815 760 816 761 MsProcessKilled = RTTimeMilliTS(); … … 821 766 for (size_t i = 0; i < 10; i++) 822 767 { 823 VBoxServiceVerbose(4, "[PID % u]: Kill attempt %d/10: Waiting to exit ...\n",824 p Thread->uPID, i + 1);768 VBoxServiceVerbose(4, "[PID %RU32]: Kill attempt %d/10: Waiting to exit ...\n", 769 pProcess->uPID, i + 1); 825 770 rc2 = RTProcWait(hProcess, RTPROCWAIT_FLAGS_NOBLOCK, &ProcessStatus); 826 771 if (RT_SUCCESS(rc2)) 827 772 { 828 VBoxServiceVerbose(4, "[PID % u]: Kill attempt %d/10: Exited\n",829 p Thread->uPID, i + 1);773 VBoxServiceVerbose(4, "[PID %RU32]: Kill attempt %d/10: Exited\n", 774 pProcess->uPID, i + 1); 830 775 fProcessAlive = false; 831 776 break; … … 833 778 if (i >= 5) 834 779 { 835 VBoxServiceVerbose(4, "[PID % u]: Kill attempt %d/10: Trying to terminate ...\n",836 p Thread->uPID, i + 1);780 VBoxServiceVerbose(4, "[PID %RU32]: Kill attempt %d/10: Trying to terminate ...\n", 781 pProcess->uPID, i + 1); 837 782 RTProcTerminate(hProcess); 838 783 } … … 841 786 842 787 if (fProcessAlive) 843 VBoxServiceVerbose(3, "[PID % u]: Could not be killed\n", pThread->uPID);788 VBoxServiceVerbose(3, "[PID %RU32]: Could not be killed\n", pProcess->uPID); 844 789 845 790 if ( pReq /* Handle deferred termination request. */ … … 865 810 if ( fProcessTimedOut && !fProcessAlive && MsProcessKilled != UINT64_MAX) 866 811 { 867 VBoxServiceVerbose(3, "[PID % u]: Timed out and got killed\n",868 p Thread->uPID);812 VBoxServiceVerbose(3, "[PID %RU32]: Timed out and got killed\n", 813 pProcess->uPID); 869 814 uStatus = PROC_STS_TOK; 870 815 } 871 816 else if (fProcessTimedOut && fProcessAlive && MsProcessKilled != UINT64_MAX) 872 817 { 873 VBoxServiceVerbose(3, "[PID % u]: Timed out and did *not* get killed\n",874 p Thread->uPID);818 VBoxServiceVerbose(3, "[PID %RU32]: Timed out and did *not* get killed\n", 819 pProcess->uPID); 875 820 uStatus = PROC_STS_TOA; 876 821 } 877 else if (p Thread->fShutdown && (fProcessAlive || MsProcessKilled != UINT64_MAX))878 { 879 VBoxServiceVerbose(3, "[PID % u]: Got terminated because system/service is about to shutdown\n",880 p Thread->uPID);822 else if (pProcess->fShutdown && (fProcessAlive || MsProcessKilled != UINT64_MAX)) 823 { 824 VBoxServiceVerbose(3, "[PID %RU32]: Got terminated because system/service is about to shutdown\n", 825 pProcess->uPID); 881 826 uStatus = PROC_STS_DWN; /* Service is stopping, process was killed. */ 882 uFlags = p Thread->uFlags; /* Return handed-in execution flags back to the host. */827 uFlags = pProcess->StartupInfo.uFlags; /* Return handed-in execution flags back to the host. */ 883 828 } 884 829 else if (fProcessAlive) 885 830 { 886 VBoxServiceError("[PID % u]: Is alive when it should not!\n",887 p Thread->uPID);831 VBoxServiceError("[PID %RU32]: Is alive when it should not!\n", 832 pProcess->uPID); 888 833 } 889 834 else if (MsProcessKilled != UINT64_MAX) 890 835 { 891 VBoxServiceError("[PID % u]: Has been killed when it should not!\n",892 p Thread->uPID);836 VBoxServiceError("[PID %RU32]: Has been killed when it should not!\n", 837 pProcess->uPID); 893 838 } 894 839 else if (ProcessStatus.enmReason == RTPROCEXITREASON_NORMAL) 895 840 { 896 VBoxServiceVerbose(3, "[PID % u]: Ended with RTPROCEXITREASON_NORMAL (Exit code: %u)\n",897 p Thread->uPID, ProcessStatus.iStatus);841 VBoxServiceVerbose(3, "[PID %RU32]: Ended with RTPROCEXITREASON_NORMAL (Exit code: %u)\n", 842 pProcess->uPID, ProcessStatus.iStatus); 898 843 899 844 uStatus = PROC_STS_TEN; … … 902 847 else if (ProcessStatus.enmReason == RTPROCEXITREASON_SIGNAL) 903 848 { 904 VBoxServiceVerbose(3, "[PID % u]: Ended with RTPROCEXITREASON_SIGNAL (Signal: %u)\n",905 p Thread->uPID, ProcessStatus.iStatus);849 VBoxServiceVerbose(3, "[PID %RU32]: Ended with RTPROCEXITREASON_SIGNAL (Signal: %u)\n", 850 pProcess->uPID, ProcessStatus.iStatus); 906 851 907 852 uStatus = PROC_STS_TES; … … 911 856 { 912 857 /* ProcessStatus.iStatus will be undefined. */ 913 VBoxServiceVerbose(3, "[PID % u]: Ended with RTPROCEXITREASON_ABEND\n",914 p Thread->uPID);858 VBoxServiceVerbose(3, "[PID %RU32]: Ended with RTPROCEXITREASON_ABEND\n", 859 pProcess->uPID); 915 860 916 861 uStatus = PROC_STS_TEA; … … 918 863 } 919 864 else 920 VBoxServiceVerbose(1, "[PID % u]: Handling process status %u not implemented\n",921 p Thread->uPID, ProcessStatus.enmReason);922 923 VBoxServiceVerbose(2, "[PID % u]: Ended, ClientID=%u, CID=%u, Status=%u, Flags=0x%x\n",924 p Thread->uPID, pThread->uClientID, pThread->uContextID, uStatus, uFlags);925 926 if (!(p Thread->uFlags & EXECUTEPROCESSFLAG_WAIT_START))927 { 928 VBGLR3GUESTCTRLCMDCTX ctx = { p Thread->uClientID, pThread->uContextID };865 VBoxServiceVerbose(1, "[PID %RU32]: Handling process status %u not implemented\n", 866 pProcess->uPID, ProcessStatus.enmReason); 867 868 VBoxServiceVerbose(2, "[PID %RU32]: Ended, ClientID=%u, CID=%u, Status=%u, Flags=0x%x\n", 869 pProcess->uPID, pProcess->uClientID, pProcess->uContextID, uStatus, uFlags); 870 871 if (!(pProcess->StartupInfo.uFlags & EXECUTEPROCESSFLAG_WAIT_START)) 872 { 873 VBGLR3GUESTCTRLCMDCTX ctx = { pProcess->uClientID, pProcess->uContextID }; 929 874 rc2 = VbglR3GuestCtrlProcCbStatus(&ctx, 930 p Thread->uPID, uStatus, uFlags,875 pProcess->uPID, uStatus, uFlags, 931 876 NULL /* pvData */, 0 /* cbData */); 932 877 if (RT_FAILURE(rc2)) 933 VBoxServiceError("[PID % u]: Error reporting final status to host; rc=%Rrc\n",934 p Thread->uPID, rc2);878 VBoxServiceError("[PID %RU32]: Error reporting final status to host; rc=%Rrc\n", 879 pProcess->uPID, rc2); 935 880 if (RT_SUCCESS(rc)) 936 881 rc = rc2; 937 882 } 938 883 else 939 VBoxServiceVerbose(3, "[PID % u]: Was started detached, no final status sent to host\n",940 p Thread->uPID);941 942 VBoxServiceVerbose(3, "[PID % u]: Process loop ended with rc=%Rrc\n",943 p Thread->uPID, rc);884 VBoxServiceVerbose(3, "[PID %RU32]: Was started detached, no final status sent to host\n", 885 pProcess->uPID); 886 887 VBoxServiceVerbose(3, "[PID %RU32]: Process loop ended with rc=%Rrc\n", 888 pProcess->uPID, rc); 944 889 } 945 890 else 946 VBoxServiceError("[PID % u]: Loop failed with rc=%Rrc\n",947 p Thread->uPID, rc);891 VBoxServiceError("[PID %RU32]: Loop failed with rc=%Rrc\n", 892 pProcess->uPID, rc); 948 893 return rc; 949 894 } … … 1034 979 1035 980 /** 1036 * Cancels a previously fired off guest thread request. 1037 * 1038 * Note: Does *not* do locking since GstCntlProcessRequestWait() 1039 * holds the lock (critsect); so only trigger the signal; the owner 1040 * needs to clean up afterwards. 981 * Cancels a previously fired off guest process request. 982 * Note: Caller is responsible for locking! 1041 983 * 1042 984 * @return IPRT status code. … … 1357 1299 1358 1300 AssertPtr(pThread->pSession); 1359 int rc = RTCritSectEnter(&pThread->pSession-> csControlThreads);1301 int rc = RTCritSectEnter(&pThread->pSession->CritSect); 1360 1302 if (RT_SUCCESS(rc)) 1361 1303 { … … 1366 1308 do 1367 1309 { 1368 RTListForEach(&pThread->pSession->lst ControlThreadsActive, pThreadCur, VBOXSERVICECTRLPROCESS, Node)1310 RTListForEach(&pThread->pSession->lstProcessesActive, pThreadCur, VBOXSERVICECTRLPROCESS, Node) 1369 1311 { 1370 1312 if (pThreadCur->uPID == uPID) … … 1384 1326 pThread->uPID = uPID; 1385 1327 1386 rc = RTCritSectLeave(&pThread->pSession-> csControlThreads);1328 rc = RTCritSectLeave(&pThread->pSession->CritSect); 1387 1329 AssertRC(rc); 1388 1330 } … … 1562 1504 AssertPtrReturn(pProcess, VERR_INVALID_POINTER); 1563 1505 VBoxServiceVerbose(3, "Thread of process pThread=0x%p = \"%s\" started\n", 1564 pProcess, pProcess-> pszCmd);1506 pProcess, pProcess->StartupInfo.szCmd); 1565 1507 1566 1508 int rc = GstCntlSessionListSet(pProcess->pSession, … … 1576 1518 } 1577 1519 VBoxServiceVerbose(3, "Guest process \"%s\" got client ID=%u, flags=0x%x\n", 1578 pProcess-> pszCmd, pProcess->uClientID, pProcess->uFlags);1520 pProcess->StartupInfo.szCmd, pProcess->uClientID, pProcess->StartupInfo.uFlags); 1579 1521 1580 1522 bool fSignalled = false; /* Indicator whether we signalled the thread user event already. */ 1523 1524 /* 1525 * Prepare argument list. 1526 */ 1527 char **papszArgs; 1528 uint32_t uNumArgs = 0; /* Initialize in case of RTGetOptArgvFromString() is failing ... */ 1529 rc = RTGetOptArgvFromString(&papszArgs, (int*)&uNumArgs, 1530 (pProcess->StartupInfo.uNumArgs > 0) ? pProcess->StartupInfo.szArgs : "", NULL); 1531 /* Did we get the same result? */ 1532 Assert(pProcess->StartupInfo.uNumArgs == uNumArgs); 1533 1534 /* 1535 * Prepare environment variables list. 1536 */ 1537 char **papszEnv; 1538 uint32_t uNumEnvVars = 0; /* Initialize in case of failing ... */ 1539 if (RT_SUCCESS(rc)) 1540 { 1541 /* Prepare environment list. */ 1542 if (pProcess->StartupInfo.uNumEnvVars) 1543 { 1544 papszEnv = (char **)RTMemAlloc(pProcess->StartupInfo.uNumEnvVars * sizeof(char*)); 1545 AssertPtr(papszEnv); 1546 uNumEnvVars = pProcess->StartupInfo.uNumEnvVars; 1547 1548 const char *pszCur = pProcess->StartupInfo.szEnv; 1549 uint32_t i = 0; 1550 uint32_t cbLen = 0; 1551 while (cbLen < pProcess->StartupInfo.cbEnv) 1552 { 1553 /* sanity check */ 1554 if (i >= pProcess->StartupInfo.uNumEnvVars) 1555 { 1556 rc = VERR_INVALID_PARAMETER; 1557 break; 1558 } 1559 int cbStr = RTStrAPrintf(&papszEnv[i++], "%s", pszCur); 1560 if (cbStr < 0) 1561 { 1562 rc = VERR_NO_STR_MEMORY; 1563 break; 1564 } 1565 pszCur += cbStr + 1; /* Skip terminating '\0' */ 1566 cbLen += cbStr + 1; /* Skip terminating '\0' */ 1567 } 1568 Assert(i == pProcess->StartupInfo.uNumEnvVars); 1569 } 1570 } 1581 1571 1582 1572 /* … … 1588 1578 { 1589 1579 size_t i; 1590 for (i = 0; i < pProcess->uNumEnvVars && pProcess->papszEnv; i++)1591 { 1592 rc = RTEnvPutEx(hEnv, p Process->papszEnv[i]);1580 for (i = 0; i < uNumEnvVars && papszEnv; i++) 1581 { 1582 rc = RTEnvPutEx(hEnv, papszEnv[i]); 1593 1583 if (RT_FAILURE(rc)) 1594 1584 break; … … 1609 1599 PRTHANDLE phStdOut; 1610 1600 RTPIPE pipeStdOutR; 1611 rc = gstcntlProcessSetupPipe( (pProcess-> uFlags & EXECUTEPROCESSFLAG_WAIT_STDOUT)1601 rc = gstcntlProcessSetupPipe( (pProcess->StartupInfo.uFlags & EXECUTEPROCESSFLAG_WAIT_STDOUT) 1612 1602 ? "|" : "/dev/null", 1613 1603 1 /*STDOUT_FILENO*/, … … 1618 1608 PRTHANDLE phStdErr; 1619 1609 RTPIPE pipeStdErrR; 1620 rc = gstcntlProcessSetupPipe( (pProcess-> uFlags & EXECUTEPROCESSFLAG_WAIT_STDERR)1610 rc = gstcntlProcessSetupPipe( (pProcess->StartupInfo.uFlags & EXECUTEPROCESSFLAG_WAIT_STDERR) 1621 1611 ? "|" : "/dev/null", 1622 1612 2 /*STDERR_FILENO*/, … … 1658 1648 1659 1649 RTPROCESS hProcess; 1660 rc = gstcntlProcessCreateProcess(pProcess-> pszCmd, pProcess->papszArgs, hEnv, pProcess->uFlags,1650 rc = gstcntlProcessCreateProcess(pProcess->StartupInfo.szCmd, papszArgs, hEnv, pProcess->StartupInfo.uFlags, 1661 1651 phStdIn, phStdOut, phStdErr, 1662 fNeedsImpersonation ? pProcess-> pszUser : NULL,1663 fNeedsImpersonation ? pProcess-> pszPassword : NULL,1652 fNeedsImpersonation ? pProcess->StartupInfo.szUser : NULL, 1653 fNeedsImpersonation ? pProcess->StartupInfo.szPassword : NULL, 1664 1654 &hProcess); 1665 1655 if (RT_FAILURE(rc)) … … 1689 1679 1690 1680 /* Enter the process loop. */ 1691 rc = gstcntlProcessProcLoop(pProcess, 1692 hProcess, pProcess->uTimeLimitMS, hPollSet, 1681 rc = gstcntlProcessProcLoop(pProcess, hProcess, hPollSet, 1693 1682 &pProcess->pipeStdInW, &pipeStdOutR, &pipeStdErrR); 1694 1683 … … 1759 1748 /* Disconnect this client from the guest control service. This also cancels all 1760 1749 * outstanding host requests. */ 1761 VBoxServiceVerbose(3, "[PID % u]: Disconnecting (client ID=%u) ...\n",1750 VBoxServiceVerbose(3, "[PID %RU32]: Disconnecting (client ID=%u) ...\n", 1762 1751 pProcess->uPID, pProcess->uClientID); 1763 1752 VbglR3GuestCtrlDisconnect(pProcess->uClientID); … … 1765 1754 } 1766 1755 1767 VBoxServiceVerbose(3, "[PID %u]: Thread of process \"%s\" ended with rc=%Rrc\n", 1768 pProcess->uPID, pProcess->pszCmd, rc); 1756 VBoxServiceVerbose(3, "[PID %RU32]: Thread of process \"%s\" ended with rc=%Rrc\n", 1757 pProcess->uPID, pProcess->StartupInfo.szCmd, rc); 1758 1759 /* Free argument + environment variable lists. */ 1760 if (uNumEnvVars) 1761 { 1762 for (uint32_t i = 0; i < uNumEnvVars; i++) 1763 RTStrFree(papszEnv[i]); 1764 RTMemFree(papszEnv); 1765 } 1766 if (uNumArgs) 1767 RTGetOptArgvFree(papszArgs); 1769 1768 1770 1769 /* Update started/stopped status. */ … … 1779 1778 RTThreadUserSignal(RTThreadSelf()); 1780 1779 1780 VBoxServiceVerbose(3, "[PID %RU32]: Thread returned with rc=%Rrc\n", 1781 pProcess->uPID, rc); 1781 1782 return rc; 1782 1783 } … … 1867 1868 * Performs a request to a specific (formerly started) guest process and waits 1868 1869 * for its response. 1869 * Note: Caller is responsible oflocking!1870 * Note: Caller is responsible for locking! 1870 1871 * 1871 1872 * @return IPRT status code. … … 1889 1890 else 1890 1891 { 1892 VBoxServiceVerbose(3, "[PID %RU32]: Sending pRequest=%p\n", 1893 pProcess->uPID, pRequest); 1894 1891 1895 /* Set request structure pointer. */ 1892 1896 pProcess->pRequest = pRequest; … … 1894 1898 /** @todo To speed up simultaneous guest process handling we could add a worker threads 1895 1899 * or queue in order to wait for the request to happen. Later. */ 1896 /* Wake up guest thr ad by sending a wakeup byte to the notification pipe so1900 /* Wake up guest thread by sending a wakeup byte to the notification pipe so 1897 1901 * that RTPoll unblocks (returns) and we then can do our requested operation. */ 1898 1902 Assert(pProcess->hNotificationPipeW != NIL_RTPIPE); 1899 size_t cbWritten ;1903 size_t cbWritten = 0; 1900 1904 if (RT_SUCCESS(rc)) 1901 1905 rc = RTPipeWrite(pProcess->hNotificationPipeW, "i", 1, &cbWritten); 1902 1906 1903 if ( RT_SUCCESS(rc)1904 && cbWritten)1905 {1906 VBoxServiceVerbose(3, "[PID % u]: Waiting for response onenmType=%u, pvData=0x%p, cbData=%u\n",1907 pProcess->uPID, pRequest ->enmType, pRequest->pvData, pRequest->cbData);1907 if (RT_SUCCESS(rc)) 1908 { 1909 Assert(cbWritten); 1910 VBoxServiceVerbose(3, "[PID %RU32]: Waiting for response on pRequest=%p, enmType=%u, pvData=0x%p, cbData=%u\n", 1911 pProcess->uPID, pRequest, pRequest->enmType, pRequest->pvData, pRequest->cbData); 1908 1912 1909 1913 rc = GstCntlProcessRequestWait(pRequest); … … 1911 1915 } 1912 1916 1913 VBoxServiceVerbose(3, "[PID % u]: PerformedenmType=%u, uCID=%u, pvData=0x%p, cbData=%u, rc=%Rrc\n",1914 pProcess->uPID, pRequest ->enmType, pRequest->uCID, pRequest->pvData, pRequest->cbData, rc);1917 VBoxServiceVerbose(3, "[PID %RU32]: Performed pRequest=%p, enmType=%u, uCID=%u, pvData=0x%p, cbData=%u, rc=%Rrc\n", 1918 pProcess->uPID, pRequest, pRequest->enmType, pRequest->uCID, pRequest->pvData, pRequest->cbData, rc); 1915 1919 return rc; 1916 1920 } -
trunk/src/VBox/Additions/common/VBoxService/VBoxServiceControlSession.cpp
r45109 r45415 70 70 VBOXSERVICESESSIONOPT_USERNAME, 71 71 VBOXSERVICESESSIONOPT_SESSION_ID, 72 VBOXSERVICESESSIONOPT_SESSION_PROTO 72 VBOXSERVICESESSIONOPT_SESSION_PROTO, 73 VBOXSERVICESESSIONOPT_THREAD_ID 73 74 }; 74 75 … … 568 569 if (RT_SUCCESS(rc)) 569 570 { 570 VBoxServiceVerbose(3, "Request to start process szCmd=%s, uFlags=0x%x, szArgs=%s, szEnv=%s, szUser=%s, szPassword=%s,uTimeout=%RU32\n",571 VBoxServiceVerbose(3, "Request to start process szCmd=%s, uFlags=0x%x, szArgs=%s, szEnv=%s, uTimeout=%RU32\n", 571 572 startupInfo.szCmd, startupInfo.uFlags, 572 573 startupInfo.uNumArgs ? startupInfo.szArgs : "<None>", 573 574 startupInfo.uNumEnvVars ? startupInfo.szEnv : "<None>", 574 startupInfo.szUser,575 #ifdef DEBUG576 startupInfo.szPassword,577 #else578 "XXX", /* Never show passwords in release mode. */579 #endif580 575 startupInfo.uTimeLimitMS); 581 576 582 rc = GstCntlSessionReapProcesses(pSession);577 /*rc = GstCntlSessionReapProcesses(pSession); 583 578 if (RT_FAILURE(rc)) 584 VBoxServiceError("Reaping stopped guest processes failed with rc=%Rrc\n", rc); 579 VBoxServiceError("Reaping stopped guest processes failed with rc=%Rrc\n", rc);*/ 585 580 /* Keep going. */ 586 581 … … 741 736 742 737 int rc = VbglR3GuestCtrlProcGetOutput(pHostCtx, &uPID, &uHandleID, &uFlags); 738 #ifdef DEBUG_andy 739 VBoxServiceVerbose(4, "[PID %RU32]: Get output CID=%RU32, uHandleID=%RU32, uFlags=%RU32\n", 740 uPID, pHostCtx->uContextID, uHandleID, uFlags); 741 #endif 743 742 if (RT_SUCCESS(rc)) 744 743 { … … 750 749 pHostCtx->uContextID, uHandleID, RT_INDEFINITE_WAIT /* Timeout */, 751 750 pBuf, _64K /* cbSize */, &cbRead); 752 VBoxServiceVerbose(3, "[PID %RU32]: Got output, rc=%Rrc, CID=% u, cbRead=%u, uHandle=%u, uFlags=%u\n",751 VBoxServiceVerbose(3, "[PID %RU32]: Got output, rc=%Rrc, CID=%RU32, cbRead=%RU32, uHandle=%RU32, uFlags=%x\n", 753 752 uPID, rc, pHostCtx->uContextID, cbRead, uHandleID, uFlags); 754 753 … … 973 972 974 973 default: 975 VBoxServiceVerbose(3, "Unsupported message from host, uMsg=%RU32, cParms=%RU32\n", 974 rc = VbglR3GuestCtrlMsgSkip(pHostCtx->uClientID); 975 VBoxServiceVerbose(3, "Unsupported message (uMsg=%RU32, cParms=%RU32) from host, skipping\n", 976 976 uMsg, pHostCtx->uNumParms); 977 /* Don't terminate here; just wait for the next message. */978 977 break; 979 978 } … … 1008 1007 } 1009 1008 else 1009 { 1010 1010 VBoxServiceError("Error connecting to guest control service, rc=%Rrc\n", rc); 1011 return rc; 1012 } 1011 1013 1012 1014 /* Let caller know that we're done initializing. */ 1013 int rc2= RTThreadUserSignal(RTThreadSelf());1014 if (RT_ SUCCESS(rc))1015 r c = rc2;1015 rc = RTThreadUserSignal(RTThreadSelf()); 1016 if (RT_FAILURE(rc)) 1017 return rc; 1016 1018 1017 1019 bool fProcessAlive = true; … … 1019 1021 RT_ZERO(ProcessStatus); 1020 1022 1023 int rcWait; 1021 1024 if (RT_SUCCESS(rc)) 1022 1025 { … … 1024 1027 uint64_t u64TimeoutStart = 0; 1025 1028 1026 int rcWait;1027 1029 for (;;) 1028 1030 { … … 1048 1050 uSessionID); 1049 1051 u64TimeoutStart = RTTimeMilliTS(); 1052 continue; /* Don't waste time on waiting. */ 1050 1053 } 1051 1054 if (RTTimeMilliTS() - u64TimeoutStart > uTimeoutsMS) … … 1084 1087 } 1085 1088 1086 VBoxServiceVerbose(2, "Guest session ID=%RU32 process has been killed with rc=%Rc\n",1089 VBoxServiceVerbose(2, "Guest session ID=%RU32 process termination resulted in rc=%Rrc\n", 1087 1090 uSessionID, rc); 1088 1091 … … 1092 1095 else 1093 1096 { 1094 switch (ProcessStatus.enmReason) 1095 { 1096 case RTPROCEXITREASON_NORMAL: 1097 uSessionStatus = GUEST_SESSION_NOTIFYTYPE_TEN; 1098 break; 1099 1100 case RTPROCEXITREASON_ABEND: 1101 uSessionStatus = GUEST_SESSION_NOTIFYTYPE_TEA; 1102 break; 1103 1104 case RTPROCEXITREASON_SIGNAL: 1105 uSessionStatus = GUEST_SESSION_NOTIFYTYPE_TES; 1106 break; 1107 1108 default: 1109 AssertMsgFailed(("Unhandled process termination reason (%ld)", 1110 ProcessStatus.enmReason)); 1111 uSessionStatus = GUEST_SESSION_NOTIFYTYPE_TEA; 1112 break; 1113 } 1114 } 1115 1116 VBoxServiceVerbose(3, "Guest session ID=%RU32 thread ended with sessionStatus=%ld, sessionRc=%Rrc\n", 1097 if (RT_SUCCESS(rcWait)) 1098 { 1099 switch (ProcessStatus.enmReason) 1100 { 1101 case RTPROCEXITREASON_NORMAL: 1102 uSessionStatus = GUEST_SESSION_NOTIFYTYPE_TEN; 1103 break; 1104 1105 case RTPROCEXITREASON_ABEND: 1106 uSessionStatus = GUEST_SESSION_NOTIFYTYPE_TEA; 1107 break; 1108 1109 case RTPROCEXITREASON_SIGNAL: 1110 uSessionStatus = GUEST_SESSION_NOTIFYTYPE_TES; 1111 break; 1112 1113 default: 1114 AssertMsgFailed(("Unhandled process termination reason (%ld)\n", 1115 ProcessStatus.enmReason)); 1116 uSessionStatus = GUEST_SESSION_NOTIFYTYPE_TEA; 1117 break; 1118 } 1119 } 1120 else 1121 { 1122 /* If we didn't find the guest process anymore, just assume it 1123 * terminated normally. */ 1124 uSessionStatus = GUEST_SESSION_NOTIFYTYPE_TEN; 1125 } 1126 } 1127 1128 VBoxServiceVerbose(3, "Guest session ID=%RU32 thread ended with sessionStatus=%RU32, sessionRc=%Rrc\n", 1117 1129 uSessionID, uSessionStatus, uSessionRc); 1118 1130 … … 1120 1132 Assert(uSessionStatus != GUEST_SESSION_NOTIFYTYPE_UNDEFINED); 1121 1133 VBGLR3GUESTCTRLCMDCTX ctx = { uClientID, VBOX_GUESTCTRL_CONTEXTID_MAKE_SESSION(uSessionID) }; 1122 rc2 = VbglR3GuestCtrlSessionNotify(&ctx,1123 uSessionStatus, uSessionRc);1134 int rc2 = VbglR3GuestCtrlSessionNotify(&ctx, 1135 uSessionStatus, uSessionRc); 1124 1136 if (RT_FAILURE(rc2)) 1125 1137 VBoxServiceError("Reporting session ID=%RU32 final status failed with rc=%Rrc\n", … … 1254 1266 AssertPtrReturn(pSession, NULL); 1255 1267 1256 PVBOXSERVICECTRLPROCESS p Thread= NULL;1257 int rc = RTCritSectEnter(&pSession-> csControlThreads);1258 if (RT_SUCCESS(rc)) 1259 { 1260 PVBOXSERVICECTRLPROCESS p ThreadCur;1261 RTListForEach(&pSession->lst ControlThreadsActive, pThreadCur, VBOXSERVICECTRLPROCESS, Node)1262 { 1263 if (p ThreadCur->uPID == uPID)1264 { 1265 rc = RTCritSectEnter(&p ThreadCur->CritSect);1268 PVBOXSERVICECTRLPROCESS pProcess = NULL; 1269 int rc = RTCritSectEnter(&pSession->CritSect); 1270 if (RT_SUCCESS(rc)) 1271 { 1272 PVBOXSERVICECTRLPROCESS pCurProcess; 1273 RTListForEach(&pSession->lstProcessesActive, pCurProcess, VBOXSERVICECTRLPROCESS, Node) 1274 { 1275 if (pCurProcess->uPID == uPID) 1276 { 1277 rc = RTCritSectEnter(&pCurProcess->CritSect); 1266 1278 if (RT_SUCCESS(rc)) 1267 p Thread = pThreadCur;1279 pProcess = pCurProcess; 1268 1280 break; 1269 1281 } 1270 1282 } 1271 1283 1272 int rc2 = RTCritSectLeave(&pSession-> csControlThreads);1284 int rc2 = RTCritSectLeave(&pSession->CritSect); 1273 1285 if (RT_SUCCESS(rc)) 1274 1286 rc = rc2; 1275 1287 } 1276 1288 1277 return p Thread;1289 return pProcess; 1278 1290 } 1279 1291 … … 1286 1298 pSession->StartupInfo.uSessionID); 1287 1299 1288 /* 1289 * Close all guest processes. 1290 */ 1291 1292 /* Signal all guest processes in the active list that we want to shutdown. */ 1293 PVBOXSERVICECTRLPROCESS pProcess; 1294 RTListForEach(&pSession->lstControlThreadsActive, pProcess, VBOXSERVICECTRLPROCESS, Node) 1295 GstCntlProcessStop(pProcess); 1296 1297 /* Wait for all active threads to shutdown and destroy the active thread list. */ 1298 pProcess = RTListGetFirst(&pSession->lstControlThreadsActive, VBOXSERVICECTRLPROCESS, Node); 1299 while (pProcess) 1300 { 1301 PVBOXSERVICECTRLPROCESS pNext = RTListNodeGetNext(&pProcess->Node, VBOXSERVICECTRLPROCESS, Node); 1302 bool fLast = RTListNodeIsLast(&pSession->lstControlThreadsActive, &pProcess->Node); 1303 1304 int rc2 = GstCntlProcessWait(pProcess, 1305 30 * 1000 /* Wait 30 seconds max. */, 1306 NULL /* rc */); 1307 if (RT_FAILURE(rc2)) 1308 { 1309 VBoxServiceError("Guest process thread failed to stop; rc=%Rrc\n", rc2); 1310 /* Keep going. */ 1311 } 1312 1313 if (fLast) 1314 break; 1315 1316 pProcess = pNext; 1317 } 1318 1319 int rc = GstCntlSessionReapProcesses(pSession); 1320 if (RT_FAILURE(rc)) 1321 VBoxServiceError("Reaping inactive threads failed with rc=%Rrc\n", rc); 1322 1323 AssertMsg(RTListIsEmpty(&pSession->lstControlThreadsActive), 1324 ("Guest process active thread list still contains entries when it should not\n")); 1325 AssertMsg(RTListIsEmpty(&pSession->lstControlThreadsInactive), 1326 ("Guest process inactive thread list still contains entries when it should not\n")); 1327 1328 /* 1329 * Close all left guest files. 1330 */ 1331 PVBOXSERVICECTRLFILE pFile; 1332 pFile = RTListGetFirst(&pSession->lstFiles, VBOXSERVICECTRLFILE, Node); 1333 while (pFile) 1334 { 1335 PVBOXSERVICECTRLFILE pNext = RTListNodeGetNext(&pFile->Node, VBOXSERVICECTRLFILE, Node); 1336 bool fLast = RTListNodeIsLast(&pSession->lstFiles, &pFile->Node); 1337 1338 int rc2 = gstcntlSessionFileDestroy(pFile); 1339 if (RT_FAILURE(rc2)) 1340 { 1341 VBoxServiceError("Unable to close file \"%s\"; rc=%Rrc\n", 1342 pFile->szName, rc2); 1343 /* Keep going. */ 1344 } 1345 1346 if (fLast) 1347 break; 1348 1349 pFile = pNext; 1350 } 1351 1352 AssertMsg(RTListIsEmpty(&pSession->lstFiles), 1353 ("Guest file list still contains entries when it should not\n")); 1300 int rc = RTCritSectEnter(&pSession->CritSect); 1301 if (RT_SUCCESS(rc)) 1302 { 1303 /* 1304 * Close all guest processes. 1305 */ 1306 1307 /* Signal all guest processes in the active list that we want to shutdown. */ 1308 PVBOXSERVICECTRLPROCESS pProcess; 1309 RTListForEach(&pSession->lstProcessesActive, pProcess, VBOXSERVICECTRLPROCESS, Node) 1310 GstCntlProcessStop(pProcess); 1311 1312 /* Wait for all active threads to shutdown and destroy the active thread list. */ 1313 pProcess = RTListGetFirst(&pSession->lstProcessesActive, VBOXSERVICECTRLPROCESS, Node); 1314 while (pProcess) 1315 { 1316 PVBOXSERVICECTRLPROCESS pNext = RTListNodeGetNext(&pProcess->Node, VBOXSERVICECTRLPROCESS, Node); 1317 bool fLast = RTListNodeIsLast(&pSession->lstProcessesActive, &pProcess->Node); 1318 1319 int rc2 = GstCntlProcessWait(pProcess, 1320 30 * 1000 /* Wait 30 seconds max. */, 1321 NULL /* rc */); 1322 if (RT_FAILURE(rc2)) 1323 { 1324 VBoxServiceError("Guest process thread failed to stop; rc=%Rrc\n", rc2); 1325 if (RT_SUCCESS(rc)) 1326 rc = rc2; 1327 /* Keep going. */ 1328 } 1329 1330 rc2 = GstCntlProcessFree(pProcess); 1331 if (RT_FAILURE(rc2)) 1332 { 1333 VBoxServiceError("Guest process thread failed to free; rc=%Rrc\n", rc2); 1334 if (RT_SUCCESS(rc)) 1335 rc = rc2; 1336 /* Keep going. */ 1337 } 1338 1339 RTListNodeRemove(&pProcess->Node); 1340 1341 if (fLast) 1342 break; 1343 1344 pProcess = pNext; 1345 } 1346 1347 /*rc = GstCntlSessionReapProcesses(pSession); 1348 if (RT_FAILURE(rc)) 1349 VBoxServiceError("Reaping inactive threads failed with rc=%Rrc\n", rc);*/ 1350 1351 AssertMsg(RTListIsEmpty(&pSession->lstProcessesActive), 1352 ("Guest process active thread list still contains entries when it should not\n")); 1353 /*AssertMsg(RTListIsEmpty(&pSession->lstProcessesInactive), 1354 ("Guest process inactive thread list still contains entries when it should not\n"));*/ 1355 1356 /* 1357 * Close all left guest files. 1358 */ 1359 PVBOXSERVICECTRLFILE pFile; 1360 pFile = RTListGetFirst(&pSession->lstFiles, VBOXSERVICECTRLFILE, Node); 1361 while (pFile) 1362 { 1363 PVBOXSERVICECTRLFILE pNext = RTListNodeGetNext(&pFile->Node, VBOXSERVICECTRLFILE, Node); 1364 bool fLast = RTListNodeIsLast(&pSession->lstFiles, &pFile->Node); 1365 1366 int rc2 = gstcntlSessionFileDestroy(pFile); 1367 if (RT_FAILURE(rc2)) 1368 { 1369 VBoxServiceError("Unable to close file \"%s\"; rc=%Rrc\n", 1370 pFile->szName, rc2); 1371 if (RT_SUCCESS(rc)) 1372 rc = rc2; 1373 /* Keep going. */ 1374 } 1375 1376 if (fLast) 1377 break; 1378 1379 pFile = pNext; 1380 } 1381 1382 AssertMsg(RTListIsEmpty(&pSession->lstFiles), 1383 ("Guest file list still contains entries when it should not\n")); 1384 1385 int rc2 = RTCritSectLeave(&pSession->CritSect); 1386 if (RT_SUCCESS(rc)) 1387 rc = rc2; 1388 } 1354 1389 1355 1390 return rc; … … 1364 1399 1365 1400 /* Destroy critical section. */ 1366 RTCritSectDelete(&pSession-> csControlThreads);1401 RTCritSectDelete(&pSession->CritSect); 1367 1402 1368 1403 return rc; … … 1441 1476 AssertPtrReturn(pSession, VERR_INVALID_POINTER); 1442 1477 1443 RTListInit(&pSession->lst ControlThreadsActive);1444 RTListInit(&pSession->lst ControlThreadsInactive);1478 RTListInit(&pSession->lstProcessesActive); 1479 RTListInit(&pSession->lstProcessesInactive); 1445 1480 RTListInit(&pSession->lstFiles); 1446 1481 … … 1457 1492 1458 1493 /* Init critical section for protecting the thread lists. */ 1459 int rc = RTCritSectInit(&pSession-> csControlThreads);1494 int rc = RTCritSectInit(&pSession->CritSect); 1460 1495 AssertRC(rc); 1461 1496 … … 1481 1516 AssertPtrReturn(pProcess, VERR_INVALID_POINTER); 1482 1517 1483 int rc = RTCritSectEnter(&pSession-> csControlThreads);1518 int rc = RTCritSectEnter(&pSession->CritSect); 1484 1519 if (RT_SUCCESS(rc)) 1485 1520 { … … 1491 1526 { 1492 1527 case VBOXSERVICECTRLTHREADLIST_STOPPED: 1493 pAnchor = &pSession->lst ControlThreadsInactive;1528 pAnchor = &pSession->lstProcessesInactive; 1494 1529 break; 1495 1530 1496 1531 case VBOXSERVICECTRLTHREADLIST_RUNNING: 1497 pAnchor = &pSession->lst ControlThreadsActive;1532 pAnchor = &pSession->lstProcessesActive; 1498 1533 break; 1499 1534 1500 1535 default: 1501 AssertMsgFailed(("Unknown list type: %u", enmList)); 1536 AssertMsgFailed(("Unknown list type: %u\n", 1537 enmList)); 1502 1538 break; 1503 1539 } … … 1520 1556 } 1521 1557 1522 int rc2 = RTCritSectLeave(&pSession-> csControlThreads);1558 int rc2 = RTCritSectLeave(&pSession->CritSect); 1523 1559 if (RT_SUCCESS(rc)) 1524 1560 rc = rc2; … … 1546 1582 AssertPtrReturn(pbAllowed, VERR_INVALID_POINTER); 1547 1583 1548 int rc = RTCritSectEnter(&pSession-> csControlThreads);1584 int rc = RTCritSectEnter(&pSession->CritSect); 1549 1585 if (RT_SUCCESS(rc)) 1550 1586 { … … 1557 1593 { 1558 1594 uint32_t uProcsRunning = 0; 1559 PVBOXSERVICECTRLPROCESS p Thread;1560 RTListForEach(&pSession->lst ControlThreadsActive, pThread, VBOXSERVICECTRLPROCESS, Node)1595 PVBOXSERVICECTRLPROCESS pProcess; 1596 RTListForEach(&pSession->lstProcessesActive, pProcess, VBOXSERVICECTRLPROCESS, Node) 1561 1597 uProcsRunning++; 1562 1598 … … 1575 1611 *pbAllowed = !fLimitReached; 1576 1612 1577 int rc2 = RTCritSectLeave(&pSession-> csControlThreads);1613 int rc2 = RTCritSectLeave(&pSession->CritSect); 1578 1614 if (RT_SUCCESS(rc)) 1579 1615 rc = rc2; … … 1583 1619 } 1584 1620 1585 1621 #if 0 1586 1622 /** 1587 1623 * Reaps all inactive guest process threads. 1624 * Does not do locking; this is the job of the caller. 1588 1625 * 1589 1626 * @return IPRT status code. … … 1593 1630 AssertPtrReturn(pSession, VERR_INVALID_POINTER); 1594 1631 1595 int rc = RTCritSectEnter(&pSession->csControlThreads); 1596 if (RT_SUCCESS(rc)) 1597 { 1598 PVBOXSERVICECTRLPROCESS pThread = 1599 RTListGetFirst(&pSession->lstControlThreadsInactive, VBOXSERVICECTRLPROCESS, Node); 1600 while (pThread) 1601 { 1602 PVBOXSERVICECTRLPROCESS pNext = RTListNodeGetNext(&pThread->Node, VBOXSERVICECTRLPROCESS, Node); 1603 bool fLast = RTListNodeIsLast(&pSession->lstControlThreadsInactive, &pThread->Node); 1604 int rc2 = GstCntlProcessWait(pThread, 30 * 1000 /* 30 seconds max. */, 1605 NULL /* rc */); 1606 if (RT_SUCCESS(rc2)) 1607 { 1608 RTListNodeRemove(&pThread->Node); 1609 1610 rc2 = GstCntlProcessFree(pThread); 1611 if (RT_FAILURE(rc2)) 1612 { 1613 VBoxServiceError("Freeing guest process thread failed with rc=%Rrc\n", rc2); 1614 if (RT_SUCCESS(rc)) /* Keep original failure. */ 1615 rc = rc2; 1616 } 1617 } 1618 else 1619 VBoxServiceError("Waiting on guest process thread failed with rc=%Rrc\n", rc2); 1620 /* Keep going. */ 1621 1622 if (fLast) 1623 break; 1624 1625 pThread = pNext; 1626 } 1627 1628 int rc2 = RTCritSectLeave(&pSession->csControlThreads); 1629 if (RT_SUCCESS(rc)) 1630 rc = rc2; 1632 PVBOXSERVICECTRLPROCESS pThread = 1633 RTListGetFirst(&pSession->lstProcessesInactive, VBOXSERVICECTRLPROCESS, Node); 1634 while (pThread) 1635 { 1636 PVBOXSERVICECTRLPROCESS pNext = RTListNodeGetNext(&pThread->Node, VBOXSERVICECTRLPROCESS, Node); 1637 bool fLast = RTListNodeIsLast(&pSession->lstProcessesInactive, &pThread->Node); 1638 int rc2 = GstCntlProcessWait(pThread, 30 * 1000 /* 30 seconds max. */, 1639 NULL /* rc */); 1640 if (RT_SUCCESS(rc2)) 1641 { 1642 RTListNodeRemove(&pThread->Node); 1643 1644 rc2 = GstCntlProcessFree(pThread); 1645 if (RT_FAILURE(rc2)) 1646 { 1647 VBoxServiceError("Freeing guest process thread failed with rc=%Rrc\n", rc2); 1648 if (RT_SUCCESS(rc)) /* Keep original failure. */ 1649 rc = rc2; 1650 } 1651 } 1652 else 1653 VBoxServiceError("Waiting on guest process thread failed with rc=%Rrc\n", rc2); 1654 /* Keep going. */ 1655 1656 if (fLast) 1657 break; 1658 1659 pThread = pNext; 1631 1660 } 1632 1661 … … 1634 1663 return rc; 1635 1664 } 1665 #endif 1636 1666 1637 1667 … … 1700 1730 * Optional. 1701 1731 */ 1702 int GstCntlSessionThread Open(PRTLISTANCHOR pList,1703 const PVBOXSERVICECTRLSESSIONSTARTUPINFO pSessionStartupInfo,1704 PVBOXSERVICECTRLSESSIONTHREAD *ppSessionThread)1732 int GstCntlSessionThreadCreate(PRTLISTANCHOR pList, 1733 const PVBOXSERVICECTRLSESSIONSTARTUPINFO pSessionStartupInfo, 1734 PVBOXSERVICECTRLSESSIONTHREAD *ppSessionThread) 1705 1735 { 1706 1736 AssertPtrReturn(pList, VERR_INVALID_POINTER); … … 1716 1746 if (pSessionCur->StartupInfo.uSessionID == pSessionStartupInfo->uSessionID) 1717 1747 { 1718 AssertMsgFailed(("Guest session %RU32 (%p) already exists when it should not",1748 AssertMsgFailed(("Guest session thread ID=%RU32 (%p) already exists when it should not\n", 1719 1749 pSessionCur->StartupInfo.uSessionID, pSessionCur)); 1720 1750 return VERR_ALREADY_EXISTS; … … 1724 1754 int rc = VINF_SUCCESS; 1725 1755 1726 PVBOXSERVICECTRLSESSIONTHREAD pSession = (PVBOXSERVICECTRLSESSIONTHREAD)RTMemAllocZ(sizeof(VBOXSERVICECTRLSESSIONTHREAD)); 1727 if (pSession) 1756 /* Static counter to help tracking session thread <-> process relations. */ 1757 static uint32_t s_uCtrlSessionThread = 0; 1758 if (s_uCtrlSessionThread++ == UINT32_MAX) 1759 s_uCtrlSessionThread = 0; /* Wrap around to not let IPRT freak out. */ 1760 1761 PVBOXSERVICECTRLSESSIONTHREAD pSessionThread = 1762 (PVBOXSERVICECTRLSESSIONTHREAD)RTMemAllocZ(sizeof(VBOXSERVICECTRLSESSIONTHREAD)); 1763 if (pSessionThread) 1728 1764 { 1729 1765 /* Copy over session startup info. */ 1730 memcpy(&pSession->StartupInfo, pSessionStartupInfo, sizeof(VBOXSERVICECTRLSESSIONSTARTUPINFO)); 1731 1732 pSession->fShutdown = false; 1733 pSession->fStarted = false; 1734 pSession->fStopped = false; 1766 memcpy(&pSessionThread->StartupInfo, pSessionStartupInfo, 1767 sizeof(VBOXSERVICECTRLSESSIONSTARTUPINFO)); 1768 1769 pSessionThread->fShutdown = false; 1770 pSessionThread->fStarted = false; 1771 pSessionThread->fStopped = false; 1735 1772 1736 1773 /* Is this an anonymous session? */ 1737 1774 /* Anonymous sessions run with the same privileges as the main VBoxService executable. */ 1738 bool fAnonymous = !RT_BOOL(strlen(pSession ->StartupInfo.szUser));1775 bool fAnonymous = !RT_BOOL(strlen(pSessionThread->StartupInfo.szUser)); 1739 1776 if (fAnonymous) 1740 1777 { 1741 Assert(!strlen(pSession ->StartupInfo.szPassword));1742 Assert(!strlen(pSession ->StartupInfo.szDomain));1778 Assert(!strlen(pSessionThread->StartupInfo.szPassword)); 1779 Assert(!strlen(pSessionThread->StartupInfo.szDomain)); 1743 1780 1744 1781 VBoxServiceVerbose(3, "New anonymous guest session ID=%RU32 created, uFlags=%x, using protocol %RU32\n", … … 1762 1799 } 1763 1800 1764 rc = RTCritSectInit(&pSession ->CritSect);1801 rc = RTCritSectInit(&pSessionThread->CritSect); 1765 1802 AssertRC(rc); 1766 1803 … … 1773 1810 if (!fAnonymous) 1774 1811 { 1775 if (!RTStrPrintf(szParmUserName, sizeof(szParmUserName), "--user name=%s", pSession->StartupInfo.szUser))1812 if (!RTStrPrintf(szParmUserName, sizeof(szParmUserName), "--user=%s", pSessionThread->StartupInfo.szUser)) 1776 1813 rc = VERR_BUFFER_OVERFLOW; 1777 1814 } 1778 1815 char szParmSessionID[32]; 1779 1816 if (RT_SUCCESS(rc) && !RTStrPrintf(szParmSessionID, sizeof(szParmSessionID), "--session-id=%RU32", 1780 pSession ->StartupInfo.uSessionID))1817 pSessionThread->StartupInfo.uSessionID)) 1781 1818 { 1782 1819 rc = VERR_BUFFER_OVERFLOW; … … 1784 1821 char szParmSessionProto[32]; 1785 1822 if (RT_SUCCESS(rc) && !RTStrPrintf(szParmSessionProto, sizeof(szParmSessionProto), "--session-proto=%RU32", 1786 pSession ->StartupInfo.uProtocol))1823 pSessionThread->StartupInfo.uProtocol)) 1787 1824 { 1788 1825 rc = VERR_BUFFER_OVERFLOW; 1789 1826 } 1790 1827 #ifdef DEBUG 1828 char szParmThreadId[32]; 1829 if (RT_SUCCESS(rc) && !RTStrPrintf(szParmThreadId, sizeof(szParmThreadId), "--thread-id=%RU32", 1830 s_uCtrlSessionThread)) 1831 { 1832 rc = VERR_BUFFER_OVERFLOW; 1833 } 1834 #endif /* DEBUG */ 1791 1835 if (RT_SUCCESS(rc)) 1792 1836 { 1793 1837 int iOptIdx = 0; /* Current index in argument vector. */ 1794 1838 1795 char const *papszArgs[ 8];1839 char const *papszArgs[16]; 1796 1840 papszArgs[iOptIdx++] = pszExeName; 1797 1841 papszArgs[iOptIdx++] = "guestsession"; 1798 1842 papszArgs[iOptIdx++] = szParmSessionID; 1799 1843 papszArgs[iOptIdx++] = szParmSessionProto; 1844 #ifdef DEBUG 1845 papszArgs[iOptIdx++] = szParmThreadId; 1846 #endif /* DEBUG */ 1800 1847 if (!fAnonymous) 1801 1848 papszArgs[iOptIdx++] = szParmUserName; … … 1829 1876 RTPathStripExt(pszLogFile); 1830 1877 char *pszLogSuffix; 1878 #ifndef DEBUG 1831 1879 if (RTStrAPrintf(&pszLogSuffix, "-%RU32-%s", 1832 1880 pSessionStartupInfo->uSessionID, … … 1835 1883 rc2 = VERR_NO_MEMORY; 1836 1884 } 1885 #else 1886 if (RTStrAPrintf(&pszLogSuffix, "-%RU32-%RU32-%s", 1887 pSessionStartupInfo->uSessionID, 1888 s_uCtrlSessionThread, 1889 pSessionStartupInfo->szUser) < 0) 1890 { 1891 rc2 = VERR_NO_MEMORY; 1892 } 1893 #endif /* DEBUG */ 1837 1894 else 1838 1895 { … … 1940 1997 hStdOutAndErr.enmType = RTHANDLETYPE_FILE; 1941 1998 1942 /** @todo Do we need a custom/cloned environment block?*/1999 /** @todo Set custom/cloned guest session environment block. */ 1943 2000 rc = RTProcCreateEx(pszExeName, papszArgs, RTENV_DEFAULT, uProcFlags, 1944 2001 &hStdIn, &hStdOutAndErr, &hStdOutAndErr, 1945 !fAnonymous ? pSession ->StartupInfo.szUser : NULL,1946 !fAnonymous ? pSession ->StartupInfo.szPassword : NULL,1947 &pSession ->hProcess);2002 !fAnonymous ? pSessionThread->StartupInfo.szUser : NULL, 2003 !fAnonymous ? pSessionThread->StartupInfo.szPassword : NULL, 2004 &pSessionThread->hProcess); 1948 2005 1949 2006 RTFileClose(hStdOutAndErr.u.hFile); 1950 2007 } 1951 2008 1952 RTFileClose(hStd OutAndErr.u.hFile);2009 RTFileClose(hStdIn.u.hFile); 1953 2010 } 1954 2011 #endif … … 1961 2018 { 1962 2019 /* Start session thread. */ 1963 static uint32_t s_uCtrlSessionThread = 0; 1964 if (s_uCtrlSessionThread++ == UINT32_MAX) 1965 s_uCtrlSessionThread = 0; /* Wrap around to not let IPRT freak out. */ 1966 rc = RTThreadCreateF(&pSession->Thread, gstcntlSessionThread, 1967 pSession /*pvUser*/, 0 /*cbStack*/, 2020 rc = RTThreadCreateF(&pSessionThread->Thread, gstcntlSessionThread, 2021 pSessionThread /*pvUser*/, 0 /*cbStack*/, 1968 2022 RTTHREADTYPE_DEFAULT, RTTHREADFLAGS_WAITABLE, "sess%u", s_uCtrlSessionThread); 1969 2023 if (RT_FAILURE(rc)) … … 1974 2028 { 1975 2029 /* Wait for the thread to initialize. */ 1976 rc = RTThreadUserWait(pSession->Thread, 60 * 1000 /* 60s timeout */); 1977 AssertRC(rc); 1978 if ( ASMAtomicReadBool(&pSession->fShutdown) 2030 rc = RTThreadUserWait(pSessionThread->Thread, 60 * 1000 /* 60s timeout */); 2031 if ( ASMAtomicReadBool(&pSessionThread->fShutdown) 1979 2032 || RT_FAILURE(rc)) 1980 2033 { 1981 2034 VBoxServiceError("Thread for session ID=%RU32 failed to start, rc=%Rrc\n", 1982 pSession ->StartupInfo.uSessionID, rc);2035 pSessionThread->StartupInfo.uSessionID, rc); 1983 2036 if (RT_SUCCESS(rc)) 1984 2037 rc = VERR_CANT_CREATE; /** @todo Find a better rc. */ … … 1986 2039 else 1987 2040 { 1988 ASMAtomicXchgBool(&pSession->fStarted, true); 2041 VBoxServiceVerbose(2, "Thread for session ID=%RU32 started\n", 2042 pSessionThread->StartupInfo.uSessionID); 2043 2044 ASMAtomicXchgBool(&pSessionThread->fStarted, true); 1989 2045 1990 2046 /* Add session to list. */ 1991 /* rc = */ RTListAppend(pList, &pSession ->Node);2047 /* rc = */ RTListAppend(pList, &pSessionThread->Node); 1992 2048 if (ppSessionThread) /* Return session if wanted. */ 1993 *ppSessionThread = pSession ;2049 *ppSessionThread = pSessionThread; 1994 2050 } 1995 2051 } … … 1998 2054 if (RT_FAILURE(rc)) 1999 2055 { 2000 RTMemFree(pSession );2056 RTMemFree(pSessionThread); 2001 2057 } 2002 2058 } … … 2004 2060 rc = VERR_NO_MEMORY; 2005 2061 2006 VBoxServiceVerbose(3, "Forking returned returned rc=%Rrc\n", rc);2062 VBoxServiceVerbose(3, "Forking session thread returned returned rc=%Rrc\n", rc); 2007 2063 return rc; 2008 2064 } … … 2010 2066 2011 2067 /** 2012 * Closes a formerly opened guest session and removes it from 2013 * the session list. 2068 * Waits for a formerly opened guest session process to close. 2014 2069 * 2015 2070 * @return IPRT status code. 2016 * @param pThread Guest session thread to close. 2071 * @param pThread Guest session thread to wait for. 2072 * @param uTimeoutMS Waiting timeout (in ms). 2017 2073 * @param uFlags Closing flags. 2018 2074 */ 2019 int GstCntlSessionThreadClose(PVBOXSERVICECTRLSESSIONTHREAD pThread, uint32_t uFlags) 2075 int GstCntlSessionThreadWait(PVBOXSERVICECTRLSESSIONTHREAD pThread, 2076 uint32_t uTimeoutMS, uint32_t uFlags) 2020 2077 { 2021 2078 AssertPtrReturn(pThread, VERR_INVALID_POINTER); … … 2024 2081 if (pThread->Thread == NIL_RTTHREAD) 2025 2082 { 2026 AssertMsgFailed(("Guest session thread of session %p does not exist when it should ",2083 AssertMsgFailed(("Guest session thread of session %p does not exist when it should\n", 2027 2084 pThread)); 2028 2085 return VERR_NOT_FOUND; … … 2031 2088 int rc = VINF_SUCCESS; 2032 2089 2033 /* The fork should have received the same closing request, 2090 /* 2091 * The fork should have received the same closing request, 2034 2092 * so just wait 30s for the process to close. On timeout kill 2035 * it in a not so gentle manner. */ 2093 * it in a not so gentle manner. 2094 */ 2036 2095 if (ASMAtomicReadBool(&pThread->fStarted)) 2037 2096 { … … 2039 2098 ASMAtomicXchgBool(&pThread->fShutdown, true); 2040 2099 2041 uint32_t cMsTimeout = 30 * 1000; /** @todo 30s default. Make this configurable. Later. */2042 2043 2100 VBoxServiceVerbose(3, "Waiting for session thread ID=%RU32 to close (%RU32ms) ...\n", 2044 pThread->StartupInfo.uSessionID, cMsTimeout);2101 pThread->StartupInfo.uSessionID, uTimeoutMS); 2045 2102 2046 2103 int rcThread; 2047 rc = RTThreadWait(pThread->Thread, cMsTimeout, &rcThread);2104 rc = RTThreadWait(pThread->Thread, uTimeoutMS, &rcThread); 2048 2105 if (RT_FAILURE(rc)) 2049 2106 { … … 2054 2111 VBoxServiceVerbose(3, "Session thread ID=%RU32 ended with rc=%Rrc\n", 2055 2112 pThread->StartupInfo.uSessionID, rcThread); 2056 2057 /* Remove session from list and destroy object. */ 2058 RTListNodeRemove(&pThread->Node); 2059 2060 if (RT_FAILURE(rc)) 2061 VBoxServiceError("Closing session ID=%RU32 failed with rc=%Rrc\n", 2062 pThread->StartupInfo.uSessionID, rc); 2063 2064 RTMemFree(pThread); 2065 pThread = NULL; 2066 } 2067 2068 return rc; 2069 } 2070 2113 } 2114 2115 return rc; 2116 } 2117 2118 /** 2119 * Waits for the specified session thread to end and remove 2120 * it from the session thread list. 2121 * 2122 * @return IPRT status code. 2123 * @param pThread Session thread to destroy. 2124 * @param uFlags Closing flags. 2125 */ 2126 int GstCntlSessionThreadDestroy(PVBOXSERVICECTRLSESSIONTHREAD pThread, uint32_t uFlags) 2127 { 2128 AssertPtrReturn(pThread, VERR_INVALID_POINTER); 2129 2130 int rc = GstCntlSessionThreadWait(pThread, 2131 30 * 1000 /* 30s timeout */, uFlags); 2132 /** @todo Kill session process if still around? */ 2133 2134 /* Remove session from list and destroy object. */ 2135 RTListNodeRemove(&pThread->Node); 2136 RTMemFree(pThread); 2137 2138 return rc; 2139 } 2071 2140 2072 2141 /** … … 2077 2146 * @param uFlags Closing flags. 2078 2147 */ 2079 int GstCntlSessionThread CloseAll(PRTLISTANCHOR pList, uint32_t uFlags)2148 int GstCntlSessionThreadDestroyAll(PRTLISTANCHOR pList, uint32_t uFlags) 2080 2149 { 2081 2150 AssertPtrReturn(pList, VERR_INVALID_POINTER); … … 2083 2152 int rc = VINF_SUCCESS; 2084 2153 2085 PVBOXSERVICECTRLSESSIONTHREAD pSession Cur2154 PVBOXSERVICECTRLSESSIONTHREAD pSessionThread 2086 2155 = RTListGetFirst(pList, VBOXSERVICECTRLSESSIONTHREAD, Node); 2087 while (pSessionCur) 2088 { 2089 PVBOXSERVICECTRLSESSIONTHREAD pSessionNext = 2090 RTListGetNext(pList, pSessionCur, VBOXSERVICECTRLSESSIONTHREAD, Node); 2091 bool fLast = RTListNodeIsLast(pList, &pSessionCur->Node); 2092 2093 int rc2 = GstCntlSessionThreadClose(pSessionCur, uFlags); 2094 if (RT_SUCCESS(rc)) 2095 { 2096 rc = rc2; 2156 while (pSessionThread) 2157 { 2158 PVBOXSERVICECTRLSESSIONTHREAD pSessionThreadNext = 2159 RTListGetNext(pList, pSessionThread, VBOXSERVICECTRLSESSIONTHREAD, Node); 2160 bool fLast = RTListNodeIsLast(pList, &pSessionThread->Node); 2161 2162 int rc2 = GstCntlSessionThreadDestroy(pSessionThread, uFlags); 2163 if (RT_FAILURE(rc2)) 2164 { 2165 VBoxServiceError("Closing session thread failed with rc=%Rrc\n", rc2); 2166 if (RT_SUCCESS(rc)) 2167 rc = rc2; 2097 2168 /* Keep going. */ 2098 2169 } … … 2101 2172 break; 2102 2173 2103 pSessionCur = pSessionNext; 2104 } 2105 2106 return rc; 2107 } 2108 2174 pSessionThread = pSessionThreadNext; 2175 } 2176 2177 return rc; 2178 } 2109 2179 2110 2180 RTEXITCODE VBoxServiceControlSessionForkInit(int argc, char **argv) … … 2113 2183 { 2114 2184 { "--logfile", VBOXSERVICESESSIONOPT_LOG_FILE, RTGETOPT_REQ_STRING }, 2115 { "--user name",VBOXSERVICESESSIONOPT_USERNAME, RTGETOPT_REQ_STRING },2185 { "--user", VBOXSERVICESESSIONOPT_USERNAME, RTGETOPT_REQ_STRING }, 2116 2186 { "--session-id", VBOXSERVICESESSIONOPT_SESSION_ID, RTGETOPT_REQ_UINT32 }, 2117 2187 { "--session-proto", VBOXSERVICESESSIONOPT_SESSION_PROTO, RTGETOPT_REQ_UINT32 }, 2188 #ifdef DEBUG 2189 { "--thread-id", VBOXSERVICESESSIONOPT_THREAD_ID, RTGETOPT_REQ_UINT32 }, 2190 #endif /* DEBUG */ 2118 2191 { "--verbose", 'v', RTGETOPT_REQ_NOTHING } 2119 2192 }; … … 2157 2230 break; 2158 2231 2232 case VBOXSERVICESESSIONOPT_THREAD_ID: 2233 /* Not handled. */ 2234 break; 2235 2159 2236 /** @todo Implement help? */ 2160 2237
Note:
See TracChangeset
for help on using the changeset viewer.