/** @file * * Guest Property Service: * Host service entry points. */ /* * Copyright (C) 2008 Sun Microsystems, Inc. * * 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. * * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa * Clara, CA 95054 USA or visit http://www.sun.com if you need * additional information or have any questions. */ /** * This HGCM service allows the guest to set and query values in a property * store on the host. The service proxies the guest requests to the service * owner on the host using a request callback provided by the owner, and is * notified of changes to properties made by the host. It forwards these * notifications to clients in the guest which have expressed interest and * are waiting for notification. * * The service currently consists of two threads. One of these is the main * HGCM service thread which waits for requests from the guest and schedules * these to the second thread. The second thread deals with the requests * sequentially by calling the callback provided by the service owner, * notifying the guest clients when it has finished dealing with a given * request. * * Guest requests to wait for notification are dealt with differently. They * are added to a list of open notification requests but do not schedule * anything in the request thread except for a possible timeout. */ #define LOG_GROUP LOG_GROUP_HGCM /******************************************************************************* * Header Files * *******************************************************************************/ #include #include /* for auto_ptr */ #include #include #include #include #include #include #include #include #include /******************************************************************************* * Internal functions * *******************************************************************************/ /** Extract a pointer value from an HGCM parameter structure */ static int VBoxHGCMParmPtrGet (VBOXHGCMSVCPARM *pParm, void **ppv, uint32_t *pcb) { if (pParm->type == VBOX_HGCM_SVC_PARM_PTR) { *ppv = pParm->u.pointer.addr; *pcb = pParm->u.pointer.size; return VINF_SUCCESS; } return VERR_INVALID_PARAMETER; } /** Set a uint32_t value to an HGCM parameter structure */ static void VBoxHGCMParmUInt32Set (VBOXHGCMSVCPARM *pParm, uint32_t u32) { pParm->type = VBOX_HGCM_SVC_PARM_32BIT; pParm->u.uint32 = u32; } /** Set a uint64_t value to an HGCM parameter structure */ static void VBoxHGCMParmUInt64Set (VBOXHGCMSVCPARM *pParm, uint64_t u64) { pParm->type = VBOX_HGCM_SVC_PARM_64BIT; pParm->u.uint64 = u64; } namespace guestProp { /** * Class containing the shared information service functionality. */ class Service : public stdx::non_copyable { private: /** Type definition for use in callback functions */ typedef Service SELF; /** HGCM helper functions. */ PVBOXHGCMSVCHELPERS mpHelpers; /** Pointer to our configuration values node. */ PCFGMNODE mpValueNode; /** Pointer to our configuration timestamps node. */ PCFGMNODE mpTimestampNode; /** Pointer to our configuration flags node. */ PCFGMNODE mpFlagsNode; public: explicit Service(PVBOXHGCMSVCHELPERS pHelpers) : mpHelpers(pHelpers), mpValueNode(NULL), mpTimestampNode(NULL), mpFlagsNode(NULL) {} /** * @copydoc VBOXHGCMSVCHELPERS::pfnUnload * Simply deletes the service object */ static DECLCALLBACK(int) svcUnload (void *pvService) { AssertLogRelReturn(VALID_PTR(pvService), VERR_INVALID_PARAMETER); SELF *pSelf = reinterpret_cast(pvService); delete pSelf; return VINF_SUCCESS; } /** * @copydoc VBOXHGCMSVCHELPERS::pfnConnect * Stub implementation of pfnConnect and pfnDisconnect. */ static DECLCALLBACK(int) svcConnectDisconnect (void * /* pvService */, uint32_t /* u32ClientID */, void * /* pvClient */) { return VINF_SUCCESS; } /** * @copydoc VBOXHGCMSVCHELPERS::pfnCall * Wraps to the call member function */ static DECLCALLBACK(void) svcCall (void * pvService, VBOXHGCMCALLHANDLE callHandle, uint32_t u32ClientID, void *pvClient, uint32_t u32Function, uint32_t cParms, VBOXHGCMSVCPARM paParms[]) { AssertLogRelReturnVoid(VALID_PTR(pvService)); SELF *pSelf = reinterpret_cast(pvService); pSelf->call(callHandle, u32ClientID, pvClient, u32Function, cParms, paParms); } /** * @copydoc VBOXHGCMSVCHELPERS::pfnHostCall * Wraps to the hostCall member function */ static DECLCALLBACK(int) svcHostCall (void *pvService, uint32_t u32Function, uint32_t cParms, VBOXHGCMSVCPARM paParms[]) { AssertLogRelReturn(VALID_PTR(pvService), VERR_INVALID_PARAMETER); SELF *pSelf = reinterpret_cast(pvService); return pSelf->hostCall(u32Function, cParms, paParms); } private: int validateKey(const char *pszKey, uint32_t cbKey); int validateValue(char *pszValue, uint32_t cbValue); int getKey(uint32_t cParms, VBOXHGCMSVCPARM paParms[]); int getProperty(uint32_t cParms, VBOXHGCMSVCPARM paParms[]); int setKey(uint32_t cParms, VBOXHGCMSVCPARM paParms[]); int delKey(uint32_t cParms, VBOXHGCMSVCPARM paParms[]); int enumProps(uint32_t cParms, VBOXHGCMSVCPARM paParms[]); void call (VBOXHGCMCALLHANDLE callHandle, uint32_t u32ClientID, void *pvClient, uint32_t eFunction, uint32_t cParms, VBOXHGCMSVCPARM paParms[]); int hostCall (uint32_t eFunction, uint32_t cParms, VBOXHGCMSVCPARM paParms[]); }; /** * Checking that the key passed by the guest fits our criteria for a * configuration key. * * @returns IPRT status code * @param pszKey the key passed by the guest * @param cbKey the number of bytes pszKey points to, including the * terminating '\0' * @thread HGCM */ int Service::validateKey(const char *pszKey, uint32_t cbKey) { LogFlowFunc(("cbKey=%d\n", cbKey)); /* * Validate the key, checking that it's proper UTF-8 and has * a string terminator. */ int rc = RTStrValidateEncodingEx(pszKey, RT_MIN(cbKey, MAX_NAME_LEN), RTSTR_VALIDATE_ENCODING_ZERO_TERMINATED); if (RT_SUCCESS(rc) && (cbKey < 2)) rc = VERR_INVALID_PARAMETER; LogFlowFunc(("returning %Rrc\n", rc)); return rc; } /** * Check that the data passed by the guest fits our criteria for the value of * a configuration key. * * @returns IPRT status code * @param pszValue the value to store in the key * @param cbValue the number of bytes in the buffer pszValue points to * @thread HGCM */ int Service::validateValue(char *pszValue, uint32_t cbValue) { LogFlowFunc(("cbValue=%d\n", cbValue)); /* * Validate the value, checking that it's proper UTF-8 and has * a string terminator. Don't pass a 0 length request to the * validator since it won't find any '\0' then. */ int rc = VINF_SUCCESS; if (cbValue) rc = RTStrValidateEncodingEx(pszValue, RT_MIN(cbValue, MAX_VALUE_LEN), RTSTR_VALIDATE_ENCODING_ZERO_TERMINATED); if (RT_SUCCESS(rc)) LogFlow((" pszValue=%s\n", cbValue > 0 ? pszValue : NULL)); LogFlowFunc(("returning %Rrc\n", rc)); return rc; } /** * Retrieve a value from the guest registry by key, checking the validity * of the arguments passed. If the guest has not allocated enough buffer * space for the value then we return VERR_OVERFLOW and set the size of the * buffer needed in the "size" HGCM parameter. If the key was not found at * all, we return VERR_NOT_FOUND. * * @returns iprt status value * @param cParms the number of HGCM parameters supplied * @param paParms the array of HGCM parameters * @thread HGCM */ int Service::getKey(uint32_t cParms, VBOXHGCMSVCPARM paParms[]) { int rc = VINF_SUCCESS; char *pszKey, *pszValue; uint32_t cbKey, cbValue; size_t cbValueActual; LogFlowThisFunc(("\n")); if ( (cParms != 3) /* Hardcoded value as the next lines depend on it. */ || (paParms[0].type != VBOX_HGCM_SVC_PARM_PTR) /* key */ || (paParms[1].type != VBOX_HGCM_SVC_PARM_PTR) /* value */ ) rc = VERR_INVALID_PARAMETER; if (RT_SUCCESS(rc)) rc = VBoxHGCMParmPtrGet(&paParms[0], (void **) &pszKey, &cbKey); if (RT_SUCCESS(rc)) rc = VBoxHGCMParmPtrGet(&paParms[1], (void **) &pszValue, &cbValue); if (RT_SUCCESS(rc)) rc = validateKey(pszKey, cbKey); if (RT_SUCCESS(rc)) rc = CFGMR3QuerySize(mpValueNode, pszKey, &cbValueActual); if (RT_SUCCESS(rc)) VBoxHGCMParmUInt32Set(&paParms[2], cbValueActual); if (RT_SUCCESS(rc) && (cbValueActual > cbValue)) rc = VERR_BUFFER_OVERFLOW; if (RT_SUCCESS(rc)) rc = CFGMR3QueryString(mpValueNode, pszKey, pszValue, cbValue); if (RT_SUCCESS(rc)) Log2(("Queried string %s, rc=%Rrc, value=%.*s\n", pszKey, rc, cbValue, pszValue)); else if (VERR_CFGM_VALUE_NOT_FOUND == rc) rc = VERR_NOT_FOUND; LogFlowThisFunc(("rc = %Rrc\n", rc)); return rc; } /** * Retrieve a value from the guest registry by key, checking the validity * of the arguments passed. If the guest has not allocated enough buffer * space for the value then we return VERR_OVERFLOW and set the size of the * buffer needed in the "size" HGCM parameter. If the key was not found at * all, we return VERR_NOT_FOUND. * * @returns iprt status value * @param cParms the number of HGCM parameters supplied * @param paParms the array of HGCM parameters * @thread HGCM */ int Service::getProperty(uint32_t cParms, VBOXHGCMSVCPARM paParms[]) { int rc = VINF_SUCCESS; char *pszName, *pchBuf; uint32_t cchName, cchBuf; size_t cchValue, cchFlags, cchBufActual; /* * Get and validate the parameters */ LogFlowThisFunc(("\n")); if ( (cParms != 4) /* Hardcoded value as the next lines depend on it. */ || (paParms[0].type != VBOX_HGCM_SVC_PARM_PTR) /* name */ || (paParms[1].type != VBOX_HGCM_SVC_PARM_PTR) /* buffer */ ) rc = VERR_INVALID_PARAMETER; if (RT_SUCCESS(rc)) rc = VBoxHGCMParmPtrGet(&paParms[0], (void **) &pszName, &cchName); if (RT_SUCCESS(rc)) rc = VBoxHGCMParmPtrGet(&paParms[1], (void **) &pchBuf, &cchBuf); if (RT_SUCCESS(rc)) rc = validateKey(pszName, cchName); /* Get the value size */ if (RT_SUCCESS(rc)) rc = CFGMR3QuerySize(mpValueNode, pszName, &cchValue); /* * Read and set the values we will return */ /* Get the flags size */ cchFlags = 1; /* Empty string if no flags set. */ if (RT_SUCCESS(rc) && (mpFlagsNode != NULL)) CFGMR3QuerySize(mpFlagsNode, pszName, &cchFlags); /* Ignore failure. */ /* Check that the buffer is big enough */ if (RT_SUCCESS(rc)) { cchBufActual = cchValue + cchFlags; VBoxHGCMParmUInt32Set(&paParms[3], cchBufActual); } if (RT_SUCCESS(rc) && (cchBufActual > cchBuf)) rc = VERR_BUFFER_OVERFLOW; /* Write the value */ if (RT_SUCCESS(rc)) rc = CFGMR3QueryString(mpValueNode, pszName, pchBuf, cchBuf); /* Write the flags if there are any */ if (RT_SUCCESS(rc)) pchBuf[cchValue] = '\0'; /* In case there aren't */ if (RT_SUCCESS(rc) && (mpFlagsNode != NULL) && (cchFlags != 1)) CFGMR3QueryString(mpFlagsNode, pszName, pchBuf + cchValue, cchBuf - cchValue); /* Timestamp */ uint64_t u64Timestamp = 0; if (RT_SUCCESS(rc) && (mpTimestampNode != NULL)) CFGMR3QueryU64(mpTimestampNode, pszName, &u64Timestamp); VBoxHGCMParmUInt64Set(&paParms[2], u64Timestamp); /* * Done! Do exit logging and return. */ if (RT_SUCCESS(rc)) Log2(("Queried string %S, value=%.*S, timestamp=%lld, flags=%.*S\n", pszName, cchValue, pchBuf, u64Timestamp, cchFlags, pchBuf + cchValue)); else if (VERR_CFGM_VALUE_NOT_FOUND == rc) rc = VERR_NOT_FOUND; LogFlowThisFunc(("rc = %Rrc\n", rc)); return rc; } /** * Set a value in the guest registry by key, checking the validity * of the arguments passed. * * @returns iprt status value * @param cParms the number of HGCM parameters supplied * @param paParms the array of HGCM parameters * @thread HGCM */ int Service::setKey(uint32_t cParms, VBOXHGCMSVCPARM paParms[]) { int rc = VINF_SUCCESS; char *pszKey, *pszValue; uint32_t cchKey, cchValue; LogFlowThisFunc(("\n")); if ( (cParms < 2) || (cParms > 4) /* Hardcoded value as the next lines depend on it. */ || (paParms[0].type != VBOX_HGCM_SVC_PARM_PTR) /* key */ || (paParms[1].type != VBOX_HGCM_SVC_PARM_PTR) /* value */ || ((3 == cParms) && (paParms[2].type != VBOX_HGCM_SVC_PARM_PTR)) ) rc = VERR_INVALID_PARAMETER; if (RT_SUCCESS(rc)) rc = VBoxHGCMParmPtrGet(&paParms[0], (void **) &pszKey, &cchKey); if (RT_SUCCESS(rc)) rc = VBoxHGCMParmPtrGet(&paParms[1], (void **) &pszValue, &cchValue); if (RT_SUCCESS(rc)) rc = validateKey(pszKey, cchKey); if (RT_SUCCESS(rc)) rc = validateValue(pszValue, cchValue); if (RT_SUCCESS(rc)) { /* Limit the number of keys that we can set. */ unsigned cChildren = 0; for (PCFGMNODE pChild = CFGMR3GetFirstChild(mpValueNode); pChild != 0; pChild = CFGMR3GetNextChild(pChild)) ++cChildren; if (cChildren >= MAX_KEYS) rc = VERR_TOO_MUCH_DATA; } /* For now, we do not support any flags */ if (RT_SUCCESS(rc) && (3 == cParms)) { char *pszFlags; uint32_t cchFlags = 0; rc = VBoxHGCMParmPtrGet(&paParms[2], (void **) &pszFlags, &cchFlags); for (size_t i = 0; (i < cchFlags - 1) && RT_SUCCESS(rc); ++i) if (pszFlags[i] != ' ') rc = VERR_INVALID_PARAMETER; } if (RT_SUCCESS(rc)) { RTTIMESPEC time; CFGMR3RemoveValue(mpValueNode, pszKey); if (mpTimestampNode != NULL) CFGMR3RemoveValue(mpTimestampNode, pszKey); if ((3 == cParms) && (mpFlagsNode != NULL)) CFGMR3RemoveValue(mpFlagsNode, pszKey); rc = CFGMR3InsertString(mpValueNode, pszKey, pszValue); if (RT_SUCCESS(rc)) rc = CFGMR3InsertInteger(mpTimestampNode, pszKey, RTTimeSpecGetNano(RTTimeNow(&time))); } if (RT_SUCCESS(rc)) Log2(("Set string %s, rc=%Rrc, value=%s\n", pszKey, rc, pszValue)); LogFlowThisFunc(("rc = %Rrc\n", rc)); return rc; } /** * Remove a value in the guest registry by key, checking the validity * of the arguments passed. * * @returns iprt status value * @param cParms the number of HGCM parameters supplied * @param paParms the array of HGCM parameters * @thread HGCM */ int Service::delKey(uint32_t cParms, VBOXHGCMSVCPARM paParms[]) { int rc = VINF_SUCCESS; char *pszKey; uint32_t cbKey; LogFlowThisFunc(("\n")); if ( (cParms != 1) /* Hardcoded value as the next lines depend on it. */ || (paParms[0].type != VBOX_HGCM_SVC_PARM_PTR) /* key */ ) rc = VERR_INVALID_PARAMETER; if (RT_SUCCESS(rc)) rc = VBoxHGCMParmPtrGet(&paParms[0], (void **) &pszKey, &cbKey); if (RT_SUCCESS(rc)) rc = validateKey(pszKey, cbKey); if (RT_SUCCESS(rc)) CFGMR3RemoveValue(mpValueNode, pszKey); LogFlowThisFunc(("rc = %Rrc\n", rc)); return rc; } /** * Matches a sample name against a pattern. * * @returns True if matches, false if not. * @param pszPat Pattern. * @param pszName Name to match against the pattern. * @todo move this into IPRT */ static bool matchesSinglePattern(const char *pszPat, const char *pszName) { /* ASSUMES ASCII */ for (;;) { char chPat = *pszPat; switch (chPat) { default: if (*pszName != chPat) return false; break; case '*': { while ((chPat = *++pszPat) == '*' || chPat == '?') /* nothing */; for (;;) { char ch = *pszName++; if ( ch == chPat && ( !chPat || matchesSinglePattern(pszPat + 1, pszName))) return true; if (!ch) return false; } /* won't ever get here */ break; } case '?': if (!*pszName) return false; break; case '\0': return !*pszName; } pszName++; pszPat++; } return true; } /* Checks to see if the given string matches against one of the patterns in * the list. */ static bool matchesPattern(const char *paszPatterns, size_t cchPatterns, const char *pszString) { size_t iOffs = 0; /* If the first pattern in the list is empty, treat it as "match all". */ bool matched = (cchPatterns > 0) && (0 == *paszPatterns) ? true : false; while ((iOffs < cchPatterns) && !matched) { size_t cchCurrent; if ( RT_SUCCESS(RTStrNLenEx(paszPatterns + iOffs, cchPatterns - iOffs, &cchCurrent)) && (cchCurrent > 0) ) { matched = matchesSinglePattern(paszPatterns + iOffs, pszString); iOffs += cchCurrent + 1; } else iOffs = cchPatterns; } return matched; } /** * Enumerate guest properties by mask, checking the validity * of the arguments passed. * * @returns iprt status value * @param cParms the number of HGCM parameters supplied * @param paParms the array of HGCM parameters * @thread HGCM */ int Service::enumProps(uint32_t cParms, VBOXHGCMSVCPARM paParms[]) { /* We reallocate the temporary buffer in which we build up our array in * increments of size BLOCK: */ enum { BLOCKINCRFULL = (MAX_NAME_LEN + MAX_VALUE_LEN + MAX_FLAGS_LEN + 2048), BLOCKINCR = BLOCKINCRFULL - BLOCKINCRFULL % 1024 }; int rc = VINF_SUCCESS; /* * Get the HGCM function arguments. */ char *paszPatterns = NULL, *pchBuf = NULL; uint32_t cchPatterns = 0, cchBuf = 0; LogFlowThisFunc(("\n")); if ( (cParms != 3) /* Hardcoded value as the next lines depend on it. */ || (paParms[0].type != VBOX_HGCM_SVC_PARM_PTR) /* patterns */ || (paParms[1].type != VBOX_HGCM_SVC_PARM_PTR) /* return buffer */ ) rc = VERR_INVALID_PARAMETER; if (RT_SUCCESS(rc)) rc = VBoxHGCMParmPtrGet(&paParms[0], (void **) &paszPatterns, &cchPatterns); if (RT_SUCCESS(rc)) rc = VBoxHGCMParmPtrGet(&paParms[1], (void **) &pchBuf, &cchBuf); /* * Start by enumerating all values in the current node into a temporary buffer. */ RTMemAutoPtr TmpBuf; uint32_t cchTmpBuf = 0, iTmpBuf = 0; PCFGMLEAF pLeaf = CFGMR3GetFirstValue(mpValueNode); while ((pLeaf != NULL) && RT_SUCCESS(rc)) { /* Reallocate the buffer if it has got too tight */ if (iTmpBuf + BLOCKINCR > cchTmpBuf) { cchTmpBuf += BLOCKINCR * 2; if (!TmpBuf.realloc(cchTmpBuf)) rc = VERR_NO_MEMORY; } /* Fetch the name into the buffer and if it matches one of the * patterns, add its value and an empty timestamp and flags. If it * doesn't match, we simply overwrite it in the buffer. */ if (RT_SUCCESS(rc)) rc = CFGMR3GetValueName(pLeaf, &TmpBuf[iTmpBuf], cchTmpBuf - iTmpBuf); /* Only increment the buffer offest if the name matches, otherwise we * overwrite it next iteration. */ if ( RT_SUCCESS(rc) && matchesPattern(paszPatterns, cchPatterns, &TmpBuf[iTmpBuf]) ) { const char *pszName = &TmpBuf[iTmpBuf]; /* Get value */ iTmpBuf += strlen(&TmpBuf[iTmpBuf]) + 1; rc = CFGMR3QueryString(mpValueNode, pszName, &TmpBuf[iTmpBuf], cchTmpBuf - iTmpBuf); if (RT_SUCCESS(rc)) { /* Get timestamp */ iTmpBuf += strlen(&TmpBuf[iTmpBuf]) + 1; uint64_t u64Timestamp = 0; if (mpTimestampNode != NULL) CFGMR3QueryU64(mpTimestampNode, pszName, &u64Timestamp); iTmpBuf += RTStrFormatNumber(&TmpBuf[iTmpBuf], u64Timestamp, 10, 0, 0, 0) + 1; /* Get flags */ TmpBuf[iTmpBuf] = '\0'; /* In case there are none. */ if (mpFlagsNode != NULL) CFGMR3QueryString(mpFlagsNode, pszName, &TmpBuf[iTmpBuf], cchTmpBuf - iTmpBuf); /* Ignore failure */ iTmpBuf += strlen(&TmpBuf[iTmpBuf]) + 1; } } if (RT_SUCCESS(rc)) pLeaf = CFGMR3GetNextValue(pLeaf); } if (RT_SUCCESS(rc)) { /* The terminator. We *do* have space left for this. */ TmpBuf[iTmpBuf] = '\0'; TmpBuf[iTmpBuf + 1] = '\0'; TmpBuf[iTmpBuf + 2] = '\0'; TmpBuf[iTmpBuf + 3] = '\0'; iTmpBuf += 4; VBoxHGCMParmUInt32Set(&paParms[2], iTmpBuf); /* Copy the memory if it fits into the guest buffer */ if (iTmpBuf <= cchBuf) memcpy(pchBuf, TmpBuf.get(), iTmpBuf); else rc = VERR_BUFFER_OVERFLOW; } return rc; } /** * Handle an HGCM service call. * @copydoc VBOXHGCMSVCFNTABLE::pfnCall * @note All functions which do not involve an unreasonable delay will be * handled synchronously. If needed, we will add a request handler * thread in future for those which do. * * @thread HGCM */ void Service::call (VBOXHGCMCALLHANDLE callHandle, uint32_t u32ClientID, void * /* pvClient */, uint32_t eFunction, uint32_t cParms, VBOXHGCMSVCPARM paParms[]) { int rc = VINF_SUCCESS; bool fCallSync = true; LogFlowFunc(("u32ClientID = %d, fn = %d, cParms = %d, pparms = %d\n", u32ClientID, eFunction, cParms, paParms)); switch (eFunction) { /* The guest wishes to read a property */ case GET_PROP: LogFlowFunc(("GET_PROP\n")); rc = getProperty(cParms, paParms); break; /* The guest wishes to set a property */ case SET_PROP: LogFlowFunc(("SET_PROP\n")); rc = setKey(cParms, paParms); break; /* The guest wishes to set a property value */ case SET_PROP_VALUE: LogFlowFunc(("SET_PROP_VALUE\n")); rc = setKey(cParms, paParms); break; /* The guest wishes to remove a configuration value */ case DEL_PROP: LogFlowFunc(("DEL_PROP\n")); rc = delKey(cParms, paParms); break; /* The guest wishes to enumerate all properties */ case ENUM_PROPS: LogFlowFunc(("ENUM_PROPS\n")); rc = enumProps(cParms, paParms); break; default: rc = VERR_NOT_IMPLEMENTED; } if (fCallSync) { LogFlowFunc(("rc = %Rrc\n", rc)); mpHelpers->pfnCallComplete (callHandle, rc); } } /** * Service call handler for the host. * @copydoc VBOXHGCMSVCFNTABLE::pfnHostCall * @thread hgcm */ int Service::hostCall (uint32_t eFunction, uint32_t cParms, VBOXHGCMSVCPARM paParms[]) { int rc = VINF_SUCCESS; LogFlowFunc(("fn = %d, cParms = %d, pparms = %d\n", eFunction, cParms, paParms)); switch (eFunction) { /* Set the root CFGM node used. This should be called when instantiating * the service. */ case SET_CFGM_NODE: { LogFlowFunc(("SET_CFGM_NODE\n")); if ((cParms < 1) || (cParms > 3)) { rc = VERR_INVALID_PARAMETER; } else if ( paParms[0].type != VBOX_HGCM_SVC_PARM_PTR /* pValue */ || ((cParms > 1) && (paParms[1].type != VBOX_HGCM_SVC_PARM_PTR)) /* pTimestamp */ || ((cParms > 2) && (paParms[2].type != VBOX_HGCM_SVC_PARM_PTR)) /* pFlags */ ) { rc = VERR_INVALID_PARAMETER; } else { PCFGMNODE pNode = NULL; uint32_t cbDummy; rc = VBoxHGCMParmPtrGet (&paParms[0], (void **) &pNode, &cbDummy); if (RT_SUCCESS(rc)) mpValueNode = pNode; if (RT_SUCCESS(rc) && (cParms > 1)) { rc = VBoxHGCMParmPtrGet (&paParms[1], (void **) &pNode, &cbDummy); mpTimestampNode = pNode; } if (RT_SUCCESS(rc) && (cParms > 2)) { rc = VBoxHGCMParmPtrGet (&paParms[2], (void **) &pNode, &cbDummy); mpFlagsNode = pNode; } } } break; /* The host wishes to read a configuration value */ case GET_PROP_HOST: LogFlowFunc(("GET_PROP_HOST\n")); rc = getProperty(cParms, paParms); break; /* The host wishes to set a configuration value */ case SET_PROP_HOST: LogFlowFunc(("SET_PROP_HOST\n")); rc = setKey(cParms, paParms); break; /* The host wishes to set a configuration value */ case SET_PROP_VALUE_HOST: LogFlowFunc(("SET_PROP_VALUE_HOST\n")); rc = setKey(cParms, paParms); break; /* The host wishes to remove a configuration value */ case DEL_PROP_HOST: LogFlowFunc(("DEL_CONFIG_KEY_HOST\n")); rc = delKey(cParms, paParms); break; /* The host wishes to enumerate all properties */ case ENUM_PROPS_HOST: LogFlowFunc(("ENUM_PROPS\n")); rc = enumProps(cParms, paParms); break; default: rc = VERR_NOT_SUPPORTED; break; } LogFlowFunc(("rc = %Vrc\n", rc)); return rc; } } /* namespace guestProp */ using guestProp::Service; /** * @copydoc VBOXHGCMSVCLOAD */ extern "C" DECLCALLBACK(DECLEXPORT(int)) VBoxHGCMSvcLoad (VBOXHGCMSVCFNTABLE *ptable) { int rc = VINF_SUCCESS; LogFlowFunc(("ptable = %p\n", ptable)); if (!VALID_PTR(ptable)) { rc = VERR_INVALID_PARAMETER; } else { LogFlowFunc(("ptable->cbSize = %d, ptable->u32Version = 0x%08X\n", ptable->cbSize, ptable->u32Version)); if ( ptable->cbSize != sizeof (VBOXHGCMSVCFNTABLE) || ptable->u32Version != VBOX_HGCM_SVC_VERSION) { rc = VERR_VERSION_MISMATCH; } else { std::auto_ptr apService; try { apService = std::auto_ptr(new Service(ptable->pHelpers)); } catch (...) { /* No exceptions may propogate outside. */ rc = VERR_UNRESOLVED_ERROR; } if (RT_SUCCESS(rc)) { /* We do not maintain connections, so no client data is needed. */ ptable->cbClient = 0; ptable->pfnUnload = Service::svcUnload; ptable->pfnConnect = Service::svcConnectDisconnect; ptable->pfnDisconnect = Service::svcConnectDisconnect; ptable->pfnCall = Service::svcCall; ptable->pfnHostCall = Service::svcHostCall; ptable->pfnSaveState = NULL; /* The service is stateless by definition, so the */ ptable->pfnLoadState = NULL; /* normal construction done before restoring suffices */ ptable->pfnRegisterExtension = NULL; /* Service specific initialization. */ ptable->pvService = apService.release(); } } } LogFlowFunc(("returning %Rrc\n", rc)); return rc; }