/** @file * * Internal helpers/structures for guest control functionality. */ /* * Copyright (C) 2011-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. */ #ifndef ____H_GUESTIMPLPRIVATE #define ____H_GUESTIMPLPRIVATE #include #include #include #include #include #include #include using namespace com; #ifdef VBOX_WITH_GUEST_CONTROL # include using namespace guestControl; #endif /* Builds a context ID out of the session ID, process ID and an * increasing count. */ #define VBOX_GUESTCTRL_CONTEXTID_MAKE(uSession, uProcess, uCount) \ ( (uint32_t)((uSession) & 0xff) << 24 \ | (uint32_t)((uProcess) & 0xff) << 16 \ | (uint32_t)((uCount) & 0xffff) \ ) typedef std::vector ProcessAffinity; typedef std::vector ProcessArguments; /** * Generic class for a all guest control callbacks. */ class GuestCtrlCallback { public: GuestCtrlCallback(void); GuestCtrlCallback(eVBoxGuestCtrlCallbackType enmType); virtual ~GuestCtrlCallback(void); /** @todo Copy/comparison operator? */ public: int Init(eVBoxGuestCtrlCallbackType enmType); void Destroy(void); eVBoxGuestCtrlCallbackType Type(void); int Wait(RTMSINTERVAL timeoutMS); protected: /** The callback type. */ eVBoxGuestCtrlCallbackType mType; /** Callback flags. */ uint32_t mFlags; /** Pointer to user-supplied data. */ void *pvData; /** Size of user-supplied data. */ size_t cbData; /** The event semaphore triggering the*/ RTSEMEVENT mEventSem; /** Extended error information, if any. */ ErrorInfo mErrorInfo; }; typedef std::map GuestCtrlCallbacks; /** * Simple structure mantaining guest credentials. */ class GuestCredentials { public: public: Utf8Str mUser; Utf8Str mPassword; Utf8Str mDomain; }; typedef std::vector GuestEnvironmentArray; class GuestEnvironment { public: int BuildEnvironmentBlock(void **ppvEnv, uint32_t *pcbEnv, uint32_t *pcEnvVars); void Clear(void); int CopyFrom(const GuestEnvironmentArray &environment); int CopyTo(GuestEnvironmentArray &environment); static void FreeEnvironmentBlock(void *pvEnv); Utf8Str Get(const Utf8Str &strKey); Utf8Str Get(size_t nPos); bool Has(const Utf8Str &strKey); int Set(const Utf8Str &strKey, const Utf8Str &strValue); int Set(const Utf8Str &strPair); size_t Size(void); int Unset(const Utf8Str &strKey); public: GuestEnvironment& operator=(const GuestEnvironmentArray &that); GuestEnvironment& operator=(const GuestEnvironment &that); protected: int appendToEnvBlock(const char *pszEnv, void **ppvList, uint32_t *pcbList, uint32_t *pcEnvVars); protected: std::map mEnvironment; }; /** * Structure for keeping all the relevant process * starting parameters around. */ struct GuestProcessInfo { Utf8Str mCommand; ProcessArguments mArguments; GuestEnvironment mEnvironment; uint32_t mFlags; ULONG mTimeoutMS; ProcessPriority_T mPriority; ProcessAffinity mAffinity; }; /** * Class representing the "value" side of a "key=value" pair. */ class GuestProcessStreamValue { public: GuestProcessStreamValue() { } GuestProcessStreamValue(const char *pszValue) : mValue(pszValue) {} GuestProcessStreamValue(const GuestProcessStreamValue& aThat) : mValue(aThat.mValue) {} Utf8Str mValue; }; /** Map containing "key=value" pairs of a guest process stream. */ typedef std::pair< Utf8Str, GuestProcessStreamValue > GuestCtrlStreamPair; typedef std::map < Utf8Str, GuestProcessStreamValue > GuestCtrlStreamPairMap; typedef std::map < Utf8Str, GuestProcessStreamValue >::iterator GuestCtrlStreamPairMapIter; typedef std::map < Utf8Str, GuestProcessStreamValue >::const_iterator GuestCtrlStreamPairMapIterConst; /** * Class representing a block of stream pairs (key=value). Each block in a raw guest * output stream is separated by "\0\0", each pair is separated by "\0". The overall * end of a guest stream is marked by "\0\0\0\0". */ class GuestProcessStreamBlock { public: GuestProcessStreamBlock(); //GuestProcessStreamBlock(GuestProcessStreamBlock &); virtual ~GuestProcessStreamBlock(); public: void Clear(); #ifdef DEBUG void Dump(); #endif int GetInt64Ex(const char *pszKey, int64_t *piVal); int64_t GetInt64(const char *pszKey); size_t GetCount(); const char* GetString(const char *pszKey); int GetUInt32Ex(const char *pszKey, uint32_t *puVal); uint32_t GetUInt32(const char *pszKey); int SetValue(const char *pszKey, const char *pszValue); protected: GuestCtrlStreamPairMap m_mapPairs; }; /** Vector containing multiple allocated stream pair objects. */ typedef std::vector< GuestProcessStreamBlock > GuestCtrlStreamObjects; typedef std::vector< GuestProcessStreamBlock >::iterator GuestCtrlStreamObjectsIter; typedef std::vector< GuestProcessStreamBlock >::const_iterator GuestCtrlStreamObjectsIterConst; /** * Class for parsing machine-readable guest process output by VBoxService' * toolbox commands ("vbox_ls", "vbox_stat" etc), aka "guest stream". */ class GuestProcessStream { public: GuestProcessStream(); virtual ~GuestProcessStream(); public: int AddData(const BYTE *pbData, size_t cbData); void Destroy(); #ifdef DEBUG void Dump(const char *pszFile); #endif uint32_t GetOffset(); uint32_t GetSize(); int ParseBlock(GuestProcessStreamBlock &streamBlock); protected: /** Currently allocated size of internal stream buffer. */ uint32_t m_cbAllocated; /** Currently used size of allocated internal stream buffer. */ uint32_t m_cbSize; /** Current offset within the internal stream buffer. */ uint32_t m_cbOffset; /** Internal stream buffer. */ BYTE *m_pbBuffer; }; class Guest; class Progress; class GuestTask { public: enum TaskType { /** Copies a file from host to the guest. */ TaskType_CopyFileToGuest = 50, /** Copies a file from guest to the host. */ TaskType_CopyFileFromGuest = 55, /** Update Guest Additions by directly copying the required installer * off the .ISO file, transfer it to the guest and execute the installer * with system privileges. */ TaskType_UpdateGuestAdditions = 100 }; GuestTask(TaskType aTaskType, Guest *aThat, Progress *aProgress); virtual ~GuestTask(); int startThread(); static int taskThread(RTTHREAD aThread, void *pvUser); static int uploadProgress(unsigned uPercent, void *pvUser); static HRESULT setProgressSuccess(ComObjPtr pProgress); static HRESULT setProgressErrorMsg(HRESULT hr, ComObjPtr pProgress, const char * pszText, ...); static HRESULT setProgressErrorParent(HRESULT hr, ComObjPtr pProgress, ComObjPtr pGuest); TaskType taskType; ComObjPtr pGuest; ComObjPtr pProgress; HRESULT rc; /* Task data. */ Utf8Str strSource; Utf8Str strDest; Utf8Str strUserName; Utf8Str strPassword; ULONG uFlags; }; #endif // ____H_GUESTIMPLPRIVATE