VirtualBox

Ignore:
Timestamp:
Apr 8, 2013 9:40:42 PM (12 years ago)
Author:
vboxsync
svn:sync-xref-src-repo-rev:
84843
Message:

GuestCtrl: Implemented using (public) VirtualBox events instead of own callback mechanisms. Bugfixes for testcases (still work in progress).

Location:
trunk/src/VBox/Additions/common/VBoxService
Files:
4 edited

Legend:

Unmodified
Added
Removed
  • trunk/src/VBox/Additions/common/VBoxService/VBoxServiceControl.cpp

    r45109 r45415  
    274274                         * skip all not wanted messages here.
    275275                         */
    276                         VBoxServiceVerbose(3, "Skipping msg=%RU32 ...\n", uMsg);
     276                        rc = VbglR3GuestCtrlMsgSkip(g_uControlSvcClientID);
     277                        VBoxServiceVerbose(3, "Skipping msg=%RU32, rc=%Rrc\n", uMsg, rc);
    277278                    }
    278279                    break;
     
    323324                           pHostCtx->uClientID, pHostCtx->uProtocol);
    324325
    325         rc = GstCntlSessionThreadOpen(&g_lstControlSessionThreads,
    326                                       &ssInfo, NULL /* Session */);
     326        rc = GstCntlSessionThreadCreate(&g_lstControlSessionThreads,
     327                                        &ssInfo, NULL /* Session */);
    327328    }
    328329
     
    350351
    351352    uint32_t uSessionID, uFlags;
    352 
    353353    int rc = VbglR3GuestCtrlSessionGetClose(pHostCtx, &uFlags, &uSessionID);
    354354    if (RT_SUCCESS(rc))
     
    361361            if (pThread->StartupInfo.uSessionID == uSessionID)
    362362            {
    363                 rc = GstCntlSessionThreadClose(pThread, uFlags);
     363                rc = GstCntlSessionThreadDestroy(pThread, uFlags);
    364364                break;
    365365            }
    366366        }
    367 
     367#if 0
    368368        if (RT_FAILURE(rc))
    369369        {
     
    379379            }
    380380        }
    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
    385388    return rc;
    386389}
     
    420423    VBoxServiceVerbose(2, "Shutting down ...\n");
    421424
    422     int rc2 = GstCntlSessionThreadCloseAll(&g_lstControlSessionThreads,
    423                                            0 /* Flags */);
     425    int rc2 = GstCntlSessionThreadDestroyAll(&g_lstControlSessionThreads,
     426                                             0 /* Flags */);
    424427    if (RT_FAILURE(rc2))
    425428        VBoxServiceError("Closing session threads failed with rc=%Rrc\n", rc2);
  • trunk/src/VBox/Additions/common/VBoxService/VBoxServiceControl.h

    r45109 r45415  
    169169/**
    170170 * 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.
    172173 */
    173174typedef struct VBOXSERVICECTRLSESSIONTHREAD
     
    243244typedef struct VBOXSERVICECTRLSESSION
    244245{
     246    /* The session's startup information. */
    245247    VBOXSERVICECTRLSESSIONSTARTUPINFO
    246248                                    StartupInfo;
    247     /** List of active guest control threads (VBOXSERVICECTRLTHREAD). */
    248     RTLISTANCHOR                    lstControlThreadsActive;
    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). */
    250252    /** @todo Still needed? */
    251     RTLISTANCHOR                    lstControlThreadsInactive;
     253    RTLISTANCHOR                    lstProcessesInactive;
    252254    /** List of guest control files (VBOXSERVICECTRLFILE). */
    253255    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;
    257258    /** Session flags. */
    258259    uint32_t                        uFlags;
     
    330331    /** Critical section for thread-safe use. */
    331332    RTCRITSECT                      CritSect;
    332     /** @todo Document me! */
     333    /** Process startup information. */
     334    VBOXSERVICECTRLPROCSTARTUPINFO
     335                                    StartupInfo;
     336    /** The process' PID assigned by the guest OS. */
    333337    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 the
    341      *  guest process under. */
    342     char                           *pszUser;
    343     /** Password of specified user account. */
    344     char                           *pszPassword;
    345     /** Overall time limit (in ms) that the guest process
    346      *  is allowed to run. 0 for indefinite time. */
    347     uint32_t                        uTimeLimitMS;
    348338    /** 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. */
    350342    PVBOXSERVICECTRLREQUEST         pRequest;
    351343    /** StdIn pipe for addressing writes to the
     
    374366
    375367/* Guest session thread handling. */
    376 extern int                      GstCntlSessionThreadOpen(PRTLISTANCHOR pList, const PVBOXSERVICECTRLSESSIONSTARTUPINFO pSessionStartupInfo, PVBOXSERVICECTRLSESSIONTHREAD *ppSessionThread);
    377 extern int                      GstCntlSessionThreadClose(PVBOXSERVICECTRLSESSIONTHREAD pSession, uint32_t uFlags);
    378 extern int                      GstCntlSessionThreadCloseAll(PRTLISTANCHOR pList, uint32_t uFlags);
     368extern int                      GstCntlSessionThreadCreate(PRTLISTANCHOR pList, const PVBOXSERVICECTRLSESSIONSTARTUPINFO pSessionStartupInfo, PVBOXSERVICECTRLSESSIONTHREAD *ppSessionThread);
     369extern int                      GstCntlSessionThreadDestroy(PVBOXSERVICECTRLSESSIONTHREAD pSession, uint32_t uFlags);
     370extern int                      GstCntlSessionThreadDestroyAll(PRTLISTANCHOR pList, uint32_t uFlags);
    379371extern int                      GstCntlSessionThreadTerminate(PVBOXSERVICECTRLSESSIONTHREAD pSession);
    380372extern RTEXITCODE               VBoxServiceControlSessionForkInit(int argc, char **argv);
     
    392384extern int                      GstCntlProcessPerform(PVBOXSERVICECTRLPROCESS pProcess, PVBOXSERVICECTRLREQUEST pRequest);
    393385extern int                      GstCntlProcessStart(const PVBOXSERVICECTRLSESSION pSession, const PVBOXSERVICECTRLPROCSTARTUPINFO pStartupInfo, uint32_t uContext);
    394 extern int                      GstCntlProcessStop(const PVBOXSERVICECTRLPROCESS pThread);
    395 extern void                     GstCntlProcessRelease(const PVBOXSERVICECTRLPROCESS pThread);
    396 extern int                      GstCntlProcessWait(const PVBOXSERVICECTRLPROCESS pThread, RTMSINTERVAL msTimeout, int *prc);
    397 extern int                      GstCntlProcessFree(PVBOXSERVICECTRLPROCESS pThread);
     386extern int                      GstCntlProcessStop(PVBOXSERVICECTRLPROCESS pProcess);
     387extern void                     GstCntlProcessRelease(const PVBOXSERVICECTRLPROCESS pProcess);
     388extern int                      GstCntlProcessWait(const PVBOXSERVICECTRLPROCESS pProcess, RTMSINTERVAL msTimeout, int *pRc);
     389extern int                      GstCntlProcessFree(PVBOXSERVICECTRLPROCESS pProcess);
    398390/* Process request handling. */
    399391extern int                      GstCntlProcessRequestAlloc(PVBOXSERVICECTRLREQUEST *ppReq, VBOXSERVICECTRLREQUESTTYPE enmType);
  • trunk/src/VBox/Additions/common/VBoxService/VBoxServiceControlProcess.cpp

    r45109 r45415  
    8787        return rc;
    8888
    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;
    14699
    147100    if (RT_FAILURE(rc)) /* Clean up on failure. */
     
    161114    AssertPtrReturn(pProcess, VERR_INVALID_POINTER);
    162115
    163     VBoxServiceVerbose(3, "[PID %u]: Freeing ...\n",
     116    VBoxServiceVerbose(3, "[PID %RU32]: Freeing ...\n",
    164117                       pProcess->uPID);
    165118
    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     }
    187119
    188120    /*
     
    198130    pProcess = NULL;
    199131
    200     return rc;
     132    return VINF_SUCCESS;
    201133}
    202134
     
    207139 *
    208140 * @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 */
     143int 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);
    228151    if (RT_SUCCESS(rc))
    229152    {
    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
    237180    return rc;
    238181}
     
    242185 * Releases (unlocks) a previously locked guest process.
    243186 *
    244  * @param   pThread                 Thread to unlock.
    245  */
    246 void GstCntlProcessRelease(const PVBOXSERVICECTRLPROCESS pThread)
    247 {
    248     AssertPtr(pThread);
    249 
    250     int rc = RTCritSectLeave(&pThread->CritSect);
     187 * @param   pProcess            Process to unlock.
     188 */
     189void GstCntlProcessRelease(const PVBOXSERVICECTRLPROCESS pProcess)
     190{
     191    AssertPtrReturnVoid(pProcess);
     192
     193    int rc = RTCritSectLeave(&pProcess->CritSect);
    251194    AssertRC(rc);
    252195}
     
    255198/**
    256199 * 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.
    260204 * @param   RTMSINTERVAL        Timeout in ms to wait for shutdown.
    261  * @param   prc                 Where to store the thread's return code. Optional.
    262  */
    263 int GstCntlProcessWait(const PVBOXSERVICECTRLPROCESS pThread,
    264                        RTMSINTERVAL msTimeout, int *prc)
    265 {
    266     AssertPtrReturn(pThread, VERR_INVALID_POINTER);
    267     /* prc is optional. */
     205 * @param   pRc                 Where to store the thread's return code. Optional.
     206 */
     207int GstCntlProcessWait(const PVBOXSERVICECTRLPROCESS pProcess,
     208                       RTMSINTERVAL msTimeout, int *pRc)
     209{
     210    AssertPtrReturn(pProcess, VERR_INVALID_POINTER);
     211    /* pRc is optional. */
    268212
    269213    int rc = VINF_SUCCESS;
    270     if (   pThread->Thread != NIL_RTTHREAD
    271         && ASMAtomicReadBool(&pThread->fStarted))
    272     {
    273         VBoxServiceVerbose(2, "[PID %u]: Waiting for shutdown of pThread=0x%p = \"%s\"...\n",
    274                            pThread->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);
    275219
    276220        /* Wait a bit ... */
    277221        int rcThread;
    278         rc = RTThreadWait(pThread->Thread, msTimeout, &rcThread);
     222        rc = RTThreadWait(pProcess->Thread, msTimeout, &rcThread);
    279223        if (RT_FAILURE(rc))
    280224        {
    281             VBoxServiceError("[PID %u]: Waiting for shutting down thread returned error rc=%Rrc\n",
    282                              pThread->uPID, rc);
     225            VBoxServiceError("[PID %RU32]: Waiting for shutting down thread returned error rc=%Rrc\n",
     226                             pProcess->uPID, rc);
    283227        }
    284228        else
    285229        {
    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
    292237    return rc;
    293238}
     
    368313        && cbReadable)
    369314    {
    370         VBoxServiceVerbose(3, "gstcntlProcessHandleOutputError: idPollHnd=%u has %ld bytes left, vetoing close\n",
     315        VBoxServiceVerbose(3, "gstcntlProcessHandleOutputError: idPollHnd=%RU32 has %zu bytes left, vetoing close\n",
    371316                           idPollHnd, cbReadable);
    372317
     
    377322    }
    378323    else
    379         VBoxServiceVerbose(3, "gstcntlProcessHandleOutputError: idPollHnd=%u will be closed\n",
     324        VBoxServiceVerbose(3, "gstcntlProcessHandleOutputError: idPollHnd=%RU32 will be closed\n",
    380325                           idPollHnd);
    381326
     
    460405    pRequest->rc = rc;
    461406
    462 #ifdef _DEBUG
    463     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",
    464409                       pRequest->enmType, pRequest->uCID, pRequest->rc,
    465410                       pRequest->cbData, pRequest->pvData);
     
    477422static int gstcntlProcessHandleRequest(RTPOLLSET hPollSet, uint32_t fPollEvt,
    478423                                       PRTPIPE phStdInW, PRTPIPE phStdOutR, PRTPIPE phStdErrR,
    479                                        PVBOXSERVICECTRLPROCESS pThread, PVBOXSERVICECTRLREQUEST pRequest)
     424                                       PVBOXSERVICECTRLPROCESS pProcess, PVBOXSERVICECTRLREQUEST pRequest)
    480425{
    481426    AssertPtrReturn(phStdInW, VERR_INVALID_POINTER);
    482427    AssertPtrReturn(phStdOutR, VERR_INVALID_POINTER);
    483428    AssertPtrReturn(phStdErrR, VERR_INVALID_POINTER);
    484     AssertPtrReturn(pThread, VERR_INVALID_POINTER);
     429    AssertPtrReturn(pProcess, VERR_INVALID_POINTER);
    485430    AssertPtrReturn(pRequest, VERR_INVALID_POINTER);
     431
     432    VBoxServiceVerbose(4, "[PID %RU32]: Handling pRequest=%p\n",
     433                       pProcess->uPID, pRequest);
    486434
    487435    /* Drain the notification pipe. */
    488436    uint8_t abBuf[8];
    489437    size_t cbIgnore;
    490     int rc = RTPipeRead(pThread->hNotificationPipeR, abBuf, sizeof(abBuf), &cbIgnore);
     438    int rc = RTPipeRead(pProcess->hNotificationPipeR, abBuf, sizeof(abBuf), &cbIgnore);
    491439    if (RT_FAILURE(rc))
    492440        VBoxServiceError("Draining IPC notification pipe failed with rc=%Rrc\n", rc);
     
    501449            /** @todo Check for some conditions to check to
    502450             *        veto quitting. */
    503             ASMAtomicXchgBool(&pThread->fShutdown, true);
     451            ASMAtomicXchgBool(&pProcess->fShutdown, true);
    504452            rcReq = VERR_CANCELLED;
    505453            break;
     
    572520
    573521        case VBOXSERVICECTRLREQUEST_PROC_TERM:
    574             ASMAtomicXchgBool(&pThread->fShutdown, true);
     522            ASMAtomicXchgBool(&pProcess->fShutdown, true);
    575523            fDefer = true;
    576524            break;
     
    589537        /* No access to pRequest here anymore -- could be out of scope
    590538         * or modified already! */
    591         pThread->pRequest = pRequest = NULL;
     539        pProcess->pRequest = pRequest = NULL;
    592540    }
    593541    else /* Completing the request defered. */
     
    603551 *
    604552 * @return  IPRT status code.
    605  * @param   pThread                     The process' thread handle.
     553 * @param   pProcess                    The guest process to handle.
    606554 * @param   hProcess                    The actual process handle.
    607555 * @param   cMsTimeout                  Time limit (in ms) of the process' life time.
     
    611559 * @param   hStdErrR                    Handle to the process' stderr read end.
    612560 */
    613 static int gstcntlProcessProcLoop(PVBOXSERVICECTRLPROCESS pThread,
    614                                   RTPROCESS hProcess, RTMSINTERVAL cMsTimeout, RTPOLLSET hPollSet,
     561static int gstcntlProcessProcLoop(PVBOXSERVICECTRLPROCESS pProcess,
     562                                  RTPROCESS hProcess, RTPOLLSET hPollSet,
    615563                                  PRTPIPE phStdInW, PRTPIPE phStdOutR, PRTPIPE phStdErrR)
    616564{
    617     AssertPtrReturn(pThread, VERR_INVALID_POINTER);
     565    AssertPtrReturn(pProcess, VERR_INVALID_POINTER);
    618566    AssertPtrReturn(phStdInW, VERR_INVALID_PARAMETER);
    619567    /* Rest is optional. */
     
    621569    int                         rc;
    622570    int                         rc2;
    623     uint64_t const              MsStart             = RTTimeMilliTS();
     571    uint64_t const              uMsStart            = RTTimeMilliTS();
    624572    RTPROCSTATUS                ProcessStatus       = { 254, RTPROCEXITREASON_ABEND };
    625573    bool                        fProcessAlive       = true;
     
    636584     * the first (stale) entry will be found and we get really weird results!
    637585     */
    638     rc = gstcntlProcessAssignPID(pThread, hProcess);
     586    rc = gstcntlProcessAssignPID(pProcess, hProcess);
    639587    if (RT_FAILURE(rc))
    640588    {
     
    648596     * and that it's now OK to send input to the process.
    649597     */
    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 };
    653602    rc = VbglR3GuestCtrlProcCbStatus(&ctx,
    654                                      pThread->uPID, PROC_STS_STARTED, 0 /* u32Flags */,
     603                                     pProcess->uPID, PROC_STS_STARTED, 0 /* u32Flags */,
    655604                                     NULL /* pvData */, 0 /* cbData */);
    656605
     
    660609    PVBOXSERVICECTRLREQUEST pReq = NULL;
    661610    while (   RT_SUCCESS(rc)
    662            && RT_UNLIKELY(!pThread->fShutdown))
     611           && RT_UNLIKELY(!pProcess->fShutdown))
    663612    {
    664613        /*
     
    668617        uint32_t fPollEvt;
    669618        rc2 = RTPollNoResume(hPollSet, cMsPollCur, &fPollEvt, &idPollHnd);
    670         if (pThread->fShutdown)
     619        if (pProcess->fShutdown)
    671620            continue;
    672621
     
    675624        if (RT_SUCCESS(rc2))
    676625        {
    677             /*VBoxServiceVerbose(4, "[PID %u}: RTPollNoResume idPollHnd=%u\n",
    678                                  pThread->uPID, idPollHnd);*/
    679626            switch (idPollHnd)
    680627            {
     
    693640
    694641                case VBOXSERVICECTRLPIPEID_IPC_NOTIFY:
    695                     pReq = pThread->pRequest; /** @todo Implement request queue. */
     642                    pReq = pProcess->pRequest; /** @todo Implement request queue. */
    696643                    rc = gstcntlProcessHandleRequest(hPollSet, fPollEvt,
    697644                                                     phStdInW, phStdOutR, phStdErrR,
    698                                                      pThread, pReq);
     645                                                     pProcess, pReq);
    699646                    if (rc != VINF_AIO_TASK_PENDING)
    700647                        pReq = NULL;
     
    708655            if (RT_FAILURE(rc) || rc == VINF_EOF)
    709656                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);
    720661#endif
     662
     663        if (RT_UNLIKELY(pProcess->fShutdown))
     664            break; /* We were asked to shutdown. */
     665
    721666        /*
    722667         * Check for process death.
     
    725670        {
    726671            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
    727678            if (RT_SUCCESS_NP(rc2))
    728679            {
     
    756707         */
    757708        uint32_t cMilliesLeft = RT_INDEFINITE_WAIT;
    758         if (cMsTimeout != RT_INDEFINITE_WAIT)
     709        if (pProcess->StartupInfo.uTimeLimitMS != RT_INDEFINITE_WAIT)
    759710        {
    760711            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)
    763714            {
    764                 VBoxServiceVerbose(3, "[PID %u]: Timed out (%ums elapsed > %ums timeout), killing ...\n",
    765                                    pThread->uPID, cMsElapsed, cMsTimeout);
     715                VBoxServiceVerbose(3, "[PID %RU32]: Timed out (%RU32ms elapsed > %RU32ms timeout), killing ...\n",
     716                                   pProcess->uPID, cMsElapsed, pProcess->StartupInfo.uTimeLimitMS);
    766717
    767718                fProcessTimedOut = true;
     
    778729            }
    779730            else
    780                 cMilliesLeft = cMsTimeout - (uint32_t)cMsElapsed;
     731                cMilliesLeft = pProcess->StartupInfo.uTimeLimitMS - (uint32_t)cMsElapsed;
    781732        }
    782733
     
    787738        if (cMilliesLeft < cMsPollCur)
    788739            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);
    798743    if (RT_SUCCESS(rc2))
    799744    {
    800         ASMAtomicXchgBool(&pThread->fShutdown, true);
    801 
    802         rc2 = RTCritSectLeave(&pThread->CritSect);
     745        ASMAtomicXchgBool(&pProcess->fShutdown, true);
     746
     747        rc2 = RTCritSectLeave(&pProcess->CritSect);
    803748        AssertRC(rc2);
    804749    }
     
    811756        if (MsProcessKilled == UINT64_MAX)
    812757        {
    813             VBoxServiceVerbose(3, "[PID %u]: Is still alive and not killed yet\n",
    814                                pThread->uPID);
     758            VBoxServiceVerbose(3, "[PID %RU32]: Is still alive and not killed yet\n",
     759                               pProcess->uPID);
    815760
    816761            MsProcessKilled = RTTimeMilliTS();
     
    821766        for (size_t i = 0; i < 10; i++)
    822767        {
    823             VBoxServiceVerbose(4, "[PID %u]: Kill attempt %d/10: Waiting to exit ...\n",
    824                                pThread->uPID, i + 1);
     768            VBoxServiceVerbose(4, "[PID %RU32]: Kill attempt %d/10: Waiting to exit ...\n",
     769                               pProcess->uPID, i + 1);
    825770            rc2 = RTProcWait(hProcess, RTPROCWAIT_FLAGS_NOBLOCK, &ProcessStatus);
    826771            if (RT_SUCCESS(rc2))
    827772            {
    828                 VBoxServiceVerbose(4, "[PID %u]: Kill attempt %d/10: Exited\n",
    829                                    pThread->uPID, i + 1);
     773                VBoxServiceVerbose(4, "[PID %RU32]: Kill attempt %d/10: Exited\n",
     774                                   pProcess->uPID, i + 1);
    830775                fProcessAlive = false;
    831776                break;
     
    833778            if (i >= 5)
    834779            {
    835                 VBoxServiceVerbose(4, "[PID %u]: Kill attempt %d/10: Trying to terminate ...\n",
    836                                    pThread->uPID, i + 1);
     780                VBoxServiceVerbose(4, "[PID %RU32]: Kill attempt %d/10: Trying to terminate ...\n",
     781                                   pProcess->uPID, i + 1);
    837782                RTProcTerminate(hProcess);
    838783            }
     
    841786
    842787        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);
    844789
    845790        if (   pReq /* Handle deferred termination request. */
     
    865810        if (     fProcessTimedOut  && !fProcessAlive && MsProcessKilled != UINT64_MAX)
    866811        {
    867             VBoxServiceVerbose(3, "[PID %u]: Timed out and got killed\n",
    868                                pThread->uPID);
     812            VBoxServiceVerbose(3, "[PID %RU32]: Timed out and got killed\n",
     813                               pProcess->uPID);
    869814            uStatus = PROC_STS_TOK;
    870815        }
    871816        else if (fProcessTimedOut  &&  fProcessAlive && MsProcessKilled != UINT64_MAX)
    872817        {
    873             VBoxServiceVerbose(3, "[PID %u]: Timed out and did *not* get killed\n",
    874                                pThread->uPID);
     818            VBoxServiceVerbose(3, "[PID %RU32]: Timed out and did *not* get killed\n",
     819                               pProcess->uPID);
    875820            uStatus = PROC_STS_TOA;
    876821        }
    877         else if (pThread->fShutdown && (fProcessAlive || MsProcessKilled != UINT64_MAX))
    878         {
    879             VBoxServiceVerbose(3, "[PID %u]: Got terminated because system/service is about to shutdown\n",
    880                                pThread->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);
    881826            uStatus = PROC_STS_DWN; /* Service is stopping, process was killed. */
    882             uFlags = pThread->uFlags; /* Return handed-in execution flags back to the host. */
     827            uFlags = pProcess->StartupInfo.uFlags; /* Return handed-in execution flags back to the host. */
    883828        }
    884829        else if (fProcessAlive)
    885830        {
    886             VBoxServiceError("[PID %u]: Is alive when it should not!\n",
    887                              pThread->uPID);
     831            VBoxServiceError("[PID %RU32]: Is alive when it should not!\n",
     832                             pProcess->uPID);
    888833        }
    889834        else if (MsProcessKilled != UINT64_MAX)
    890835        {
    891             VBoxServiceError("[PID %u]: Has been killed when it should not!\n",
    892                              pThread->uPID);
     836            VBoxServiceError("[PID %RU32]: Has been killed when it should not!\n",
     837                             pProcess->uPID);
    893838        }
    894839        else if (ProcessStatus.enmReason == RTPROCEXITREASON_NORMAL)
    895840        {
    896             VBoxServiceVerbose(3, "[PID %u]: Ended with RTPROCEXITREASON_NORMAL (Exit code: %u)\n",
    897                                pThread->uPID, ProcessStatus.iStatus);
     841            VBoxServiceVerbose(3, "[PID %RU32]: Ended with RTPROCEXITREASON_NORMAL (Exit code: %u)\n",
     842                               pProcess->uPID, ProcessStatus.iStatus);
    898843
    899844            uStatus = PROC_STS_TEN;
     
    902847        else if (ProcessStatus.enmReason == RTPROCEXITREASON_SIGNAL)
    903848        {
    904             VBoxServiceVerbose(3, "[PID %u]: Ended with RTPROCEXITREASON_SIGNAL (Signal: %u)\n",
    905                                pThread->uPID, ProcessStatus.iStatus);
     849            VBoxServiceVerbose(3, "[PID %RU32]: Ended with RTPROCEXITREASON_SIGNAL (Signal: %u)\n",
     850                               pProcess->uPID, ProcessStatus.iStatus);
    906851
    907852            uStatus = PROC_STS_TES;
     
    911856        {
    912857            /* ProcessStatus.iStatus will be undefined. */
    913             VBoxServiceVerbose(3, "[PID %u]: Ended with RTPROCEXITREASON_ABEND\n",
    914                                pThread->uPID);
     858            VBoxServiceVerbose(3, "[PID %RU32]: Ended with RTPROCEXITREASON_ABEND\n",
     859                               pProcess->uPID);
    915860
    916861            uStatus = PROC_STS_TEA;
     
    918863        }
    919864        else
    920             VBoxServiceVerbose(1, "[PID %u]: Handling process status %u not implemented\n",
    921                                pThread->uPID, ProcessStatus.enmReason);
    922 
    923         VBoxServiceVerbose(2, "[PID %u]: Ended, ClientID=%u, CID=%u, Status=%u, Flags=0x%x\n",
    924                            pThread->uPID, pThread->uClientID, pThread->uContextID, uStatus, uFlags);
    925 
    926         if (!(pThread->uFlags & EXECUTEPROCESSFLAG_WAIT_START))
    927         {
    928             VBGLR3GUESTCTRLCMDCTX ctx = { pThread->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 };
    929874            rc2 = VbglR3GuestCtrlProcCbStatus(&ctx,
    930                                               pThread->uPID, uStatus, uFlags,
     875                                              pProcess->uPID, uStatus, uFlags,
    931876                                              NULL /* pvData */, 0 /* cbData */);
    932877            if (RT_FAILURE(rc2))
    933                 VBoxServiceError("[PID %u]: Error reporting final status to host; rc=%Rrc\n",
    934                                  pThread->uPID, rc2);
     878                VBoxServiceError("[PID %RU32]: Error reporting final status to host; rc=%Rrc\n",
     879                                 pProcess->uPID, rc2);
    935880            if (RT_SUCCESS(rc))
    936881                rc = rc2;
    937882        }
    938883        else
    939             VBoxServiceVerbose(3, "[PID %u]: Was started detached, no final status sent to host\n",
    940                                pThread->uPID);
    941 
    942         VBoxServiceVerbose(3, "[PID %u]: Process loop ended with rc=%Rrc\n",
    943                            pThread->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);
    944889    }
    945890    else
    946         VBoxServiceError("[PID %u]: Loop failed with rc=%Rrc\n",
    947                          pThread->uPID, rc);
     891        VBoxServiceError("[PID %RU32]: Loop failed with rc=%Rrc\n",
     892                         pProcess->uPID, rc);
    948893    return rc;
    949894}
     
    1034979
    1035980/**
    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!
    1041983 *
    1042984 * @return  IPRT status code.
     
    13571299
    13581300    AssertPtr(pThread->pSession);
    1359     int rc = RTCritSectEnter(&pThread->pSession->csControlThreads);
     1301    int rc = RTCritSectEnter(&pThread->pSession->CritSect);
    13601302    if (RT_SUCCESS(rc))
    13611303    {
     
    13661308        do
    13671309        {
    1368             RTListForEach(&pThread->pSession->lstControlThreadsActive, pThreadCur, VBOXSERVICECTRLPROCESS, Node)
     1310            RTListForEach(&pThread->pSession->lstProcessesActive, pThreadCur, VBOXSERVICECTRLPROCESS, Node)
    13691311            {
    13701312                if (pThreadCur->uPID == uPID)
     
    13841326        pThread->uPID = uPID;
    13851327
    1386         rc = RTCritSectLeave(&pThread->pSession->csControlThreads);
     1328        rc = RTCritSectLeave(&pThread->pSession->CritSect);
    13871329        AssertRC(rc);
    13881330    }
     
    15621504    AssertPtrReturn(pProcess, VERR_INVALID_POINTER);
    15631505    VBoxServiceVerbose(3, "Thread of process pThread=0x%p = \"%s\" started\n",
    1564                        pProcess, pProcess->pszCmd);
     1506                       pProcess, pProcess->StartupInfo.szCmd);
    15651507
    15661508    int rc = GstCntlSessionListSet(pProcess->pSession,
     
    15761518    }
    15771519    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);
    15791521
    15801522    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    }
    15811571
    15821572    /*
     
    15881578    {
    15891579        size_t i;
    1590         for (i = 0; i < pProcess->uNumEnvVars && pProcess->papszEnv; i++)
    1591         {
    1592             rc = RTEnvPutEx(hEnv, pProcess->papszEnv[i]);
     1580        for (i = 0; i < uNumEnvVars && papszEnv; i++)
     1581        {
     1582            rc = RTEnvPutEx(hEnv, papszEnv[i]);
    15931583            if (RT_FAILURE(rc))
    15941584                break;
     
    16091599                PRTHANDLE   phStdOut;
    16101600                RTPIPE      pipeStdOutR;
    1611                 rc = gstcntlProcessSetupPipe(  (pProcess->uFlags & EXECUTEPROCESSFLAG_WAIT_STDOUT)
     1601                rc = gstcntlProcessSetupPipe(  (pProcess->StartupInfo.uFlags & EXECUTEPROCESSFLAG_WAIT_STDOUT)
    16121602                                             ? "|" : "/dev/null",
    16131603                                             1 /*STDOUT_FILENO*/,
     
    16181608                    PRTHANDLE   phStdErr;
    16191609                    RTPIPE      pipeStdErrR;
    1620                     rc = gstcntlProcessSetupPipe(  (pProcess->uFlags & EXECUTEPROCESSFLAG_WAIT_STDERR)
     1610                    rc = gstcntlProcessSetupPipe(  (pProcess->StartupInfo.uFlags & EXECUTEPROCESSFLAG_WAIT_STDERR)
    16211611                                                 ? "|" : "/dev/null",
    16221612                                                 2 /*STDERR_FILENO*/,
     
    16581648
    16591649                                RTPROCESS hProcess;
    1660                                 rc = gstcntlProcessCreateProcess(pProcess->pszCmd, pProcess->papszArgs, hEnv, pProcess->uFlags,
     1650                                rc = gstcntlProcessCreateProcess(pProcess->StartupInfo.szCmd, papszArgs, hEnv, pProcess->StartupInfo.uFlags,
    16611651                                                                 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,
    16641654                                                                 &hProcess);
    16651655                                if (RT_FAILURE(rc))
     
    16891679
    16901680                                    /* Enter the process loop. */
    1691                                     rc = gstcntlProcessProcLoop(pProcess,
    1692                                                                 hProcess, pProcess->uTimeLimitMS, hPollSet,
     1681                                    rc = gstcntlProcessProcLoop(pProcess, hProcess, hPollSet,
    16931682                                                                &pProcess->pipeStdInW, &pipeStdOutR, &pipeStdErrR);
    16941683
     
    17591748        /* Disconnect this client from the guest control service. This also cancels all
    17601749         * outstanding host requests. */
    1761         VBoxServiceVerbose(3, "[PID %u]: Disconnecting (client ID=%u) ...\n",
     1750        VBoxServiceVerbose(3, "[PID %RU32]: Disconnecting (client ID=%u) ...\n",
    17621751                           pProcess->uPID, pProcess->uClientID);
    17631752        VbglR3GuestCtrlDisconnect(pProcess->uClientID);
     
    17651754    }
    17661755
    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);
    17691768
    17701769    /* Update started/stopped status. */
     
    17791778        RTThreadUserSignal(RTThreadSelf());
    17801779
     1780    VBoxServiceVerbose(3, "[PID %RU32]: Thread returned with rc=%Rrc\n",
     1781                       pProcess->uPID, rc);
    17811782    return rc;
    17821783}
     
    18671868 * Performs a request to a specific (formerly started) guest process and waits
    18681869 * for its response.
    1869  * Note: Caller is responsible of locking!
     1870 * Note: Caller is responsible for locking!
    18701871 *
    18711872 * @return  IPRT status code.
     
    18891890    else
    18901891    {
     1892        VBoxServiceVerbose(3, "[PID %RU32]: Sending pRequest=%p\n",
     1893                           pProcess->uPID, pRequest);
     1894
    18911895        /* Set request structure pointer. */
    18921896        pProcess->pRequest = pRequest;
     
    18941898        /** @todo To speed up simultaneous guest process handling we could add a worker threads
    18951899         *        or queue in order to wait for the request to happen. Later. */
    1896         /* Wake up guest thrad by sending a wakeup byte to the notification pipe so
     1900        /* Wake up guest thread by sending a wakeup byte to the notification pipe so
    18971901         * that RTPoll unblocks (returns) and we then can do our requested operation. */
    18981902        Assert(pProcess->hNotificationPipeW != NIL_RTPIPE);
    1899         size_t cbWritten;
     1903        size_t cbWritten = 0;
    19001904        if (RT_SUCCESS(rc))
    19011905            rc = RTPipeWrite(pProcess->hNotificationPipeW, "i", 1, &cbWritten);
    19021906
    1903         if (   RT_SUCCESS(rc)
    1904             && cbWritten)
    1905         {
    1906             VBoxServiceVerbose(3, "[PID %u]: Waiting for response on enmType=%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);
    19081912
    19091913            rc = GstCntlProcessRequestWait(pRequest);
     
    19111915    }
    19121916
    1913     VBoxServiceVerbose(3, "[PID %u]: Performed enmType=%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);
    19151919    return rc;
    19161920}
  • trunk/src/VBox/Additions/common/VBoxService/VBoxServiceControlSession.cpp

    r45109 r45415  
    7070    VBOXSERVICESESSIONOPT_USERNAME,
    7171    VBOXSERVICESESSIONOPT_SESSION_ID,
    72     VBOXSERVICESESSIONOPT_SESSION_PROTO
     72    VBOXSERVICESESSIONOPT_SESSION_PROTO,
     73    VBOXSERVICESESSIONOPT_THREAD_ID
    7374};
    7475
     
    568569        if (RT_SUCCESS(rc))
    569570        {
    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",
    571572                               startupInfo.szCmd, startupInfo.uFlags,
    572573                               startupInfo.uNumArgs ? startupInfo.szArgs : "<None>",
    573574                               startupInfo.uNumEnvVars ? startupInfo.szEnv : "<None>",
    574                                startupInfo.szUser,
    575 #ifdef DEBUG
    576                                startupInfo.szPassword,
    577 #else
    578                                "XXX", /* Never show passwords in release mode. */
    579 #endif
    580575                               startupInfo.uTimeLimitMS);
    581576
    582             rc = GstCntlSessionReapProcesses(pSession);
     577            /*rc = GstCntlSessionReapProcesses(pSession);
    583578            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);*/
    585580            /* Keep going. */
    586581
     
    741736
    742737    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
    743742    if (RT_SUCCESS(rc))
    744743    {
     
    750749                                         pHostCtx->uContextID, uHandleID, RT_INDEFINITE_WAIT /* Timeout */,
    751750                                         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",
    753752                               uPID, rc, pHostCtx->uContextID, cbRead, uHandleID, uFlags);
    754753
     
    973972
    974973        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",
    976976                               uMsg, pHostCtx->uNumParms);
    977             /* Don't terminate here; just wait for the next message. */
    978977            break;
    979978    }
     
    10081007    }
    10091008    else
     1009    {
    10101010        VBoxServiceError("Error connecting to guest control service, rc=%Rrc\n", rc);
     1011        return rc;
     1012    }
    10111013
    10121014    /* Let caller know that we're done initializing. */
    1013     int rc2 = RTThreadUserSignal(RTThreadSelf());
    1014     if (RT_SUCCESS(rc))
    1015         rc = rc2;
     1015    rc = RTThreadUserSignal(RTThreadSelf());
     1016    if (RT_FAILURE(rc))
     1017        return rc;
    10161018
    10171019    bool fProcessAlive = true;
     
    10191021    RT_ZERO(ProcessStatus);
    10201022
     1023    int rcWait;
    10211024    if (RT_SUCCESS(rc))
    10221025    {
     
    10241027        uint64_t u64TimeoutStart = 0;
    10251028
    1026         int rcWait;
    10271029        for (;;)
    10281030        {
     
    10481050                                       uSessionID);
    10491051                    u64TimeoutStart = RTTimeMilliTS();
     1052                    continue; /* Don't waste time on waiting. */
    10501053                }
    10511054                if (RTTimeMilliTS() - u64TimeoutStart > uTimeoutsMS)
     
    10841087        }
    10851088
    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",
    10871090                           uSessionID, rc);
    10881091
     
    10921095    else
    10931096    {
    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",
    11171129                       uSessionID, uSessionStatus, uSessionRc);
    11181130
     
    11201132    Assert(uSessionStatus != GUEST_SESSION_NOTIFYTYPE_UNDEFINED);
    11211133    VBGLR3GUESTCTRLCMDCTX ctx = { uClientID, VBOX_GUESTCTRL_CONTEXTID_MAKE_SESSION(uSessionID) };
    1122     rc2 = VbglR3GuestCtrlSessionNotify(&ctx,
    1123                                        uSessionStatus, uSessionRc);
     1134    int rc2 = VbglR3GuestCtrlSessionNotify(&ctx,
     1135                                           uSessionStatus, uSessionRc);
    11241136    if (RT_FAILURE(rc2))
    11251137        VBoxServiceError("Reporting session ID=%RU32 final status failed with rc=%Rrc\n",
     
    12541266    AssertPtrReturn(pSession, NULL);
    12551267
    1256     PVBOXSERVICECTRLPROCESS pThread = NULL;
    1257     int rc = RTCritSectEnter(&pSession->csControlThreads);
    1258     if (RT_SUCCESS(rc))
    1259     {
    1260         PVBOXSERVICECTRLPROCESS pThreadCur;
    1261         RTListForEach(&pSession->lstControlThreadsActive, pThreadCur, VBOXSERVICECTRLPROCESS, Node)
    1262         {
    1263             if (pThreadCur->uPID == uPID)
    1264             {
    1265                 rc = RTCritSectEnter(&pThreadCur->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);
    12661278                if (RT_SUCCESS(rc))
    1267                     pThread = pThreadCur;
     1279                    pProcess = pCurProcess;
    12681280                break;
    12691281            }
    12701282        }
    12711283
    1272         int rc2 = RTCritSectLeave(&pSession->csControlThreads);
     1284        int rc2 = RTCritSectLeave(&pSession->CritSect);
    12731285        if (RT_SUCCESS(rc))
    12741286            rc = rc2;
    12751287    }
    12761288
    1277     return pThread;
     1289    return pProcess;
    12781290}
    12791291
     
    12861298                       pSession->StartupInfo.uSessionID);
    12871299
    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    }
    13541389
    13551390    return rc;
     
    13641399
    13651400    /* Destroy critical section. */
    1366     RTCritSectDelete(&pSession->csControlThreads);
     1401    RTCritSectDelete(&pSession->CritSect);
    13671402
    13681403    return rc;
     
    14411476    AssertPtrReturn(pSession, VERR_INVALID_POINTER);
    14421477
    1443     RTListInit(&pSession->lstControlThreadsActive);
    1444     RTListInit(&pSession->lstControlThreadsInactive);
     1478    RTListInit(&pSession->lstProcessesActive);
     1479    RTListInit(&pSession->lstProcessesInactive);
    14451480    RTListInit(&pSession->lstFiles);
    14461481
     
    14571492
    14581493    /* Init critical section for protecting the thread lists. */
    1459     int rc = RTCritSectInit(&pSession->csControlThreads);
     1494    int rc = RTCritSectInit(&pSession->CritSect);
    14601495    AssertRC(rc);
    14611496
     
    14811516    AssertPtrReturn(pProcess, VERR_INVALID_POINTER);
    14821517
    1483     int rc = RTCritSectEnter(&pSession->csControlThreads);
     1518    int rc = RTCritSectEnter(&pSession->CritSect);
    14841519    if (RT_SUCCESS(rc))
    14851520    {
     
    14911526        {
    14921527            case VBOXSERVICECTRLTHREADLIST_STOPPED:
    1493                 pAnchor = &pSession->lstControlThreadsInactive;
     1528                pAnchor = &pSession->lstProcessesInactive;
    14941529                break;
    14951530
    14961531            case VBOXSERVICECTRLTHREADLIST_RUNNING:
    1497                 pAnchor = &pSession->lstControlThreadsActive;
     1532                pAnchor = &pSession->lstProcessesActive;
    14981533                break;
    14991534
    15001535            default:
    1501                 AssertMsgFailed(("Unknown list type: %u", enmList));
     1536                AssertMsgFailed(("Unknown list type: %u\n",
     1537                                 enmList));
    15021538                break;
    15031539        }
     
    15201556        }
    15211557
    1522         int rc2 = RTCritSectLeave(&pSession->csControlThreads);
     1558        int rc2 = RTCritSectLeave(&pSession->CritSect);
    15231559        if (RT_SUCCESS(rc))
    15241560            rc = rc2;
     
    15461582    AssertPtrReturn(pbAllowed, VERR_INVALID_POINTER);
    15471583
    1548     int rc = RTCritSectEnter(&pSession->csControlThreads);
     1584    int rc = RTCritSectEnter(&pSession->CritSect);
    15491585    if (RT_SUCCESS(rc))
    15501586    {
     
    15571593        {
    15581594            uint32_t uProcsRunning = 0;
    1559             PVBOXSERVICECTRLPROCESS pThread;
    1560             RTListForEach(&pSession->lstControlThreadsActive, pThread, VBOXSERVICECTRLPROCESS, Node)
     1595            PVBOXSERVICECTRLPROCESS pProcess;
     1596            RTListForEach(&pSession->lstProcessesActive, pProcess, VBOXSERVICECTRLPROCESS, Node)
    15611597                uProcsRunning++;
    15621598
     
    15751611        *pbAllowed = !fLimitReached;
    15761612
    1577         int rc2 = RTCritSectLeave(&pSession->csControlThreads);
     1613        int rc2 = RTCritSectLeave(&pSession->CritSect);
    15781614        if (RT_SUCCESS(rc))
    15791615            rc = rc2;
     
    15831619}
    15841620
    1585 
     1621#if 0
    15861622/**
    15871623 * Reaps all inactive guest process threads.
     1624 * Does not do locking; this is the job of the caller.
    15881625 *
    15891626 * @return  IPRT status code.
     
    15931630    AssertPtrReturn(pSession, VERR_INVALID_POINTER);
    15941631
    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;
    16311660    }
    16321661
     
    16341663    return rc;
    16351664}
     1665#endif
    16361666
    16371667
     
    17001730 *                                  Optional.
    17011731 */
    1702 int GstCntlSessionThreadOpen(PRTLISTANCHOR pList,
    1703                              const PVBOXSERVICECTRLSESSIONSTARTUPINFO pSessionStartupInfo,
    1704                              PVBOXSERVICECTRLSESSIONTHREAD *ppSessionThread)
     1732int GstCntlSessionThreadCreate(PRTLISTANCHOR pList,
     1733                               const PVBOXSERVICECTRLSESSIONSTARTUPINFO pSessionStartupInfo,
     1734                               PVBOXSERVICECTRLSESSIONTHREAD *ppSessionThread)
    17051735{
    17061736    AssertPtrReturn(pList, VERR_INVALID_POINTER);
     
    17161746        if (pSessionCur->StartupInfo.uSessionID == pSessionStartupInfo->uSessionID)
    17171747        {
    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",
    17191749                             pSessionCur->StartupInfo.uSessionID, pSessionCur));
    17201750            return VERR_ALREADY_EXISTS;
     
    17241754    int rc = VINF_SUCCESS;
    17251755
    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)
    17281764    {
    17291765        /* 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;
    17351772
    17361773        /* Is this an anonymous session? */
    17371774        /* 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));
    17391776        if (fAnonymous)
    17401777        {
    1741             Assert(!strlen(pSession->StartupInfo.szPassword));
    1742             Assert(!strlen(pSession->StartupInfo.szDomain));
     1778            Assert(!strlen(pSessionThread->StartupInfo.szPassword));
     1779            Assert(!strlen(pSessionThread->StartupInfo.szDomain));
    17431780
    17441781            VBoxServiceVerbose(3, "New anonymous guest session ID=%RU32 created, uFlags=%x, using protocol %RU32\n",
     
    17621799        }
    17631800
    1764         rc = RTCritSectInit(&pSession->CritSect);
     1801        rc = RTCritSectInit(&pSessionThread->CritSect);
    17651802        AssertRC(rc);
    17661803
     
    17731810            if (!fAnonymous)
    17741811            {
    1775                 if (!RTStrPrintf(szParmUserName, sizeof(szParmUserName), "--username=%s", pSession->StartupInfo.szUser))
     1812                if (!RTStrPrintf(szParmUserName, sizeof(szParmUserName), "--user=%s", pSessionThread->StartupInfo.szUser))
    17761813                    rc = VERR_BUFFER_OVERFLOW;
    17771814            }
    17781815            char szParmSessionID[32];
    17791816            if (RT_SUCCESS(rc) && !RTStrPrintf(szParmSessionID, sizeof(szParmSessionID), "--session-id=%RU32",
    1780                                                pSession->StartupInfo.uSessionID))
     1817                                               pSessionThread->StartupInfo.uSessionID))
    17811818            {
    17821819                rc = VERR_BUFFER_OVERFLOW;
     
    17841821            char szParmSessionProto[32];
    17851822            if (RT_SUCCESS(rc) && !RTStrPrintf(szParmSessionProto, sizeof(szParmSessionProto), "--session-proto=%RU32",
    1786                                                pSession->StartupInfo.uProtocol))
     1823                                               pSessionThread->StartupInfo.uProtocol))
    17871824            {
    17881825                rc = VERR_BUFFER_OVERFLOW;
    17891826            }
    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 */
    17911835            if (RT_SUCCESS(rc))
    17921836            {
    17931837                int iOptIdx = 0; /* Current index in argument vector. */
    17941838
    1795                 char const *papszArgs[8];
     1839                char const *papszArgs[16];
    17961840                papszArgs[iOptIdx++] = pszExeName;
    17971841                papszArgs[iOptIdx++] = "guestsession";
    17981842                papszArgs[iOptIdx++] = szParmSessionID;
    17991843                papszArgs[iOptIdx++] = szParmSessionProto;
     1844#ifdef DEBUG
     1845                papszArgs[iOptIdx++] = szParmThreadId;
     1846#endif /* DEBUG */
    18001847                if (!fAnonymous)
    18011848                    papszArgs[iOptIdx++] = szParmUserName;
     
    18291876                        RTPathStripExt(pszLogFile);
    18301877                        char *pszLogSuffix;
     1878#ifndef DEBUG
    18311879                        if (RTStrAPrintf(&pszLogSuffix, "-%RU32-%s",
    18321880                                         pSessionStartupInfo->uSessionID,
     
    18351883                            rc2 = VERR_NO_MEMORY;
    18361884                        }
     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 */
    18371894                        else
    18381895                        {
     
    19401997                        hStdOutAndErr.enmType = RTHANDLETYPE_FILE;
    19411998
    1942                         /** @todo Do we need a custom/cloned environment block? */
     1999                        /** @todo Set custom/cloned guest session environment block. */
    19432000                        rc = RTProcCreateEx(pszExeName, papszArgs, RTENV_DEFAULT, uProcFlags,
    19442001                                            &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);
    19482005
    19492006                        RTFileClose(hStdOutAndErr.u.hFile);
    19502007                    }
    19512008
    1952                     RTFileClose(hStdOutAndErr.u.hFile);
     2009                    RTFileClose(hStdIn.u.hFile);
    19532010                }
    19542011#endif
     
    19612018        {
    19622019            /* 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*/,
    19682022                                 RTTHREADTYPE_DEFAULT, RTTHREADFLAGS_WAITABLE, "sess%u", s_uCtrlSessionThread);
    19692023            if (RT_FAILURE(rc))
     
    19742028            {
    19752029                /* 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)
    19792032                    || RT_FAILURE(rc))
    19802033                {
    19812034                    VBoxServiceError("Thread for session ID=%RU32 failed to start, rc=%Rrc\n",
    1982                                      pSession->StartupInfo.uSessionID, rc);
     2035                                     pSessionThread->StartupInfo.uSessionID, rc);
    19832036                    if (RT_SUCCESS(rc))
    19842037                        rc = VERR_CANT_CREATE; /** @todo Find a better rc. */
     
    19862039                else
    19872040                {
    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);
    19892045
    19902046                    /* Add session to list. */
    1991                     /* rc = */ RTListAppend(pList, &pSession->Node);
     2047                    /* rc = */ RTListAppend(pList, &pSessionThread->Node);
    19922048                    if (ppSessionThread) /* Return session if wanted. */
    1993                         *ppSessionThread = pSession;
     2049                        *ppSessionThread = pSessionThread;
    19942050                }
    19952051            }
     
    19982054        if (RT_FAILURE(rc))
    19992055        {
    2000             RTMemFree(pSession);
     2056            RTMemFree(pSessionThread);
    20012057        }
    20022058    }
     
    20042060        rc = VERR_NO_MEMORY;
    20052061
    2006     VBoxServiceVerbose(3, "Forking returned returned rc=%Rrc\n", rc);
     2062    VBoxServiceVerbose(3, "Forking session thread returned returned rc=%Rrc\n", rc);
    20072063    return rc;
    20082064}
     
    20102066
    20112067/**
    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.
    20142069 *
    20152070 * @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).
    20172073 * @param   uFlags                  Closing flags.
    20182074 */
    2019 int GstCntlSessionThreadClose(PVBOXSERVICECTRLSESSIONTHREAD pThread, uint32_t uFlags)
     2075int GstCntlSessionThreadWait(PVBOXSERVICECTRLSESSIONTHREAD pThread,
     2076                             uint32_t uTimeoutMS, uint32_t uFlags)
    20202077{
    20212078    AssertPtrReturn(pThread, VERR_INVALID_POINTER);
     
    20242081    if (pThread->Thread == NIL_RTTHREAD)
    20252082    {
    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",
    20272084                         pThread));
    20282085        return VERR_NOT_FOUND;
     
    20312088    int rc = VINF_SUCCESS;
    20322089
    2033     /* The fork should have received the same closing request,
     2090    /*
     2091     * The fork should have received the same closing request,
    20342092     * 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     */
    20362095    if (ASMAtomicReadBool(&pThread->fStarted))
    20372096    {
     
    20392098        ASMAtomicXchgBool(&pThread->fShutdown, true);
    20402099
    2041         uint32_t cMsTimeout = 30 * 1000; /** @todo 30s default. Make this configurable. Later. */
    2042 
    20432100        VBoxServiceVerbose(3, "Waiting for session thread ID=%RU32 to close (%RU32ms) ...\n",
    2044                            pThread->StartupInfo.uSessionID, cMsTimeout);
     2101                           pThread->StartupInfo.uSessionID, uTimeoutMS);
    20452102
    20462103        int rcThread;
    2047         rc = RTThreadWait(pThread->Thread, cMsTimeout, &rcThread);
     2104        rc = RTThreadWait(pThread->Thread, uTimeoutMS, &rcThread);
    20482105        if (RT_FAILURE(rc))
    20492106        {
     
    20542111            VBoxServiceVerbose(3, "Session thread ID=%RU32 ended with rc=%Rrc\n",
    20552112                               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 */
     2126int 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}
    20712140
    20722141/**
     
    20772146 * @param   uFlags                  Closing flags.
    20782147 */
    2079 int GstCntlSessionThreadCloseAll(PRTLISTANCHOR pList, uint32_t uFlags)
     2148int GstCntlSessionThreadDestroyAll(PRTLISTANCHOR pList, uint32_t uFlags)
    20802149{
    20812150    AssertPtrReturn(pList, VERR_INVALID_POINTER);
     
    20832152    int rc = VINF_SUCCESS;
    20842153
    2085     PVBOXSERVICECTRLSESSIONTHREAD pSessionCur
     2154    PVBOXSERVICECTRLSESSIONTHREAD pSessionThread
    20862155         = 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;
    20972168            /* Keep going. */
    20982169        }
     
    21012172            break;
    21022173
    2103         pSessionCur = pSessionNext;
    2104     }
    2105 
    2106     return rc;
    2107 }
    2108 
     2174        pSessionThread = pSessionThreadNext;
     2175    }
     2176
     2177    return rc;
     2178}
    21092179
    21102180RTEXITCODE VBoxServiceControlSessionForkInit(int argc, char **argv)
     
    21132183    {
    21142184        { "--logfile",         VBOXSERVICESESSIONOPT_LOG_FILE,        RTGETOPT_REQ_STRING },
    2115         { "--username",        VBOXSERVICESESSIONOPT_USERNAME,        RTGETOPT_REQ_STRING },
     2185        { "--user",            VBOXSERVICESESSIONOPT_USERNAME,        RTGETOPT_REQ_STRING },
    21162186        { "--session-id",      VBOXSERVICESESSIONOPT_SESSION_ID,      RTGETOPT_REQ_UINT32 },
    21172187        { "--session-proto",   VBOXSERVICESESSIONOPT_SESSION_PROTO,   RTGETOPT_REQ_UINT32 },
     2188#ifdef DEBUG
     2189        { "--thread-id",       VBOXSERVICESESSIONOPT_THREAD_ID,       RTGETOPT_REQ_UINT32 },
     2190#endif /* DEBUG */
    21182191        { "--verbose",         'v',                                   RTGETOPT_REQ_NOTHING }
    21192192    };
     
    21572230                break;
    21582231
     2232            case VBOXSERVICESESSIONOPT_THREAD_ID:
     2233                /* Not handled. */
     2234                break;
     2235
    21592236            /** @todo Implement help? */
    21602237
Note: See TracChangeset for help on using the changeset viewer.

© 2025 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette