/* $Id: GuestCtrlImpl.cpp 42758 2012-08-10 15:19:09Z vboxsync $ */ /** @file * VirtualBox COM class implementation: Guest */ /* * Copyright (C) 2006-2012 Oracle Corporation * * This file is part of VirtualBox Open Source Edition (OSE), as * available from http://www.virtualbox.org. This file is free software; * you can redistribute it and/or modify it under the terms of the GNU * General Public License (GPL) as published by the Free Software * Foundation, in version 2 as it comes in the "COPYING" file of the * VirtualBox OSE distribution. VirtualBox OSE is distributed in the * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind. */ #include "GuestImpl.h" #include "GuestSessionImpl.h" #include "GuestCtrlImplPrivate.h" #include "Global.h" #include "ConsoleImpl.h" #include "ProgressImpl.h" #include "VMMDev.h" #include "AutoCaller.h" #include #ifdef VBOX_WITH_GUEST_CONTROL # include # include #endif #include #include #include #include #include #include #include #include // public methods only for internal purposes ///////////////////////////////////////////////////////////////////////////// #ifdef VBOX_WITH_GUEST_CONTROL /** * Appends environment variables to the environment block. * * Each var=value pair is separated by the null character ('\\0'). The whole * block will be stored in one blob and disassembled on the guest side later to * fit into the HGCM param structure. * * @returns VBox status code. * * @param pszEnvVar The environment variable=value to append to the * environment block. * @param ppvList This is actually a pointer to a char pointer * variable which keeps track of the environment block * that we're constructing. * @param pcbList Pointer to the variable holding the current size of * the environment block. (List is a misnomer, go * ahead a be confused.) * @param pcEnvVars Pointer to the variable holding count of variables * stored in the environment block. */ int Guest::prepareExecuteEnv(const char *pszEnv, void **ppvList, uint32_t *pcbList, uint32_t *pcEnvVars) { int rc = VINF_SUCCESS; uint32_t cchEnv = strlen(pszEnv); Assert(cchEnv >= 2); if (*ppvList) { uint32_t cbNewLen = *pcbList + cchEnv + 1; /* Include zero termination. */ char *pvTmp = (char *)RTMemRealloc(*ppvList, cbNewLen); if (pvTmp == NULL) rc = VERR_NO_MEMORY; else { memcpy(pvTmp + *pcbList, pszEnv, cchEnv); pvTmp[cbNewLen - 1] = '\0'; /* Add zero termination. */ *ppvList = (void **)pvTmp; } } else { char *pszTmp; if (RTStrAPrintf(&pszTmp, "%s", pszEnv) >= 0) { *ppvList = (void **)pszTmp; /* Reset counters. */ *pcEnvVars = 0; *pcbList = 0; } } if (RT_SUCCESS(rc)) { *pcbList += cchEnv + 1; /* Include zero termination. */ *pcEnvVars += 1; /* Increase env variable count. */ } return rc; } /** * Adds a callback with a user provided data block and an optional progress object * to the callback map. A callback is identified by a unique context ID which is used * to identify a callback from the guest side. * * @return IPRT status code. * @param pCallback * @param puContextID */ int Guest::callbackAdd(const PVBOXGUESTCTRL_CALLBACK pCallback, uint32_t *puContextID) { AssertPtrReturn(pCallback, VERR_INVALID_PARAMETER); /* puContextID is optional. */ int rc = VERR_NOT_FOUND; AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS); /* Create a new context ID and assign it. */ uint32_t uNewContextID = 0; uint32_t uTries = 0; for (;;) { /* Create a new context ID ... */ uNewContextID = ASMAtomicIncU32(&mNextContextID); if (uNewContextID == UINT32_MAX) ASMAtomicUoWriteU32(&mNextContextID, 1000); /* Is the context ID already used? Try next ID ... */ if (!callbackExists(uNewContextID)) { /* Callback with context ID was not found. This means * we can use this context ID for our new callback we want * to add below. */ rc = VINF_SUCCESS; break; } if (++uTries == UINT32_MAX) break; /* Don't try too hard. */ } if (RT_SUCCESS(rc)) { /* Add callback with new context ID to our callback map. */ mCallbackMap[uNewContextID] = *pCallback; Assert(mCallbackMap.size()); /* Report back new context ID. */ if (puContextID) *puContextID = uNewContextID; } return rc; } /** * Assigns a host PID to a specified context. * Does not do locking! * * @param uContextID * @param uHostPID */ int Guest::callbackAssignHostPID(uint32_t uContextID, uint32_t uHostPID) { AssertReturn(uContextID, VERR_INVALID_PARAMETER); AssertReturn(uHostPID, VERR_INVALID_PARAMETER); int rc = VINF_SUCCESS; CallbackMapIter it = mCallbackMap.find(uContextID); if (it == mCallbackMap.end()) return VERR_NOT_FOUND; it->second.uHostPID = uHostPID; return VINF_SUCCESS; } /** * Destroys the formerly allocated callback data. The callback then * needs to get removed from the callback map via callbackRemove(). * Does not do locking! * * @param uContextID */ void Guest::callbackDestroy(uint32_t uContextID) { AssertReturnVoid(uContextID); CallbackMapIter it = mCallbackMap.find(uContextID); if (it != mCallbackMap.end()) { LogFlowFunc(("Callback with CID=%u found\n", uContextID)); if (it->second.pvData) { LogFlowFunc(("Destroying callback with CID=%u ...\n", uContextID)); callbackFreeUserData(it->second.pvData); it->second.cbData = 0; } } } /** * Removes a callback from the callback map. * Does not do locking! * * @param uContextID */ void Guest::callbackRemove(uint32_t uContextID) { callbackDestroy(uContextID); mCallbackMap.erase(uContextID); } /** * Checks whether a callback with the given context ID * exists or not. * Does not do locking! * * @return bool True, if callback exists, false if not. * @param uContextID Context ID to check. */ bool Guest::callbackExists(uint32_t uContextID) { AssertReturn(uContextID, false); CallbackMapIter it = mCallbackMap.find(uContextID); return (it == mCallbackMap.end()) ? false : true; } /** * Frees the user data (actual context data) of a callback. * Does not do locking! * * @param pvData Data to free. */ void Guest::callbackFreeUserData(void *pvData) { if (pvData) { RTMemFree(pvData); pvData = NULL; } } /** * Retrieves the (generated) host PID of a given callback. * * @return The host PID, if found, 0 otherwise. * @param uContextID Context ID to lookup host PID for. * @param puHostPID Where to store the host PID. */ uint32_t Guest::callbackGetHostPID(uint32_t uContextID) { AssertReturn(uContextID, 0); AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS); CallbackMapIterConst it = mCallbackMap.find(uContextID); if (it == mCallbackMap.end()) return 0; return it->second.uHostPID; } int Guest::callbackGetUserData(uint32_t uContextID, eVBoxGuestCtrlCallbackType *pEnmType, void **ppvData, size_t *pcbData) { AssertReturn(uContextID, VERR_INVALID_PARAMETER); /* pEnmType is optional. */ /* ppvData is optional. */ /* pcbData is optional. */ AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS); CallbackMapIterConst it = mCallbackMap.find(uContextID); if (it == mCallbackMap.end()) return VERR_NOT_FOUND; if (pEnmType) *pEnmType = it->second.mType; if ( ppvData && it->second.cbData) { void *pvData = RTMemAlloc(it->second.cbData); AssertPtrReturn(pvData, VERR_NO_MEMORY); memcpy(pvData, it->second.pvData, it->second.cbData); *ppvData = pvData; } if (pcbData) *pcbData = it->second.cbData; return VINF_SUCCESS; } /* Does not do locking! Caller has to take care of it because the caller needs to * modify the data ...*/ void* Guest::callbackGetUserDataMutableRaw(uint32_t uContextID, size_t *pcbData) { /* uContextID can be 0. */ /* pcbData is optional. */ CallbackMapIterConst it = mCallbackMap.find(uContextID); if (it != mCallbackMap.end()) { if (pcbData) *pcbData = it->second.cbData; return it->second.pvData; } return NULL; } int Guest::callbackInit(PVBOXGUESTCTRL_CALLBACK pCallback, eVBoxGuestCtrlCallbackType enmType, ComPtr pProgress) { AssertPtrReturn(pCallback, VERR_INVALID_POINTER); /* Everything else is optional. */ int vrc = VINF_SUCCESS; switch (enmType) { case VBOXGUESTCTRLCALLBACKTYPE_EXEC_START: { PCALLBACKDATAEXECSTATUS pData = (PCALLBACKDATAEXECSTATUS)RTMemAlloc(sizeof(CALLBACKDATAEXECSTATUS)); AssertPtrReturn(pData, VERR_NO_MEMORY); RT_BZERO(pData, sizeof(CALLBACKDATAEXECSTATUS)); pCallback->cbData = sizeof(CALLBACKDATAEXECSTATUS); pCallback->pvData = pData; break; } case VBOXGUESTCTRLCALLBACKTYPE_EXEC_OUTPUT: { PCALLBACKDATAEXECOUT pData = (PCALLBACKDATAEXECOUT)RTMemAlloc(sizeof(CALLBACKDATAEXECOUT)); AssertPtrReturn(pData, VERR_NO_MEMORY); RT_BZERO(pData, sizeof(CALLBACKDATAEXECOUT)); pCallback->cbData = sizeof(CALLBACKDATAEXECOUT); pCallback->pvData = pData; break; } case VBOXGUESTCTRLCALLBACKTYPE_EXEC_INPUT_STATUS: { PCALLBACKDATAEXECINSTATUS pData = (PCALLBACKDATAEXECINSTATUS)RTMemAlloc(sizeof(CALLBACKDATAEXECINSTATUS)); AssertPtrReturn(pData, VERR_NO_MEMORY); RT_BZERO(pData, sizeof(CALLBACKDATAEXECINSTATUS)); pCallback->cbData = sizeof(CALLBACKDATAEXECINSTATUS); pCallback->pvData = pData; break; } default: vrc = VERR_INVALID_PARAMETER; break; } if (RT_SUCCESS(vrc)) { /* Init/set common stuff. */ pCallback->mType = enmType; pCallback->pProgress = pProgress; } return vrc; } bool Guest::callbackIsCanceled(uint32_t uContextID) { if (!uContextID) return true; /* If no context ID given then take a shortcut. */ ComPtr pProgress; { AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS); CallbackMapIterConst it = mCallbackMap.find(uContextID); if (it != mCallbackMap.end()) pProgress = it->second.pProgress; } if (pProgress) { BOOL fCanceled = FALSE; HRESULT hRC = pProgress->COMGETTER(Canceled)(&fCanceled); if ( SUCCEEDED(hRC) && !fCanceled) { return false; } } return true; /* No progress / error means canceled. */ } bool Guest::callbackIsComplete(uint32_t uContextID) { AssertReturn(uContextID, true); ComPtr pProgress; { AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS); CallbackMapIterConst it = mCallbackMap.find(uContextID); if (it != mCallbackMap.end()) pProgress = it->second.pProgress; } if (pProgress) { BOOL fCompleted = FALSE; HRESULT hRC = pProgress->COMGETTER(Completed)(&fCompleted); if ( SUCCEEDED(hRC) && fCompleted) { return true; } } return false; } int Guest::callbackMoveForward(uint32_t uContextID, const char *pszMessage) { AssertReturn(uContextID, VERR_INVALID_PARAMETER); AssertPtrReturn(pszMessage, VERR_INVALID_PARAMETER); ComPtr pProgress; { AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS); CallbackMapIterConst it = mCallbackMap.find(uContextID); if (it != mCallbackMap.end()) pProgress = it->second.pProgress; } if (pProgress) { HRESULT hr = pProgress->SetNextOperation(Bstr(pszMessage).raw(), 1 /* Weight */); if (FAILED(hr)) return VERR_CANCELLED; return VINF_SUCCESS; } return VERR_NOT_FOUND; } /** * Notifies a specified callback about its final status. * * @return IPRT status code. * @param uContextID * @param iRC * @param pszMessage */ int Guest::callbackNotifyEx(uint32_t uContextID, int iRC, const char *pszMessage) { AssertReturn(uContextID, VERR_INVALID_PARAMETER); if (RT_FAILURE(iRC)) AssertReturn(pszMessage, VERR_INVALID_PARAMETER); LogFlowFunc(("Checking whether callback (CID=%u) needs notification iRC=%Rrc, pszMsg=%s\n", uContextID, iRC, pszMessage ? pszMessage : "")); ComObjPtr pProgress; { AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS); CallbackMapIterConst it = mCallbackMap.find(uContextID); if (it != mCallbackMap.end()) pProgress = it->second.pProgress; } #if 0 BOOL fCanceled = FALSE; HRESULT hRC = pProgress->COMGETTER(Canceled)(&fCanceled); if ( SUCCEEDED(hRC) && fCanceled) { /* If progress already canceled do nothing here. */ return VINF_SUCCESS; } #endif if (pProgress) { /* * Assume we didn't complete to make sure we clean up even if the * following call fails. */ BOOL fCompleted = FALSE; HRESULT hRC = pProgress->COMGETTER(Completed)(&fCompleted); if ( SUCCEEDED(hRC) && !fCompleted) { LogFlowFunc(("Notifying callback with CID=%u, iRC=%Rrc, pszMsg=%s\n", uContextID, iRC, pszMessage ? pszMessage : "")); /* * To get waitForCompletion completed (unblocked) we have to notify it if necessary (only * cancel won't work!). This could happen if the client thread (e.g. VBoxService, thread of a spawned process) * is disconnecting without having the chance to sending a status message before, so we * have to abort here to make sure the host never hangs/gets stuck while waiting for the * progress object to become signalled. */ if (RT_SUCCESS(iRC)) { hRC = pProgress->notifyComplete(S_OK); } else { hRC = pProgress->notifyComplete(VBOX_E_IPRT_ERROR /* Must not be S_OK. */, COM_IIDOF(IGuest), Guest::getStaticComponentName(), pszMessage); } LogFlowFunc(("Notified callback with CID=%u returned %Rhrc (0x%x)\n", uContextID, hRC, hRC)); } else LogFlowFunc(("Callback with CID=%u already notified\n", uContextID)); /* * Do *not* NULL pProgress here, because waiting function like executeProcess() * will still rely on this object for checking whether they have to give up! */ } /* If pProgress is not found (anymore) that's fine. * Might be destroyed already. */ return S_OK; } /** * Notifies all callbacks which are assigned to a certain guest PID to set a certain * return/error code and an optional (error) message. * * @return IPRT status code. * @param uGuestPID Guest PID to find all callbacks for. * @param iRC Return (error) code to set for the found callbacks. * @param pszMessage Optional (error) message to set. */ int Guest::callbackNotifyAllForPID(uint32_t uGuestPID, int iRC, const char *pszMessage) { AssertReturn(uGuestPID, VERR_INVALID_PARAMETER); int vrc = VINF_SUCCESS; AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS); CallbackMapIter it; for (it = mCallbackMap.begin(); it != mCallbackMap.end(); it++) { switch (it->second.mType) { case VBOXGUESTCTRLCALLBACKTYPE_EXEC_START: break; /* When waiting for process output while the process is destroyed, * make sure we also destroy the actual waiting operation (internal progress object) * in order to not block the caller. */ case VBOXGUESTCTRLCALLBACKTYPE_EXEC_OUTPUT: { PCALLBACKDATAEXECOUT pItData = (PCALLBACKDATAEXECOUT)it->second.pvData; AssertPtr(pItData); if (pItData->u32PID == uGuestPID) vrc = callbackNotifyEx(it->first, iRC, pszMessage); break; } /* When waiting for injecting process input while the process is destroyed, * make sure we also destroy the actual waiting operation (internal progress object) * in order to not block the caller. */ case VBOXGUESTCTRLCALLBACKTYPE_EXEC_INPUT_STATUS: { PCALLBACKDATAEXECINSTATUS pItData = (PCALLBACKDATAEXECINSTATUS)it->second.pvData; AssertPtr(pItData); if (pItData->u32PID == uGuestPID) vrc = callbackNotifyEx(it->first, iRC, pszMessage); break; } default: vrc = VERR_INVALID_PARAMETER; AssertMsgFailed(("Unknown callback type %d, iRC=%d, message=%s\n", it->second.mType, iRC, pszMessage ? pszMessage : "")); break; } if (RT_FAILURE(vrc)) break; } return vrc; } int Guest::callbackNotifyComplete(uint32_t uContextID) { return callbackNotifyEx(uContextID, S_OK, NULL /* No message */); } /** * Waits for a callback (using its context ID) to complete. * * @return IPRT status code. * @param uContextID Context ID to wait for. * @param lStage Stage to wait for. Specify -1 if no staging is present/required. * Specifying a stage is only needed if there's a multi operation progress * object to wait for. * @param lTimeout Timeout (in ms) to wait for. */ int Guest::callbackWaitForCompletion(uint32_t uContextID, LONG lStage, LONG lTimeout) { AssertReturn(uContextID, VERR_INVALID_PARAMETER); /* * Wait for the HGCM low level callback until the process * has been started (or something went wrong). This is necessary to * get the PID. */ int vrc = VINF_SUCCESS; ComPtr pProgress; { AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS); CallbackMapIterConst it = mCallbackMap.find(uContextID); if (it != mCallbackMap.end()) pProgress = it->second.pProgress; else vrc = VERR_NOT_FOUND; } if (RT_SUCCESS(vrc)) { LogFlowFunc(("Waiting for callback completion (CID=%u, Stage=%RI32, timeout=%RI32ms) ...\n", uContextID, lStage, lTimeout)); HRESULT rc; if (lStage < 0) rc = pProgress->WaitForCompletion(lTimeout); else rc = pProgress->WaitForOperationCompletion((ULONG)lStage, lTimeout); if (SUCCEEDED(rc)) { if (!callbackIsComplete(uContextID)) vrc = callbackIsCanceled(uContextID) ? VERR_CANCELLED : VINF_SUCCESS; } else vrc = VERR_TIMEOUT; } LogFlowFunc(("Callback (CID=%u) completed with rc=%Rrc\n", uContextID, vrc)); return vrc; } /** * Static callback function for receiving updates on guest control commands * from the guest. Acts as a dispatcher for the actual class instance. * * @returns VBox status code. * * @todo * */ DECLCALLBACK(int) Guest::notifyCtrlDispatcher(void *pvExtension, uint32_t u32Function, void *pvParms, uint32_t cbParms) { using namespace guestControl; /* * No locking, as this is purely a notification which does not make any * changes to the object state. */ LogFlowFunc(("pvExtension=%p, u32Function=%RU32, pvParms=%p, cbParms=%RU32\n", pvExtension, u32Function, pvParms, cbParms)); ComObjPtr pGuest = reinterpret_cast(pvExtension); Assert(!pGuest.isNull()); /* * For guest control 2.0 using the legacy commands we need to do the following here: * - Get the callback header to access the context ID * - Get the context ID of the callback * - Extract the session ID out of the context ID * - Dispatch the whole stuff to the appropriate session (if still exists) */ PCALLBACKHEADER pHeader = (PCALLBACKHEADER)pvParms; AssertPtr(pHeader); #ifdef DEBUG LogFlowFunc(("CID=%RU32, uSession=%RU32, uProcess=%RU32, uCount=%RU32\n", pHeader->u32ContextID, VBOX_GUESTCTRL_CONTEXTID_GET_SESSION(pHeader->u32ContextID), VBOX_GUESTCTRL_CONTEXTID_GET_PROCESS(pHeader->u32ContextID), VBOX_GUESTCTRL_CONTEXTID_GET_COUNT(pHeader->u32ContextID))); #endif bool fDispatch = true; #ifdef DEBUG /* * Pre-check: If we got a status message with an error and VERR_TOO_MUCH_DATA * it means that that guest could not handle the entire message * because of its exceeding size. This should not happen on daily * use but testcases might try this. It then makes no sense to dispatch * this further because we don't have a valid context ID. */ if (u32Function == GUEST_EXEC_SEND_STATUS) { PCALLBACKDATAEXECSTATUS pCallbackData = reinterpret_cast(pvParms); AssertPtr(pCallbackData); AssertReturn(sizeof(CALLBACKDATAEXECSTATUS) == cbParms, VERR_INVALID_PARAMETER); AssertReturn(CALLBACKDATAMAGIC_EXEC_STATUS == pCallbackData->hdr.u32Magic, VERR_INVALID_PARAMETER); if ( pCallbackData->u32Status == PROC_STS_ERROR && ((int)pCallbackData->u32Flags) == VERR_TOO_MUCH_DATA) { LogFlowFunc(("Requested command with too much data, skipping dispatching ...\n")); Assert(pCallbackData->u32PID == 0); fDispatch = false; } } #endif int rc = VINF_SUCCESS; if (fDispatch) { rc = pGuest->dispatchToSession(pHeader->u32ContextID, u32Function, pvParms, cbParms); if (RT_SUCCESS(rc)) return rc; } #ifdef DEBUG /* @todo Don't use legacy stuff in debug mode. Remove for final! */ return rc; #endif /* Legacy handling. */ switch (u32Function) { case GUEST_DISCONNECTED: { PCALLBACKDATACLIENTDISCONNECTED pCBData = reinterpret_cast(pvParms); AssertPtr(pCBData); AssertReturn(sizeof(CALLBACKDATACLIENTDISCONNECTED) == cbParms, VERR_INVALID_PARAMETER); AssertReturn(CALLBACKDATAMAGIC_CLIENT_DISCONNECTED == pCBData->hdr.u32Magic, VERR_INVALID_PARAMETER); rc = pGuest->notifyCtrlClientDisconnected(u32Function, pCBData); break; } case GUEST_EXEC_SEND_STATUS: { PCALLBACKDATAEXECSTATUS pCBData = reinterpret_cast(pvParms); AssertPtr(pCBData); AssertReturn(sizeof(CALLBACKDATAEXECSTATUS) == cbParms, VERR_INVALID_PARAMETER); AssertReturn(CALLBACKDATAMAGIC_EXEC_STATUS == pCBData->hdr.u32Magic, VERR_INVALID_PARAMETER); rc = pGuest->notifyCtrlExecStatus(u32Function, pCBData); break; } case GUEST_EXEC_SEND_OUTPUT: { PCALLBACKDATAEXECOUT pCBData = reinterpret_cast(pvParms); AssertPtr(pCBData); AssertReturn(sizeof(CALLBACKDATAEXECOUT) == cbParms, VERR_INVALID_PARAMETER); AssertReturn(CALLBACKDATAMAGIC_EXEC_OUT == pCBData->hdr.u32Magic, VERR_INVALID_PARAMETER); rc = pGuest->notifyCtrlExecOut(u32Function, pCBData); break; } case GUEST_EXEC_SEND_INPUT_STATUS: { PCALLBACKDATAEXECINSTATUS pCBData = reinterpret_cast(pvParms); AssertPtr(pCBData); AssertReturn(sizeof(CALLBACKDATAEXECINSTATUS) == cbParms, VERR_INVALID_PARAMETER); AssertReturn(CALLBACKDATAMAGIC_EXEC_IN_STATUS == pCBData->hdr.u32Magic, VERR_INVALID_PARAMETER); rc = pGuest->notifyCtrlExecInStatus(u32Function, pCBData); break; } default: /* Silently ignore not implemented functions. */ rc = VERR_NOT_IMPLEMENTED; break; } LogFlowFuncLeaveRC(rc); return rc; } /* Function for handling the execution start/termination notification. */ /* Callback can be called several times. */ int Guest::notifyCtrlExecStatus(uint32_t u32Function, PCALLBACKDATAEXECSTATUS pData) { AssertReturn(u32Function, VERR_INVALID_PARAMETER); AssertPtrReturn(pData, VERR_INVALID_PARAMETER); uint32_t uContextID = pData->hdr.u32ContextID; /* The context ID might be 0 in case the guest was not able to fetch * actual command. So we can't do much now but report an error. */ /* Scope write locks as much as possible. */ { AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS); LogFlowFunc(("Execution status (CID=%u, pData=0x%p)\n", uContextID, pData)); PCALLBACKDATAEXECSTATUS pCallbackData = (PCALLBACKDATAEXECSTATUS)callbackGetUserDataMutableRaw(uContextID, NULL /* cbData */); if (pCallbackData) { pCallbackData->u32PID = pData->u32PID; pCallbackData->u32Status = pData->u32Status; pCallbackData->u32Flags = pData->u32Flags; /** @todo Copy void* buffer contents? */ } /* If pCallbackData is NULL this might be an old request for which no user data * might exist anymore. */ } int vrc = VINF_SUCCESS; /* Function result. */ int rcCallback = VINF_SUCCESS; /* Callback result. */ Utf8Str errMsg; /* Was progress canceled before? */ bool fCanceled = callbackIsCanceled(uContextID); if (!fCanceled) { /* Handle process map. This needs to be done first in order to have a valid * map in case some callback gets notified a bit below. */ uint32_t uHostPID = 0; /* Note: PIDs never get removed here in case the guest process signalled its * end; instead the next call of GetProcessStatus() will remove the PID * from the process map after we got the process' final (exit) status. * See waitpid() for an example. */ if (pData->u32PID) /* Only add/change a process if it has a valid PID (>0). */ { uHostPID = callbackGetHostPID(uContextID); Assert(uHostPID); switch (pData->u32Status) { /* Just reach through flags. */ case PROC_STS_TES: case PROC_STS_TOK: vrc = processSetStatus(uHostPID, pData->u32PID, (ExecuteProcessStatus_T)pData->u32Status, 0 /* Exit code */, pData->u32Flags); break; /* Interprete u32Flags as the guest process' exit code. */ default: vrc = processSetStatus(uHostPID, pData->u32PID, (ExecuteProcessStatus_T)pData->u32Status, pData->u32Flags /* Exit code */, 0 /* Flags */); break; } } if (RT_SUCCESS(vrc)) { /* Do progress handling. */ switch (pData->u32Status) { case PROC_STS_STARTED: vrc = callbackMoveForward(uContextID, Guest::tr("Waiting for process to exit ...")); LogRel(("Guest process (PID %u) started\n", pData->u32PID)); /** @todo Add process name */ break; case PROC_STS_TEN: /* Terminated normally. */ vrc = callbackNotifyComplete(uContextID); LogRel(("Guest process (PID %u) exited normally (exit code: %u)\n", pData->u32PID, pData->u32Flags)); /** @todo Add process name */ break; case PROC_STS_TEA: /* Terminated abnormally. */ LogRel(("Guest process (PID %u) terminated abnormally (exit code: %u)\n", pData->u32PID, pData->u32Flags)); /** @todo Add process name */ errMsg = Utf8StrFmt(Guest::tr("Process terminated abnormally with status '%u'"), pData->u32Flags); rcCallback = VERR_GENERAL_FAILURE; /** @todo */ break; case PROC_STS_TES: /* Terminated through signal. */ LogRel(("Guest process (PID %u) terminated through signal (exit code: %u)\n", pData->u32PID, pData->u32Flags)); /** @todo Add process name */ errMsg = Utf8StrFmt(Guest::tr("Process terminated via signal with status '%u'"), pData->u32Flags); rcCallback = VERR_GENERAL_FAILURE; /** @todo */ break; case PROC_STS_TOK: LogRel(("Guest process (PID %u) timed out and was killed\n", pData->u32PID)); /** @todo Add process name */ errMsg = Utf8StrFmt(Guest::tr("Process timed out and was killed")); rcCallback = VERR_TIMEOUT; break; case PROC_STS_TOA: LogRel(("Guest process (PID %u) timed out and could not be killed\n", pData->u32PID)); /** @todo Add process name */ errMsg = Utf8StrFmt(Guest::tr("Process timed out and could not be killed")); rcCallback = VERR_TIMEOUT; break; case PROC_STS_DWN: LogRel(("Guest process (PID %u) killed because system is shutting down\n", pData->u32PID)); /** @todo Add process name */ /* * If u32Flags has ExecuteProcessFlag_IgnoreOrphanedProcesses set, we don't report an error to * our progress object. This is helpful for waiters which rely on the success of our progress object * even if the executed process was killed because the system/VBoxService is shutting down. * * In this case u32Flags contains the actual execution flags reached in via Guest::ExecuteProcess(). */ if (pData->u32Flags & ExecuteProcessFlag_IgnoreOrphanedProcesses) { vrc = callbackNotifyComplete(uContextID); } else { errMsg = Utf8StrFmt(Guest::tr("Process killed because system is shutting down")); rcCallback = VERR_CANCELLED; } break; case PROC_STS_ERROR: { Utf8Str errDetail; if (pData->u32PID) { errDetail = Utf8StrFmt(Guest::tr("Guest process (PID %u) could not be started because of rc=%Rrc"), pData->u32PID, pData->u32Flags); } else { switch (pData->u32Flags) /* u32Flags member contains the IPRT error code from guest side. */ { case VERR_FILE_NOT_FOUND: /* This is the most likely error. */ errDetail = Utf8StrFmt(Guest::tr("The specified file was not found on guest")); break; case VERR_PATH_NOT_FOUND: errDetail = Utf8StrFmt(Guest::tr("Could not resolve path to specified file was not found on guest")); break; case VERR_BAD_EXE_FORMAT: errDetail = Utf8StrFmt(Guest::tr("The specified file is not an executable format on guest")); break; case VERR_AUTHENTICATION_FAILURE: errDetail = Utf8StrFmt(Guest::tr("The specified user was not able to logon on guest")); break; case VERR_TIMEOUT: errDetail = Utf8StrFmt(Guest::tr("The guest did not respond within time")); break; case VERR_CANCELLED: errDetail = Utf8StrFmt(Guest::tr("The execution operation was canceled")); break; case VERR_PERMISSION_DENIED: errDetail = Utf8StrFmt(Guest::tr("Invalid user/password credentials")); break; case VERR_MAX_PROCS_REACHED: errDetail = Utf8StrFmt(Guest::tr("Guest process could not be started because maximum number of parallel guest processes has been reached")); break; default: errDetail = Utf8StrFmt(Guest::tr("Guest process reported error %Rrc"), pData->u32Flags); break; } } errMsg = Utf8StrFmt(Guest::tr("Process execution failed: "), pData->u32Flags) + errDetail; rcCallback = pData->u32Flags; /* Report back guest rc. */ LogRel((errMsg.c_str())); break; } default: vrc = VERR_INVALID_PARAMETER; break; } } } else { errMsg = Utf8StrFmt(Guest::tr("Process execution canceled")); rcCallback = VERR_CANCELLED; } /* Do we need to handle the callback error? */ if (RT_FAILURE(rcCallback)) { AssertMsg(!errMsg.isEmpty(), ("Error message must not be empty!\n")); if (uContextID) { /* Notify all callbacks which are still waiting on something * which is related to the current PID. */ if (pData->u32PID) { int rc2 = callbackNotifyAllForPID(pData->u32PID, rcCallback, errMsg.c_str()); if (RT_FAILURE(rc2)) { LogFlowFunc(("Failed to notify other callbacks for PID=%u\n", pData->u32PID)); if (RT_SUCCESS(vrc)) vrc = rc2; } } /* Let the caller know what went wrong ... */ int rc2 = callbackNotifyEx(uContextID, rcCallback, errMsg.c_str()); if (RT_FAILURE(rc2)) { LogFlowFunc(("Failed to notify callback CID=%u for PID=%u\n", uContextID, pData->u32PID)); if (RT_SUCCESS(vrc)) vrc = rc2; } } else { /* Since we don't know which context exactly failed all we can do is to shutdown * all contexts ... */ errMsg = Utf8StrFmt(Guest::tr("Client reported critical error %Rrc -- shutting down"), pData->u32Flags); /* Cancel all callbacks. */ CallbackMapIter it; for (it = mCallbackMap.begin(); it != mCallbackMap.end(); it++) { int rc2 = callbackNotifyEx(it->first, VERR_CANCELLED, errMsg.c_str()); AssertRC(rc2); } } LogFlowFunc(("Process (CID=%u, status=%u) reported: %s\n", uContextID, pData->u32Status, errMsg.c_str())); } LogFlowFunc(("Returned with rc=%Rrc, rcCallback=%Rrc\n", vrc, rcCallback)); return vrc; } /* Function for handling the execution output notification. */ int Guest::notifyCtrlExecOut(uint32_t u32Function, PCALLBACKDATAEXECOUT pData) { AssertReturn(u32Function, VERR_INVALID_PARAMETER); AssertPtrReturn(pData, VERR_INVALID_PARAMETER); uint32_t uContextID = pData->hdr.u32ContextID; Assert(uContextID); /* Scope write locks as much as possible. */ { AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS); LogFlowFunc(("Output status (CID=%u, pData=0x%p)\n", uContextID, pData)); PCALLBACKDATAEXECOUT pCallbackData = (PCALLBACKDATAEXECOUT)callbackGetUserDataMutableRaw(uContextID, NULL /* cbData */); if (pCallbackData) { pCallbackData->u32PID = pData->u32PID; pCallbackData->u32HandleId = pData->u32HandleId; pCallbackData->u32Flags = pData->u32Flags; /* Make sure we really got something! */ if ( pData->cbData && pData->pvData) { callbackFreeUserData(pCallbackData->pvData); /* Allocate data buffer and copy it */ pCallbackData->pvData = RTMemAlloc(pData->cbData); pCallbackData->cbData = pData->cbData; AssertReturn(pCallbackData->pvData, VERR_NO_MEMORY); memcpy(pCallbackData->pvData, pData->pvData, pData->cbData); } else /* Nothing received ... */ { pCallbackData->pvData = NULL; pCallbackData->cbData = 0; } } /* If pCallbackData is NULL this might be an old request for which no user data * might exist anymore. */ } int vrc; if (callbackIsCanceled(pData->u32PID)) { vrc = callbackNotifyEx(uContextID, VERR_CANCELLED, Guest::tr("The output operation was canceled")); } else vrc = callbackNotifyComplete(uContextID); return vrc; } /* Function for handling the execution input status notification. */ int Guest::notifyCtrlExecInStatus(uint32_t u32Function, PCALLBACKDATAEXECINSTATUS pData) { AssertReturn(u32Function, VERR_INVALID_PARAMETER); AssertPtrReturn(pData, VERR_INVALID_PARAMETER); uint32_t uContextID = pData->hdr.u32ContextID; Assert(uContextID); /* Scope write locks as much as possible. */ { AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS); LogFlowFunc(("Input status (CID=%u, pData=0x%p)\n", uContextID, pData)); PCALLBACKDATAEXECINSTATUS pCallbackData = (PCALLBACKDATAEXECINSTATUS)callbackGetUserDataMutableRaw(uContextID, NULL /* cbData */); if (pCallbackData) { /* Save bytes processed. */ pCallbackData->cbProcessed = pData->cbProcessed; pCallbackData->u32Status = pData->u32Status; pCallbackData->u32Flags = pData->u32Flags; pCallbackData->u32PID = pData->u32PID; } /* If pCallbackData is NULL this might be an old request for which no user data * might exist anymore. */ } return callbackNotifyComplete(uContextID); } int Guest::notifyCtrlClientDisconnected(uint32_t u32Function, PCALLBACKDATACLIENTDISCONNECTED pData) { /* u32Function is 0. */ AssertPtrReturn(pData, VERR_INVALID_PARAMETER); uint32_t uContextID = pData->hdr.u32ContextID; Assert(uContextID); LogFlowFunc(("Client disconnected (CID=%u)\n,", uContextID)); return callbackNotifyEx(uContextID, VERR_CANCELLED, Guest::tr("Client disconnected")); } uint32_t Guest::processGetGuestPID(uint32_t uHostPID) { AssertReturn(uHostPID, 0); AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS); GuestProcessMapIter it = mGuestProcessMap.find(uHostPID); if (it == mGuestProcessMap.end()) return 0; return it->second.mGuestPID; } /** * Gets guest process information. Removes the process from the map * after the process was marked as exited/terminated. * * @return IPRT status code. * @param uHostPID Host PID of guest process to get status for. * @param pProcess Where to store the process information. Optional. * @param fRemove Flag indicating whether to remove the * process from the map when process marked a * exited/terminated. */ int Guest::processGetStatus(uint32_t uHostPID, PVBOXGUESTCTRL_PROCESS pProcess, bool fRemove) { AssertReturn(uHostPID, VERR_INVALID_PARAMETER); AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS); GuestProcessMapIter it = mGuestProcessMap.find(uHostPID); if (it != mGuestProcessMap.end()) { if (pProcess) { pProcess->mGuestPID = it->second.mGuestPID; pProcess->mStatus = it->second.mStatus; pProcess->mExitCode = it->second.mExitCode; pProcess->mFlags = it->second.mFlags; } /* Only remove processes from our map when they signalled their final * status. */ if ( fRemove && ( it->second.mStatus != ExecuteProcessStatus_Undefined && it->second.mStatus != ExecuteProcessStatus_Started)) { mGuestProcessMap.erase(it); } return VINF_SUCCESS; } return VERR_NOT_FOUND; } /** * Sets the current status of a guest process. * * @return IPRT status code. * @param uHostPID Host PID of guest process to set status (and guest PID) for. * @param uGuestPID Guest PID to assign to the host PID. * @param enmStatus Current status to set. * @param uExitCode Exit code (if any). * @param uFlags Additional flags. */ int Guest::processSetStatus(uint32_t uHostPID, uint32_t uGuestPID, ExecuteProcessStatus_T enmStatus, uint32_t uExitCode, uint32_t uFlags) { AssertReturn(uHostPID, VERR_INVALID_PARAMETER); /* Assigning a guest PID is optional. */ AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS); GuestProcessMapIter it = mGuestProcessMap.find(uHostPID); if (it != mGuestProcessMap.end()) { it->second.mGuestPID = uGuestPID; it->second.mStatus = enmStatus; it->second.mExitCode = uExitCode; it->second.mFlags = uFlags; } else { VBOXGUESTCTRL_PROCESS process; /* uGuestPID is optional -- the caller could call this function * before the guest process actually was started and a (valid) guest PID * was returned. */ process.mGuestPID = uGuestPID; process.mStatus = enmStatus; process.mExitCode = uExitCode; process.mFlags = uFlags; mGuestProcessMap[uHostPID] = process; } return VINF_SUCCESS; } HRESULT Guest::setErrorCompletion(int rc) { HRESULT hRC; if (rc == VERR_NOT_FOUND) hRC = setErrorNoLog(VBOX_E_VM_ERROR, tr("VMM device is not available (is the VM running?)")); else if (rc == VERR_CANCELLED) hRC = setErrorNoLog(VBOX_E_IPRT_ERROR, tr("Process execution has been canceled")); else if (rc == VERR_TIMEOUT) hRC= setErrorNoLog(VBOX_E_IPRT_ERROR, tr("The guest did not respond within time")); else hRC = setErrorNoLog(E_UNEXPECTED, tr("Waiting for completion failed with error %Rrc"), rc); return hRC; } HRESULT Guest::setErrorFromProgress(ComPtr pProgress) { BOOL fCompleted; HRESULT rc = pProgress->COMGETTER(Completed)(&fCompleted); ComAssertComRC(rc); LONG rcProc = S_OK; Utf8Str strError; if (!fCompleted) { BOOL fCanceled; rc = pProgress->COMGETTER(Canceled)(&fCanceled); ComAssertComRC(rc); strError = fCanceled ? Utf8StrFmt(Guest::tr("Process execution was canceled")) : Utf8StrFmt(Guest::tr("Process neither completed nor canceled; this shouldn't happen")); } else { rc = pProgress->COMGETTER(ResultCode)(&rcProc); ComAssertComRC(rc); if (FAILED(rcProc)) { ProgressErrorInfo info(pProgress); strError = info.getText(); } } if (FAILED(rcProc)) { AssertMsg(!strError.isEmpty(), ("Error message must not be empty!\n")); return setErrorInternal(rcProc, this->getClassIID(), this->getComponentName(), strError, false /* aWarning */, false /* aLogIt */); } return S_OK; } HRESULT Guest::setErrorHGCM(int rc) { HRESULT hRC; if (rc == VERR_INVALID_VM_HANDLE) hRC = setErrorNoLog(VBOX_E_VM_ERROR, tr("VMM device is not available (is the VM running?)")); else if (rc == VERR_NOT_FOUND) hRC = setErrorNoLog(VBOX_E_IPRT_ERROR, tr("The guest execution service is not ready (yet)")); else if (rc == VERR_HGCM_SERVICE_NOT_FOUND) hRC= setErrorNoLog(VBOX_E_IPRT_ERROR, tr("The guest execution service is not available")); else /* HGCM call went wrong. */ hRC = setErrorNoLog(E_UNEXPECTED, tr("The HGCM call failed with error %Rrc"), rc); return hRC; } #endif /* VBOX_WITH_GUEST_CONTROL */ STDMETHODIMP Guest::ExecuteProcess(IN_BSTR aCommand, ULONG aFlags, ComSafeArrayIn(IN_BSTR, aArguments), ComSafeArrayIn(IN_BSTR, aEnvironment), IN_BSTR aUsername, IN_BSTR aPassword, ULONG aTimeoutMS, ULONG *aPID, IProgress **aProgress) { /** @todo r=bird: Eventually we should clean up all the timeout parameters * in the API and have the same way of specifying infinite waits! */ #ifndef VBOX_WITH_GUEST_CONTROL ReturnComNotImplemented(); #else /* VBOX_WITH_GUEST_CONTROL */ using namespace guestControl; CheckComArgStrNotEmptyOrNull(aCommand); CheckComArgOutPointerValid(aPID); CheckComArgOutPointerValid(aProgress); /* Do not allow anonymous executions (with system rights). */ if (RT_UNLIKELY((aUsername) == NULL || *(aUsername) == '\0')) return setError(E_INVALIDARG, tr("No user name specified")); LogRel(("Executing guest process \"%s\" as user \"%s\" ...\n", Utf8Str(aCommand).c_str(), Utf8Str(aUsername).c_str())); return executeProcessInternal(aCommand, aFlags, ComSafeArrayInArg(aArguments), ComSafeArrayInArg(aEnvironment), aUsername, aPassword, aTimeoutMS, aPID, aProgress, NULL /* rc */); #endif } #ifdef VBOX_WITH_GUEST_CONTROL /** * Executes and waits for an internal tool (that is, a tool which is integrated into * VBoxService, beginning with "vbox_" (e.g. "vbox_ls")) to finish its operation. * * @return HRESULT * @param aTool Name of tool to execute. * @param aDescription Friendly description of the operation. * @param aFlags Execution flags. * @param aUsername Username to execute tool under. * @param aPassword The user's password. * @param uFlagsToAdd ExecuteProcessFlag flags to add to the execution operation. * @param aProgress Pointer which receives the tool's progress object. Optional. * @param aPID Pointer which receives the tool's PID. Optional. */ HRESULT Guest::executeAndWaitForTool(IN_BSTR aTool, IN_BSTR aDescription, ComSafeArrayIn(IN_BSTR, aArguments), ComSafeArrayIn(IN_BSTR, aEnvironment), IN_BSTR aUsername, IN_BSTR aPassword, ULONG uFlagsToAdd, GuestCtrlStreamObjects *pObjStdOut, GuestCtrlStreamObjects *pObjStdErr, IProgress **aProgress, ULONG *aPID) { ComPtr pProgress; ULONG uPID; ULONG uFlags = ExecuteProcessFlag_Hidden; if (uFlagsToAdd) uFlags |= uFlagsToAdd; bool fParseOutput = false; if ( ( (uFlags & ExecuteProcessFlag_WaitForStdOut) && pObjStdOut) || ( (uFlags & ExecuteProcessFlag_WaitForStdErr) && pObjStdErr)) { fParseOutput = true; } HRESULT hr = ExecuteProcess(aTool, uFlags, ComSafeArrayInArg(aArguments), ComSafeArrayInArg(aEnvironment), aUsername, aPassword, 0 /* No timeout */, &uPID, pProgress.asOutParam()); if (SUCCEEDED(hr)) { /* Wait for tool being started. */ hr = pProgress->WaitForOperationCompletion( 0 /* Stage, starting the process */, -1 /* No timeout */); } if ( SUCCEEDED(hr) && !(uFlags & ExecuteProcessFlag_WaitForProcessStartOnly)) { if (!fParseOutput) { if ( !(uFlags & ExecuteProcessFlag_WaitForStdOut) && !(uFlags & ExecuteProcessFlag_WaitForStdErr)) { hr = executeWaitForExit(uPID, pProgress, 0 /* No timeout */); } } else { BOOL fCompleted; while ( SUCCEEDED(pProgress->COMGETTER(Completed)(&fCompleted)) && !fCompleted) { BOOL fCanceled; hr = pProgress->COMGETTER(Canceled)(&fCanceled); AssertComRC(hr); if (fCanceled) { hr = setErrorNoLog(VBOX_E_IPRT_ERROR, tr("%s was cancelled"), Utf8Str(aDescription).c_str()); break; } if ( (uFlags & ExecuteProcessFlag_WaitForStdOut) && pObjStdOut) { hr = executeStreamParse(uPID, ProcessOutputFlag_None /* StdOut */, *pObjStdOut); } if ( (uFlags & ExecuteProcessFlag_WaitForStdErr) && pObjStdErr) { hr = executeStreamParse(uPID, ProcessOutputFlag_StdErr, *pObjStdErr); } if (FAILED(hr)) break; } } } if (SUCCEEDED(hr)) { if (aProgress) { /* Return the progress to the caller. */ pProgress.queryInterfaceTo(aProgress); } else if (!pProgress.isNull()) pProgress.setNull(); if (aPID) *aPID = uPID; } return hr; } /** * TODO * * @return HRESULT * @param aObjName * @param pStreamBlock * @param pObjInfo * @param enmAddAttribs */ int Guest::executeStreamQueryFsObjInfo(IN_BSTR aObjName, GuestProcessStreamBlock &streamBlock, PRTFSOBJINFO pObjInfo, RTFSOBJATTRADD enmAddAttribs) { Utf8Str Utf8ObjName(aObjName); int64_t iVal; int rc = streamBlock.GetInt64Ex("st_size", &iVal); if (RT_SUCCESS(rc)) pObjInfo->cbObject = iVal; /** @todo Add more stuff! */ return rc; } /** * Tries to drain the guest's output and fill it into * a guest process stream object for later usage. * * @todo What's about specifying stderr? * @return IPRT status code. * @param aPID PID of process to get the output from. * @param aFlags Which stream to drain (stdout or stderr). * @param pStream Pointer to guest process stream to fill. If NULL, * data goes to /dev/null. */ int Guest::executeStreamDrain(ULONG aPID, ULONG aFlags, GuestProcessStream *pStream) { AssertReturn(aPID, VERR_INVALID_PARAMETER); /* pStream is optional. */ int rc = VINF_SUCCESS; for (;;) { SafeArray aData; HRESULT hr = getProcessOutputInternal(aPID, aFlags, 0 /* Infinite timeout */, _64K, ComSafeArrayAsOutParam(aData), &rc); if (RT_SUCCESS(rc)) { if ( pStream && aData.size()) { rc = pStream->AddData(aData.raw(), aData.size()); if (RT_UNLIKELY(RT_FAILURE(rc))) break; } continue; /* Try one more time. */ } else /* No more output and/or error! */ { if ( rc == VERR_NOT_FOUND || rc == VERR_BROKEN_PIPE) { rc = VINF_SUCCESS; } break; } } return rc; } /** * Tries to retrieve the next stream block of a given stream and * drains the process output only as much as needed to get this next * stream block. * * @return IPRT status code. * @param ulPID * @param ulFlags * @param stream * @param streamBlock */ int Guest::executeStreamGetNextBlock(ULONG ulPID, ULONG ulFlags, GuestProcessStream &stream, GuestProcessStreamBlock &streamBlock) { AssertReturn(!streamBlock.GetCount(), VERR_INVALID_PARAMETER); LogFlowFunc(("Getting next stream block of PID=%u, Flags=%u; cbStrmSize=%u, cbStrmOff=%u\n", ulPID, ulFlags, stream.GetSize(), stream.GetOffset())); int rc; uint32_t cPairs = 0; bool fDrainStream = true; do { rc = stream.ParseBlock(streamBlock); LogFlowFunc(("Parsing block rc=%Rrc, strmBlockCnt=%ld\n", rc, streamBlock.GetCount())); if (RT_FAILURE(rc)) /* More data needed or empty buffer? */ { if (fDrainStream) { SafeArray aData; HRESULT hr = getProcessOutputInternal(ulPID, ulFlags, 0 /* Infinite timeout */, _64K, ComSafeArrayAsOutParam(aData), &rc); if (SUCCEEDED(hr)) { LogFlowFunc(("Got %ld bytes of additional data\n", aData.size())); if (RT_FAILURE(rc)) { if (rc == VERR_BROKEN_PIPE) rc = VINF_SUCCESS; /* No more data because process already ended. */ break; } if (aData.size()) { rc = stream.AddData(aData.raw(), aData.size()); if (RT_UNLIKELY(RT_FAILURE(rc))) break; } /* Reset found pairs to not break out too early and let all the new * data to be parsed as well. */ cPairs = 0; continue; /* Try one more time. */ } else { LogFlowFunc(("Getting output returned hr=%Rhrc\n", hr)); /* No more output to drain from stream. */ if (rc == VERR_NOT_FOUND) { fDrainStream = false; continue; } break; } } else { /* We haved drained the stream as much as we can and reached the * end of our stream buffer -- that means that there simply is no * stream block anymore, which is ok. */ if (rc == VERR_NO_DATA) rc = VINF_SUCCESS; break; } } else cPairs = streamBlock.GetCount(); } while (!cPairs); LogFlowFunc(("Returned with strmBlockCnt=%ld, cPairs=%ld, rc=%Rrc\n", streamBlock.GetCount(), cPairs, rc)); return rc; } /** * Tries to get the next upcoming value block from a started guest process * by first draining its output and then processing the received guest stream. * * @return IPRT status code. * @param ulPID PID of process to get/parse the output from. * @param ulFlags ? * @param stream Reference to process stream object to use. * @param streamBlock Reference that receives the next stream block data. * */ int Guest::executeStreamParseNextBlock(ULONG ulPID, ULONG ulFlags, GuestProcessStream &stream, GuestProcessStreamBlock &streamBlock) { AssertReturn(!streamBlock.GetCount(), VERR_INVALID_PARAMETER); int rc; do { rc = stream.ParseBlock(streamBlock); if (RT_FAILURE(rc)) break; } while (!streamBlock.GetCount()); return rc; } /** * Gets output from a formerly started guest process, tries to parse all of its guest * stream (as long as data is available) and returns a stream object which can contain * multiple stream blocks (which in turn then can contain key=value pairs). * * @return HRESULT * @param ulPID PID of process to get/parse the output from. * @param ulFlags ? * @param streamObjects Reference to a guest stream object structure for * storing the parsed data. */ HRESULT Guest::executeStreamParse(ULONG ulPID, ULONG ulFlags, GuestCtrlStreamObjects &streamObjects) { GuestProcessStream stream; int rc = executeStreamDrain(ulPID, ulFlags, &stream); if (RT_SUCCESS(rc)) { do { /* Try to parse the stream output we gathered until now. If we still need more * data the parsing routine will tell us and we just do another poll round. */ GuestProcessStreamBlock curBlock; rc = executeStreamParseNextBlock(ulPID, ulFlags, stream, curBlock); if (RT_SUCCESS(rc)) streamObjects.push_back(curBlock); } while (RT_SUCCESS(rc)); if (rc == VERR_NO_DATA) /* End of data reached. */ rc = VINF_SUCCESS; } if (RT_FAILURE(rc)) return setError(VBOX_E_IPRT_ERROR, tr("Error while parsing guest output (%Rrc)"), rc); return rc; } /** * Waits for a fomerly started guest process to exit using its progress * object and returns its final status. Returns E_ABORT if guest process * was canceled. * * @return IPRT status code. * @param uPID PID of guest process to wait for. * @param pProgress Progress object to wait for. * @param uTimeoutMS Timeout (in ms) for waiting; use 0 for * an indefinite timeout. * @param pRetStatus Pointer where to store the final process * status. Optional. * @param puRetExitCode Pointer where to store the final process * exit code. Optional. */ HRESULT Guest::executeWaitForExit(ULONG uPID, ComPtr pProgress, ULONG uTimeoutMS) { HRESULT rc = S_OK; BOOL fCanceled = FALSE; if ( SUCCEEDED(pProgress->COMGETTER(Canceled(&fCanceled))) && fCanceled) { return E_ABORT; } BOOL fCompleted = FALSE; if ( SUCCEEDED(pProgress->COMGETTER(Completed(&fCompleted))) && !fCompleted) { rc = pProgress->WaitForCompletion( !uTimeoutMS ? -1 /* No timeout */ : uTimeoutMS); if (FAILED(rc)) rc = setError(VBOX_E_IPRT_ERROR, tr("Waiting for guest process to end failed (%Rhrc)"), rc); } return rc; } /** * Does the actual guest process execution, internal function. * * @return HRESULT * @param aCommand Command line to execute. * @param aFlags Execution flags. * @param Username Username to execute the process with. * @param aPassword The user's password. * @param aTimeoutMS Timeout (in ms) to wait for the execution operation. * @param aPID Pointer that receives the guest process' PID. * @param aProgress Pointer that receives the guest process' progress object. * @param pRC Pointer that receives the internal IPRT return code. Optional. */ HRESULT Guest::executeProcessInternal(IN_BSTR aCommand, ULONG aFlags, ComSafeArrayIn(IN_BSTR, aArguments), ComSafeArrayIn(IN_BSTR, aEnvironment), IN_BSTR aUsername, IN_BSTR aPassword, ULONG aTimeoutMS, ULONG *aPID, IProgress **aProgress, int *pRC) { /** @todo r=bird: Eventually we should clean up all the timeout parameters * in the API and have the same way of specifying infinite waits! */ using namespace guestControl; AutoCaller autoCaller(this); if (FAILED(autoCaller.rc())) return autoCaller.rc(); /* Validate flags. */ if (aFlags != ExecuteProcessFlag_None) { if ( !(aFlags & ExecuteProcessFlag_IgnoreOrphanedProcesses) && !(aFlags & ExecuteProcessFlag_WaitForProcessStartOnly) && !(aFlags & ExecuteProcessFlag_Hidden) && !(aFlags & ExecuteProcessFlag_NoProfile) && !(aFlags & ExecuteProcessFlag_WaitForStdOut) && !(aFlags & ExecuteProcessFlag_WaitForStdErr)) { if (pRC) *pRC = VERR_INVALID_PARAMETER; return setError(E_INVALIDARG, tr("Unknown flags (%#x)"), aFlags); } } HRESULT rc = S_OK; try { /* * Create progress object. Note that this is a multi operation * object to perform the following steps: * - Operation 1 (0): Create/start process. * - Operation 2 (1): Wait for process to exit. * If this progress completed successfully (S_OK), the process * started and exited normally. In any other case an error/exception * occurred. */ ComObjPtr pProgress; rc = pProgress.createObject(); if (SUCCEEDED(rc)) { rc = pProgress->init(static_cast(this), Bstr(tr("Executing process")).raw(), TRUE, 2, /* Number of operations. */ Bstr(tr("Starting process ...")).raw()); /* Description of first stage. */ } ComAssertComRC(rc); /* * Prepare process execution. */ int vrc = VINF_SUCCESS; Utf8Str Utf8Command(aCommand); /* Adjust timeout. If set to 0, we define * an infinite timeout. */ if (aTimeoutMS == 0) aTimeoutMS = UINT32_MAX; /* Prepare arguments. */ char **papszArgv = NULL; uint32_t uNumArgs = 0; if (aArguments) { com::SafeArray args(ComSafeArrayInArg(aArguments)); uNumArgs = args.size(); papszArgv = (char**)RTMemAlloc(sizeof(char*) * (uNumArgs + 1)); AssertReturn(papszArgv, E_OUTOFMEMORY); for (unsigned i = 0; RT_SUCCESS(vrc) && i < uNumArgs; i++) vrc = RTUtf16ToUtf8(args[i], &papszArgv[i]); papszArgv[uNumArgs] = NULL; } Utf8Str Utf8UserName(aUsername); Utf8Str Utf8Password(aPassword); uint32_t uHostPID = 0; if (RT_SUCCESS(vrc)) { uint32_t uContextID = 0; char *pszArgs = NULL; if (uNumArgs > 0) vrc = RTGetOptArgvToString(&pszArgs, papszArgv, RTGETOPTARGV_CNV_QUOTE_MS_CRT); if (RT_SUCCESS(vrc)) { uint32_t cbArgs = pszArgs ? strlen(pszArgs) + 1 : 0; /* Include terminating zero. */ /* Prepare environment. */ void *pvEnv = NULL; uint32_t uNumEnv = 0; uint32_t cbEnv = 0; if (aEnvironment) { com::SafeArray env(ComSafeArrayInArg(aEnvironment)); for (unsigned i = 0; i < env.size(); i++) { vrc = prepareExecuteEnv(Utf8Str(env[i]).c_str(), &pvEnv, &cbEnv, &uNumEnv); if (RT_FAILURE(vrc)) break; } } if (RT_SUCCESS(vrc)) { VBOXGUESTCTRL_CALLBACK callback; vrc = callbackInit(&callback, VBOXGUESTCTRLCALLBACKTYPE_EXEC_START, pProgress); if (RT_SUCCESS(vrc)) vrc = callbackAdd(&callback, &uContextID); if (RT_SUCCESS(vrc)) { VBOXHGCMSVCPARM paParms[15]; int i = 0; paParms[i++].setUInt32(uContextID); paParms[i++].setPointer((void*)Utf8Command.c_str(), (uint32_t)Utf8Command.length() + 1); paParms[i++].setUInt32(aFlags); paParms[i++].setUInt32(uNumArgs); paParms[i++].setPointer((void*)pszArgs, cbArgs); paParms[i++].setUInt32(uNumEnv); paParms[i++].setUInt32(cbEnv); paParms[i++].setPointer((void*)pvEnv, cbEnv); paParms[i++].setPointer((void*)Utf8UserName.c_str(), (uint32_t)Utf8UserName.length() + 1); paParms[i++].setPointer((void*)Utf8Password.c_str(), (uint32_t)Utf8Password.length() + 1); /* * If the WaitForProcessStartOnly flag is set, we only want to define and wait for a timeout * until the process was started - the process itself then gets an infinite timeout for execution. * This is handy when we want to start a process inside a worker thread within a certain timeout * but let the started process perform lengthly operations then. */ if (aFlags & ExecuteProcessFlag_WaitForProcessStartOnly) paParms[i++].setUInt32(UINT32_MAX /* Infinite timeout */); else paParms[i++].setUInt32(aTimeoutMS); VMMDev *pVMMDev = NULL; { /* Make sure mParent is valid, so set the read lock while using. * Do not keep this lock while doing the actual call, because in the meanwhile * another thread could request a write lock which would be a bad idea ... */ AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS); /* Forward the information to the VMM device. */ AssertPtr(mParent); pVMMDev = mParent->getVMMDev(); } if (pVMMDev) { LogFlowFunc(("hgcmHostCall numParms=%d\n", i)); vrc = pVMMDev->hgcmHostCall("VBoxGuestControlSvc", HOST_EXEC_CMD, i, paParms); } else vrc = VERR_INVALID_VM_HANDLE; } RTMemFree(pvEnv); } RTStrFree(pszArgs); } if (RT_SUCCESS(vrc)) { AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS); /* * Generate a host-driven PID so that we immediately can return to the caller and * don't need to wait until the guest started the desired process to return the * PID generated by the guest OS. * * The guest PID will later be mapped to the host PID for later lookup. */ vrc = VERR_NOT_FOUND; /* We did not find a host PID yet ... */ uint32_t uTries = 0; for (;;) { /* Create a new context ID ... */ uHostPID = ASMAtomicIncU32(&mNextHostPID); if (uHostPID == UINT32_MAX) ASMAtomicUoWriteU32(&mNextHostPID, 1000); /* Is the host PID already used? Try next PID ... */ GuestProcessMapIter it = mGuestProcessMap.find(uHostPID); if (it == mGuestProcessMap.end()) { /* Host PID not used (anymore), we're done here ... */ vrc = VINF_SUCCESS; break; } if (++uTries == UINT32_MAX) break; /* Don't try too hard. */ } if (RT_SUCCESS(vrc)) vrc = processSetStatus(uHostPID, 0 /* No guest PID yet */, ExecuteProcessStatus_Undefined /* Process not started yet */, 0 /* uExitCode */, 0 /* uFlags */); if (RT_SUCCESS(vrc)) vrc = callbackAssignHostPID(uContextID, uHostPID); } else rc = setErrorHGCM(vrc); for (unsigned i = 0; i < uNumArgs; i++) RTMemFree(papszArgv[i]); RTMemFree(papszArgv); if (RT_FAILURE(vrc)) rc = VBOX_E_IPRT_ERROR; } if (SUCCEEDED(rc)) { /* Return host PID. */ *aPID = uHostPID; /* Return the progress to the caller. */ pProgress.queryInterfaceTo(aProgress); } else { if (!pRC) /* Skip logging internal calls. */ LogRel(("Executing guest process \"%s\" as user \"%s\" failed with %Rrc\n", Utf8Command.c_str(), Utf8UserName.c_str(), vrc)); } if (pRC) *pRC = vrc; } catch (std::bad_alloc &) { rc = E_OUTOFMEMORY; if (pRC) *pRC = VERR_NO_MEMORY; } return rc; } #endif /* VBOX_WITH_GUEST_CONTROL */ STDMETHODIMP Guest::SetProcessInput(ULONG aPID, ULONG aFlags, ULONG aTimeoutMS, ComSafeArrayIn(BYTE, aData), ULONG *aBytesWritten) { #ifndef VBOX_WITH_GUEST_CONTROL ReturnComNotImplemented(); #else /* VBOX_WITH_GUEST_CONTROL */ using namespace guestControl; CheckComArgExpr(aPID, aPID > 0); CheckComArgOutPointerValid(aBytesWritten); /* Validate flags. */ if (aFlags) { if (!(aFlags & ProcessInputFlag_EndOfFile)) return setError(E_INVALIDARG, tr("Unknown flags (%#x)"), aFlags); } AutoCaller autoCaller(this); if (FAILED(autoCaller.rc())) return autoCaller.rc(); HRESULT rc = S_OK; try { VBOXGUESTCTRL_PROCESS process; int vrc = processGetStatus(aPID, &process, false /* Don't remove */); if (RT_SUCCESS(vrc)) { /* PID exists; check if process is still running. */ if (process.mStatus != ExecuteProcessStatus_Started) rc = setError(VBOX_E_IPRT_ERROR, Guest::tr("Cannot inject input to a not running process (PID %u)"), aPID); } else rc = setError(VBOX_E_IPRT_ERROR, Guest::tr("Cannot inject input to a non-existent process (PID %u)"), aPID); if (RT_SUCCESS(vrc)) { uint32_t uContextID = 0; uint32_t uGuestPID = processGetGuestPID(aPID); Assert(uGuestPID); /* * Create progress object. * This progress object, compared to the one in executeProgress() above, * is only single-stage local and is used to determine whether the operation * finished or got canceled. */ ComObjPtr pProgress; rc = pProgress.createObject(); if (SUCCEEDED(rc)) { rc = pProgress->init(static_cast(this), Bstr(tr("Setting input for process")).raw(), TRUE /* Cancelable */); } if (FAILED(rc)) throw rc; ComAssert(!pProgress.isNull()); /* Adjust timeout. */ if (aTimeoutMS == 0) aTimeoutMS = UINT32_MAX; VBOXGUESTCTRL_CALLBACK callback; vrc = callbackInit(&callback, VBOXGUESTCTRLCALLBACKTYPE_EXEC_INPUT_STATUS, pProgress); if (RT_SUCCESS(vrc)) { PCALLBACKDATAEXECINSTATUS pData = (PCALLBACKDATAEXECINSTATUS)callback.pvData; /* Save PID + output flags for later use. */ pData->u32PID = uGuestPID; pData->u32Flags = aFlags; } if (RT_SUCCESS(vrc)) vrc = callbackAdd(&callback, &uContextID); if (RT_SUCCESS(vrc)) { com::SafeArray sfaData(ComSafeArrayInArg(aData)); uint32_t cbSize = sfaData.size(); VBOXHGCMSVCPARM paParms[6]; int i = 0; paParms[i++].setUInt32(uContextID); paParms[i++].setUInt32(uGuestPID); paParms[i++].setUInt32(aFlags); paParms[i++].setPointer(sfaData.raw(), cbSize); paParms[i++].setUInt32(cbSize); { VMMDev *pVMMDev = NULL; { /* Make sure mParent is valid, so set the read lock while using. * Do not keep this lock while doing the actual call, because in the meanwhile * another thread could request a write lock which would be a bad idea ... */ AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS); /* Forward the information to the VMM device. */ AssertPtr(mParent); pVMMDev = mParent->getVMMDev(); } if (pVMMDev) { LogFlowFunc(("hgcmHostCall numParms=%d\n", i)); vrc = pVMMDev->hgcmHostCall("VBoxGuestControlSvc", HOST_EXEC_SET_INPUT, i, paParms); if (RT_FAILURE(vrc)) rc = setErrorHGCM(vrc); } } } if (RT_SUCCESS(vrc)) { LogFlowFunc(("Waiting for HGCM callback ...\n")); /* * Wait for getting back the input response from the guest. */ vrc = callbackWaitForCompletion(uContextID, -1 /* No staging required */, aTimeoutMS); if (RT_SUCCESS(vrc)) { PCALLBACKDATAEXECINSTATUS pExecStatusIn; vrc = callbackGetUserData(uContextID, NULL /* We know the type. */, (void**)&pExecStatusIn, NULL /* Don't need the size. */); if (RT_SUCCESS(vrc)) { AssertPtr(pExecStatusIn); switch (pExecStatusIn->u32Status) { case INPUT_STS_WRITTEN: *aBytesWritten = pExecStatusIn->cbProcessed; break; case INPUT_STS_ERROR: rc = setError(VBOX_E_IPRT_ERROR, tr("Client reported error %Rrc while processing input data"), pExecStatusIn->u32Flags); break; case INPUT_STS_TERMINATED: rc = setError(VBOX_E_IPRT_ERROR, tr("Client terminated while processing input data")); break; case INPUT_STS_OVERFLOW: rc = setError(VBOX_E_IPRT_ERROR, tr("Client reported buffer overflow while processing input data")); break; default: /*AssertReleaseMsgFailed(("Client reported unknown input error, status=%u, flags=%u\n", pExecStatusIn->u32Status, pExecStatusIn->u32Flags));*/ break; } callbackFreeUserData(pExecStatusIn); } else { rc = setErrorNoLog(VBOX_E_IPRT_ERROR, tr("Unable to retrieve process input status data")); } } else rc = setErrorCompletion(vrc); } { AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS); /* The callback isn't needed anymore -- just was kept locally. */ callbackRemove(uContextID); } /* Cleanup. */ if (!pProgress.isNull()) pProgress->uninit(); pProgress.setNull(); } } catch (std::bad_alloc &) { rc = E_OUTOFMEMORY; } return rc; #endif } STDMETHODIMP Guest::GetProcessOutput(ULONG aPID, ULONG aFlags, ULONG aTimeoutMS, LONG64 aSize, ComSafeArrayOut(BYTE, aData)) { #ifndef VBOX_WITH_GUEST_CONTROL ReturnComNotImplemented(); #else /* VBOX_WITH_GUEST_CONTROL */ using namespace guestControl; return getProcessOutputInternal(aPID, aFlags, aTimeoutMS, aSize, ComSafeArrayOutArg(aData), NULL /* rc */); #endif } HRESULT Guest::getProcessOutputInternal(ULONG aPID, ULONG aFlags, ULONG aTimeoutMS, LONG64 aSize, ComSafeArrayOut(BYTE, aData), int *pRC) { /** @todo r=bird: Eventually we should clean up all the timeout parameters * in the API and have the same way of specifying infinite waits! */ #ifndef VBOX_WITH_GUEST_CONTROL ReturnComNotImplemented(); #else /* VBOX_WITH_GUEST_CONTROL */ using namespace guestControl; CheckComArgExpr(aPID, aPID > 0); if (aSize < 0) return setError(E_INVALIDARG, tr("The size argument (%lld) is negative"), aSize); if (aSize == 0) return setError(E_INVALIDARG, tr("The size (%lld) is zero"), aSize); if (aFlags) { if (!(aFlags & ProcessOutputFlag_StdErr)) { return setError(E_INVALIDARG, tr("Unknown flags (%#x)"), aFlags); } } AutoCaller autoCaller(this); if (FAILED(autoCaller.rc())) return autoCaller.rc(); HRESULT rc = S_OK; try { VBOXGUESTCTRL_PROCESS proc; int vrc = processGetStatus(aPID, &proc, false /* Don't remove */); if (RT_FAILURE(vrc)) { rc = setError(VBOX_E_IPRT_ERROR, Guest::tr("Guest process (PID %u) does not exist"), aPID); } else if (proc.mStatus != ExecuteProcessStatus_Started) { /* If the process is already or still in the process table but does not run yet * (or anymore) don't remove it but report back an appropriate error. */ vrc = VERR_BROKEN_PIPE; /* Not getting any output is fine, so don't report an API error (rc) * and only signal something through internal error code (vrc). */ } if (RT_SUCCESS(vrc)) { uint32_t uContextID = 0; /* * Create progress object. * This progress object, compared to the one in executeProgress() above, * is only single-stage local and is used to determine whether the operation * finished or got canceled. */ ComObjPtr pProgress; rc = pProgress.createObject(); if (SUCCEEDED(rc)) { rc = pProgress->init(static_cast(this), Bstr(tr("Getting output for guest process")).raw(), TRUE /* Cancelable */); } if (FAILED(rc)) throw rc; ComAssert(!pProgress.isNull()); /* Adjust timeout. */ if (aTimeoutMS == 0) aTimeoutMS = UINT32_MAX; /* Set handle ID. */ uint32_t uHandleID = OUTPUT_HANDLE_ID_STDOUT; /* Default */ if (aFlags & ProcessOutputFlag_StdErr) uHandleID = OUTPUT_HANDLE_ID_STDERR; uint32_t uGuestPID = processGetGuestPID(aPID); Assert(uGuestPID); /** @todo Use a buffer for next iteration if returned data is too big * for current read. * aSize is bogus -- will be ignored atm! */ VBOXGUESTCTRL_CALLBACK callback; vrc = callbackInit(&callback, VBOXGUESTCTRLCALLBACKTYPE_EXEC_OUTPUT, pProgress); if (RT_SUCCESS(vrc)) { PCALLBACKDATAEXECOUT pData = (PCALLBACKDATAEXECOUT)callback.pvData; /* Save PID + output flags for later use. */ pData->u32PID = uGuestPID; pData->u32Flags = aFlags; } if (RT_SUCCESS(vrc)) vrc = callbackAdd(&callback, &uContextID); if (RT_SUCCESS(vrc)) { VBOXHGCMSVCPARM paParms[5]; int i = 0; paParms[i++].setUInt32(uContextID); paParms[i++].setUInt32(uGuestPID); paParms[i++].setUInt32(uHandleID); paParms[i++].setUInt32(0 /* Flags, none set yet */); VMMDev *pVMMDev = NULL; { /* Make sure mParent is valid, so set the read lock while using. * Do not keep this lock while doing the actual call, because in the meanwhile * another thread could request a write lock which would be a bad idea ... */ AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS); /* Forward the information to the VMM device. */ AssertPtr(mParent); pVMMDev = mParent->getVMMDev(); } if (pVMMDev) { LogFlowFunc(("hgcmHostCall numParms=%d\n", i)); vrc = pVMMDev->hgcmHostCall("VBoxGuestControlSvc", HOST_EXEC_GET_OUTPUT, i, paParms); } } if (RT_SUCCESS(vrc)) { LogFlowFunc(("Waiting for HGCM callback (timeout=%RI32ms) ...\n", aTimeoutMS)); /* * Wait for the HGCM low level callback until the process * has been started (or something went wrong). This is necessary to * get the PID. */ /* * Wait for the first output callback notification to arrive. */ vrc = callbackWaitForCompletion(uContextID, -1 /* No staging required */, aTimeoutMS); if (RT_SUCCESS(vrc)) { PCALLBACKDATAEXECOUT pExecOut = NULL; vrc = callbackGetUserData(uContextID, NULL /* We know the type. */, (void**)&pExecOut, NULL /* Don't need the size. */); if (RT_SUCCESS(vrc)) { com::SafeArray outputData((size_t)aSize); if (pExecOut->cbData) { bool fResize; /* Do we need to resize the array? */ if (pExecOut->cbData > aSize) { fResize = outputData.resize(pExecOut->cbData); Assert(fResize); } /* Fill output in supplied out buffer. */ Assert(outputData.size() >= pExecOut->cbData); memcpy(outputData.raw(), pExecOut->pvData, pExecOut->cbData); fResize = outputData.resize(pExecOut->cbData); /* Shrink to fit actual buffer size. */ Assert(fResize); } else { /* No data within specified timeout available. */ bool fResize = outputData.resize(0); Assert(fResize); } /* Detach output buffer to output argument. */ outputData.detachTo(ComSafeArrayOutArg(aData)); callbackFreeUserData(pExecOut); } else { rc = setErrorNoLog(VBOX_E_IPRT_ERROR, tr("Unable to retrieve process output data (%Rrc)"), vrc); } } else rc = setErrorCompletion(vrc); } else rc = setErrorHGCM(vrc); { AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS); /* The callback isn't needed anymore -- just was kept locally. */ callbackRemove(uContextID); } /* Cleanup. */ if (!pProgress.isNull()) pProgress->uninit(); pProgress.setNull(); } if (pRC) *pRC = vrc; } catch (std::bad_alloc &) { rc = E_OUTOFMEMORY; if (pRC) *pRC = VERR_NO_MEMORY; } return rc; #endif } STDMETHODIMP Guest::GetProcessStatus(ULONG aPID, ULONG *aExitCode, ULONG *aFlags, ExecuteProcessStatus_T *aStatus) { #ifndef VBOX_WITH_GUEST_CONTROL ReturnComNotImplemented(); #else /* VBOX_WITH_GUEST_CONTROL */ AutoCaller autoCaller(this); if (FAILED(autoCaller.rc())) return autoCaller.rc(); HRESULT rc = S_OK; try { VBOXGUESTCTRL_PROCESS process; int vrc = processGetStatus(aPID, &process, true /* Remove when terminated */); if (RT_SUCCESS(vrc)) { if (aExitCode) *aExitCode = process.mExitCode; if (aFlags) *aFlags = process.mFlags; if (aStatus) *aStatus = process.mStatus; } else rc = setError(VBOX_E_IPRT_ERROR, tr("Process (PID %u) not found!"), aPID); } catch (std::bad_alloc &) { rc = E_OUTOFMEMORY; } return rc; #endif } STDMETHODIMP Guest::CopyFromGuest(IN_BSTR aSource, IN_BSTR aDest, IN_BSTR aUsername, IN_BSTR aPassword, ULONG aFlags, IProgress **aProgress) { #ifndef VBOX_WITH_GUEST_CONTROL ReturnComNotImplemented(); #else /* VBOX_WITH_GUEST_CONTROL */ CheckComArgStrNotEmptyOrNull(aSource); CheckComArgStrNotEmptyOrNull(aDest); CheckComArgOutPointerValid(aProgress); /* Do not allow anonymous executions (with system rights). */ if (RT_UNLIKELY((aUsername) == NULL || *(aUsername) == '\0')) return setError(E_INVALIDARG, tr("No user name specified")); AutoCaller autoCaller(this); if (FAILED(autoCaller.rc())) return autoCaller.rc(); /* Validate flags. */ if (aFlags != CopyFileFlag_None) { if ( !(aFlags & CopyFileFlag_Recursive) && !(aFlags & CopyFileFlag_Update) && !(aFlags & CopyFileFlag_FollowLinks)) { return setError(E_INVALIDARG, tr("Unknown flags (%#x)"), aFlags); } } AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS); HRESULT rc = S_OK; ComObjPtr progress; try { /* Create the progress object. */ progress.createObject(); rc = progress->init(static_cast(this), Bstr(tr("Copying file from guest to host")).raw(), TRUE /* aCancelable */); if (FAILED(rc)) throw rc; /* Initialize our worker task. */ GuestTask *pTask = new GuestTask(GuestTask::TaskType_CopyFileFromGuest, this, progress); AssertPtr(pTask); std::auto_ptr task(pTask); /* Assign data - aSource is the source file on the host, * aDest reflects the full path on the guest. */ task->strSource = (Utf8Str(aSource)); task->strDest = (Utf8Str(aDest)); task->strUserName = (Utf8Str(aUsername)); task->strPassword = (Utf8Str(aPassword)); task->uFlags = aFlags; rc = task->startThread(); if (FAILED(rc)) throw rc; /* Don't destruct on success. */ task.release(); } catch (HRESULT aRC) { rc = aRC; } if (SUCCEEDED(rc)) { /* Return progress to the caller. */ progress.queryInterfaceTo(aProgress); } return rc; #endif /* VBOX_WITH_GUEST_CONTROL */ } STDMETHODIMP Guest::CopyToGuest(IN_BSTR aSource, IN_BSTR aDest, IN_BSTR aUsername, IN_BSTR aPassword, ULONG aFlags, IProgress **aProgress) { #ifndef VBOX_WITH_GUEST_CONTROL ReturnComNotImplemented(); #else /* VBOX_WITH_GUEST_CONTROL */ CheckComArgStrNotEmptyOrNull(aSource); CheckComArgStrNotEmptyOrNull(aDest); CheckComArgOutPointerValid(aProgress); /* Do not allow anonymous executions (with system rights). */ if (RT_UNLIKELY((aUsername) == NULL || *(aUsername) == '\0')) return setError(E_INVALIDARG, tr("No user name specified")); AutoCaller autoCaller(this); if (FAILED(autoCaller.rc())) return autoCaller.rc(); /* Validate flags. */ if (aFlags != CopyFileFlag_None) { if ( !(aFlags & CopyFileFlag_Recursive) && !(aFlags & CopyFileFlag_Update) && !(aFlags & CopyFileFlag_FollowLinks)) { return setError(E_INVALIDARG, tr("Unknown flags (%#x)"), aFlags); } } AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS); HRESULT rc = S_OK; ComObjPtr pProgress; try { /* Create the progress object. */ pProgress.createObject(); rc = pProgress->init(static_cast(this), Bstr(tr("Copying file from host to guest")).raw(), TRUE /* aCancelable */); if (FAILED(rc)) throw rc; /* Initialize our worker task. */ GuestTask *pTask = new GuestTask(GuestTask::TaskType_CopyFileToGuest, this, pProgress); AssertPtr(pTask); std::auto_ptr task(pTask); /* Assign data - aSource is the source file on the host, * aDest reflects the full path on the guest. */ task->strSource = (Utf8Str(aSource)); task->strDest = (Utf8Str(aDest)); task->strUserName = (Utf8Str(aUsername)); task->strPassword = (Utf8Str(aPassword)); task->uFlags = aFlags; rc = task->startThread(); if (FAILED(rc)) throw rc; /* Don't destruct on success. */ task.release(); } catch (HRESULT aRC) { rc = aRC; } if (SUCCEEDED(rc)) { /* Return progress to the caller. */ pProgress.queryInterfaceTo(aProgress); } return rc; #endif /* VBOX_WITH_GUEST_CONTROL */ } STDMETHODIMP Guest::UpdateGuestAdditions(IN_BSTR aSource, ComSafeArrayIn(AdditionsUpdateFlag_T, aFlags), IProgress **aProgress) { #ifndef VBOX_WITH_GUEST_CONTROL ReturnComNotImplemented(); #else /* VBOX_WITH_GUEST_CONTROL */ CheckComArgStrNotEmptyOrNull(aSource); CheckComArgOutPointerValid(aProgress); AutoCaller autoCaller(this); if (FAILED(autoCaller.rc())) return autoCaller.rc(); /* Validate flags. */ uint32_t fFlags = AdditionsUpdateFlag_None; if (aFlags) { com::SafeArray flags(ComSafeArrayInArg(aFlags)); for (size_t i = 0; i < flags.size(); i++) fFlags |= flags[i]; } if (fFlags) { if (!(fFlags & AdditionsUpdateFlag_WaitForUpdateStartOnly)) return setError(E_INVALIDARG, tr("Unknown flags (%#x)"), aFlags); } HRESULT hr = S_OK; #if 1 /* Create an anonymous session. This is required to run the Guest Additions * update process with administrative rights. */ ComObjPtr pSession; int rc = sessionCreate("" /* User */, "" /* Password */, "" /* Domain */, "Updating Guest Additions" /* Name */, pSession); if (RT_FAILURE(rc)) { switch (rc) { case VERR_MAX_PROCS_REACHED: hr = setError(VBOX_E_IPRT_ERROR, tr("Maximum number of guest sessions (%ld) reached"), VBOX_GUESTCTRL_MAX_SESSIONS); break; /** @todo Add more errors here. */ default: hr = setError(VBOX_E_IPRT_ERROR, tr("Could not create guest session: %Rrc"), rc); break; } } else { Assert(!pSession.isNull()); rc = pSession->queryInfo(); if (RT_FAILURE(rc)) { hr = setError(VBOX_E_IPRT_ERROR, tr("Could not query guest session information: %Rrc"), rc); } else { ComObjPtr pProgress; SessionTaskUpdateAdditions *pTask = new SessionTaskUpdateAdditions(pSession /* GuestSession */, Utf8Str(aSource), fFlags); AssertPtrReturn(pTask, VERR_NO_MEMORY); rc = pSession->startTaskAsync(tr("Updating Guest Additions"), pTask, pProgress); if (RT_SUCCESS(rc)) { /* Return progress to the caller. */ hr = pProgress.queryInterfaceTo(aProgress); } else hr = setError(VBOX_E_IPRT_ERROR, tr("Starting task for updating Guest Additions on the guest failed: %Rrc"), rc); } } return hr; #else /* Legacy, can be removed later. */ ComObjPtr progress; try { /* Create the progress object. */ progress.createObject(); rc = progress->init(static_cast(this), Bstr(tr("Updating Guest Additions")).raw(), TRUE /* aCancelable */); if (FAILED(rc)) throw rc; /* Initialize our worker task. */ GuestTask *pTask = new GuestTask(GuestTask::TaskType_UpdateGuestAdditions, this, progress); AssertPtr(pTask); std::auto_ptr task(pTask); /* Assign data - in that case aSource is the full path * to the Guest Additions .ISO we want to mount. */ task->strSource = (Utf8Str(aSource)); task->uFlags = aFlags; rc = task->startThread(); if (FAILED(rc)) throw rc; /* Don't destruct on success. */ task.release(); } catch (HRESULT aRC) { rc = aRC; } if (SUCCEEDED(rc)) { /* Return progress to the caller. */ progress.queryInterfaceTo(aProgress); } return rc; #endif #endif /* VBOX_WITH_GUEST_CONTROL */ } // private methods ///////////////////////////////////////////////////////////////////////////// int Guest::dispatchToSession(uint32_t uContextID, uint32_t uFunction, void *pvData, size_t cbData) { LogFlowFuncEnter(); AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS); uint32_t uSessionID = VBOX_GUESTCTRL_CONTEXTID_GET_SESSION(uContextID); #ifdef DEBUG LogFlowFunc(("uSessionID=%RU32 (%RU32 total)\n", uSessionID, mData.mGuestSessions.size())); #endif int rc; GuestSessions::const_iterator itSession = mData.mGuestSessions.find(uSessionID); if (itSession != mData.mGuestSessions.end()) { ComObjPtr pSession(itSession->second); Assert(!pSession.isNull()); alock.release(); rc = pSession->dispatchToProcess(uContextID, uFunction, pvData, cbData); } else rc = VERR_NOT_FOUND; LogFlowFuncLeaveRC(rc); return rc; } int Guest::sessionClose(ComObjPtr pSession) { AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS); for (GuestSessions::iterator itSessions = mData.mGuestSessions.begin(); itSessions != mData.mGuestSessions.end(); ++itSessions) { if (pSession == itSessions->second) { mData.mGuestSessions.erase(itSessions); return VINF_SUCCESS; } } return VERR_NOT_FOUND; } int Guest::sessionCreate(const Utf8Str &strUser, const Utf8Str &strPassword, const Utf8Str &strDomain, const Utf8Str &strSessionName, ComObjPtr &pGuestSession) { AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS); int rc = VERR_MAX_PROCS_REACHED; if (mData.mGuestSessions.size() >= VBOX_GUESTCTRL_MAX_SESSIONS) return rc; try { /* Create a new session ID and assign it. */ uint32_t uNewSessionID = 0; uint32_t uTries = 0; for (;;) { /* Is the context ID already used? */ if (!sessionExists(uNewSessionID)) { rc = VINF_SUCCESS; break; } uNewSessionID++; if (uNewSessionID >= VBOX_GUESTCTRL_MAX_SESSIONS) uNewSessionID = 0; if (++uTries == UINT32_MAX) break; /* Don't try too hard. */ } if (RT_FAILURE(rc)) throw rc; /* Create the session object. */ HRESULT hr = pGuestSession.createObject(); if (FAILED(hr)) throw VERR_COM_UNEXPECTED; rc = pGuestSession->init(this, uNewSessionID, strUser, strPassword, strDomain, strSessionName); if (RT_FAILURE(rc)) throw rc; mData.mGuestSessions[uNewSessionID] = pGuestSession; LogFlowFunc(("Added new session with session ID=%RU32 (now %ld sessions total)\n", uNewSessionID, mData.mGuestSessions.size())); } catch (int rc2) { rc = rc2; } return rc; } inline bool Guest::sessionExists(uint32_t uSessionID) { GuestSessions::const_iterator itSessions = mData.mGuestSessions.find(uSessionID); return (itSessions == mData.mGuestSessions.end()) ? false : true; } // implementation of public methods ///////////////////////////////////////////////////////////////////////////// STDMETHODIMP Guest::CreateSession(IN_BSTR aUser, IN_BSTR aPassword, IN_BSTR aDomain, IN_BSTR aSessionName, IGuestSession **aGuestSession) { #ifndef VBOX_WITH_GUEST_CONTROL ReturnComNotImplemented(); #else /* VBOX_WITH_GUEST_CONTROL */ LogFlowFuncEnter(); /* Do not allow anonymous sessions (with system rights) with official API. */ if (RT_UNLIKELY((aUser) == NULL || *(aUser) == '\0')) return setError(E_INVALIDARG, tr("No user name specified")); CheckComArgOutPointerValid(aGuestSession); /* Rest is optional. */ AutoCaller autoCaller(this); if (FAILED(autoCaller.rc())) return autoCaller.rc(); HRESULT hr = S_OK; ComObjPtr pSession; int rc = sessionCreate(aUser, aPassword, aDomain, aSessionName, pSession); if (RT_SUCCESS(rc)) { /* Return guest session to the caller. */ HRESULT hr2 = pSession.queryInterfaceTo(aGuestSession); if (FAILED(hr2)) rc = VERR_COM_OBJECT_NOT_FOUND; if (RT_SUCCESS(rc)) rc = pSession->queryInfo(); } if (RT_FAILURE(rc)) { switch (rc) { case VERR_MAX_PROCS_REACHED: hr = setError(VBOX_E_IPRT_ERROR, tr("Maximum number of guest sessions (%ld) reached"), VBOX_GUESTCTRL_MAX_SESSIONS); break; /** @todo Add more errors here. */ default: hr = setError(VBOX_E_IPRT_ERROR, tr("Could not create guest session, rc=%Rrc"), rc); break; } } LogFlowFuncLeaveRC(rc); return hr; #endif /* VBOX_WITH_GUEST_CONTROL */ } STDMETHODIMP Guest::FindSession(IN_BSTR aSessionName, ComSafeArrayOut(IGuestSession *, aSessions)) { #ifndef VBOX_WITH_GUEST_CONTROL ReturnComNotImplemented(); #else /* VBOX_WITH_GUEST_CONTROL */ CheckComArgOutSafeArrayPointerValid(aSessions); LogFlowFuncEnter(); AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS); Utf8Str strName(aSessionName); std::list < ComObjPtr > listSessions; GuestSessions::const_iterator itSessions = mData.mGuestSessions.begin(); while (itSessions != mData.mGuestSessions.end()) { if (strName.contains(itSessions->second->getName())) /** @todo Use a (simple) pattern match (IPRT?). */ listSessions.push_back(itSessions->second); itSessions++; } LogFlowFunc(("Sessions with \"%ls\" = %RU32\n", aSessionName, listSessions.size())); if (listSessions.size()) { SafeIfaceArray sessionIfacs(listSessions); sessionIfacs.detachTo(ComSafeArrayOutArg(aSessions)); return S_OK; } return setErrorNoLog(VBOX_E_OBJECT_NOT_FOUND, tr("Could not find sessions with name '%ls'"), aSessionName); #endif /* VBOX_WITH_GUEST_CONTROL */ }