diff options
Diffstat (limited to 'src/VBox/Main/src-all/win/VBoxProxyStub.c')
-rw-r--r-- | src/VBox/Main/src-all/win/VBoxProxyStub.c | 2599 |
1 files changed, 2599 insertions, 0 deletions
diff --git a/src/VBox/Main/src-all/win/VBoxProxyStub.c b/src/VBox/Main/src-all/win/VBoxProxyStub.c new file mode 100644 index 00000000..44b9cd84 --- /dev/null +++ b/src/VBox/Main/src-all/win/VBoxProxyStub.c @@ -0,0 +1,2599 @@ +/* $Id: VBoxProxyStub.c $ */ +/** @file + * VBoxProxyStub - Proxy Stub and Typelib, COM DLL exports and DLL init/term. + * + * @remarks This is a C file and not C++ because rpcproxy.h isn't C++ clean, + * at least not in SDK v7.1. + */ + +/* + * Copyright (C) 2006-2022 Oracle and/or its affiliates. + * + * This file is part of VirtualBox base platform packages, as + * available from https://www.virtualbox.org. + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License + * as published by the Free Software Foundation, in version 3 of the + * License. + * + * This program is distributed in the hope that it will be useful, but + * WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, see <https://www.gnu.org/licenses>. + * + * SPDX-License-Identifier: GPL-3.0-only + */ + + +/********************************************************************************************************************************* +* Header Files * +*********************************************************************************************************************************/ +#define LOG_GROUP LOG_GROUP_MAIN +#define PROXY_DELEGATION /* see generated dlldata.c */ +#include <iprt/nt/nt-and-windows.h> +#include <rpcproxy.h> +#include <iprt/win/shlwapi.h> +#include <stdio.h> + +#include "VirtualBox.h" +#include <VBox/cdefs.h> /* for VBOX_STRICT */ +#include <VBox/log.h> +#include <iprt/alloca.h> +#include <iprt/assert.h> +#include <iprt/ctype.h> +#include <iprt/initterm.h> +#include <iprt/path.h> +#include <iprt/string.h> +#include <iprt/uuid.h> +#include <iprt/utf16.h> + + +/********************************************************************************************************************************* +* Defined Constants And Macros * +*********************************************************************************************************************************/ +#ifdef DEBUG_bird +# define VBSP_LOG_ENABLED +#endif + +#ifdef VBSP_LOG_ENABLED +# define VBSP_LOG_VALUE_CHANGE(a) RTAssertMsg2 a +#else +# define VBSP_LOG_VALUE_CHANGE(a) do { } while (0) +#endif + +#ifdef VBSP_LOG_ENABLED +# define VBSP_LOG_SET_VALUE(a) RTAssertMsg2 a +#else +# define VBSP_LOG_SET_VALUE(a) do { } while (0) +#endif + +#ifdef VBSP_LOG_ENABLED +# define VBSP_LOG_NEW_KEY(a) RTAssertMsg2 a +#else +# define VBSP_LOG_NEW_KEY(a) do { } while (0) +#endif + +#ifdef VBSP_LOG_ENABLED +# define VBSP_LOG_DEL_KEY(a) RTAssertMsg2 a +#else +# define VBSP_LOG_DEL_KEY(a) do { } while (0) +#endif + +/** + * Selects the proxy stub DLL based on 32-on-64-bit and host OS version. + * + * The legacy DLL covers 64-bit pre Windows 7 versions of Windows. W2K3-amd64 + * has trouble parsing the result when MIDL /target NT51 or higher. Vista and + * windows server 2008 seems to have trouble with newer IDL compilers. + */ +#if ARCH_BITS == 64 || defined(VBOX_IN_32_ON_64_MAIN_API) +# define VBPS_PROXY_STUB_FILE(a_fIs32On64) ( (a_fIs32On64) ? "x86\\VBoxProxyStub-x86.dll" : VBPS_PROXY_STUB_FILE_SUB() ) +#else +# define VBPS_PROXY_STUB_FILE(a_fIs32On64) VBPS_PROXY_STUB_FILE_SUB() +#endif +#define VBPS_PROXY_STUB_FILE_SUB() \ + ( RT_MAKE_U64(((PKUSER_SHARED_DATA)MM_SHARED_USER_DATA_VA)->NtMinorVersion, \ + ((PKUSER_SHARED_DATA)MM_SHARED_USER_DATA_VA)->NtMajorVersion) >= RT_MAKE_U64(1/*Lo*/,6/*Hi*/) \ + ? "VBoxProxyStub.dll" : "VBoxProxyStubLegacy.dll" ) + +/** For use with AssertLogRel except a_Expr1 from assertions but not LogRel. */ +#ifdef RT_STRICT +# define VBPS_LOGREL_NO_ASSERT(a_Expr) (a_Expr) +#else +# define VBPS_LOGREL_NO_ASSERT(a_Expr) false +#endif + + +/********************************************************************************************************************************* +* Global Variables * +*********************************************************************************************************************************/ +/** For NdrXxx. */ +CStdPSFactoryBuffer g_ProxyStubFactory = /* see generated dlldata.c */ +{ + NULL, + 0, + NULL, + 0 +}; +/** Reference to VirtualBox_p.c structure. */ +EXTERN_PROXY_FILE(VirtualBox) /* see generated dlldata.c */ +/** For NdrXxx and for returning. */ +static const ProxyFileInfo *g_apProxyFiles[] = +{ + REFERENCE_PROXY_FILE(VirtualBox), + NULL /* terminator */ +}; +/** The class ID for this proxy stub factory (see Makefile). */ +static const CLSID g_ProxyClsId = PROXY_CLSID_IS; +/** The instance handle of this DLL. For use in registration routines. */ +static HINSTANCE g_hDllSelf; + + +/** Type library GUIDs to clean up manually. + * Must be upper case! */ +static PCRTUTF16 const g_apwszTypeLibIds[] = +{ + L"{46137EEC-703B-4FE5-AFD4-7C9BBBBA0259}", + L"{D7569351-1750-46F0-936E-BD127D5BC264}", +}; + +/** Type library version to clean up manually. */ +static PCRTUTF16 const g_apwszTypelibVersions[] = +{ + L"1.0", + L"1.3", +}; + +/** Proxy stub class IDs we wish to clean up manually. + * Must be upper case! */ +static PCRTUTF16 const g_apwszProxyStubClsIds[] = +{ + L"{0BB3B78C-1807-4249-5BA5-EA42D66AF0BF}", + L"{327E3C00-EE61-462F-AED3-0DFF6CBF9904}", +}; + + +/** + * DLL main function. + * + * @returns TRUE (/ FALSE). + * @param hInstance The DLL handle. + * @param dwReason The rason for the call (DLL_XXX). + * @param lpReserved Reserved. + */ +BOOL WINAPI DllMain(HINSTANCE hInstance, DWORD dwReason, LPVOID lpReserved) +{ + switch (dwReason) + { + case DLL_PROCESS_ATTACH: + /* Save the DLL handle so we can get the path to this DLL during + registration and updating. */ + g_hDllSelf = hInstance; + + /* We don't need callbacks for thread creation and destruction. */ + DisableThreadLibraryCalls(hInstance); + + /* Init IPRT. */ + RTR3InitDll(RTR3INIT_FLAGS_UNOBTRUSIVE); + Log12(("VBoxProxyStub[%u]/DllMain: DLL_PROCESS_ATTACH\n", GetCurrentProcessId())); + +#ifdef VBOX_STRICT + { + /* + * Check that no interface has more than 256 methods in the stub vtable. + */ + const ProxyFileInfo **ppProxyFile = &g_apProxyFiles[0]; + const ProxyFileInfo *pProxyFile; + while ((pProxyFile = *ppProxyFile++) != NULL) + { + const PCInterfaceStubVtblList * const papStubVtbls = pProxyFile->pStubVtblList; + const char * const *papszNames = pProxyFile->pNamesArray; + unsigned iIf = pProxyFile->TableSize; + AssertStmt(iIf < 1024, iIf = 0); + Assert(pProxyFile->TableVersion == 2); + + while (iIf-- > 0) + AssertMsg(papStubVtbls[iIf]->header.DispatchTableCount <= 256, + ("%s: DispatchTableCount=%d\n", papszNames[iIf], papStubVtbls[iIf]->header.DispatchTableCount)); + } + } +#endif + break; + + case DLL_PROCESS_DETACH: + Log12(("VBoxProxyStub[%u]/DllMain: DLL_PROCESS_DETACH\n", GetCurrentProcessId())); + break; + } + + NOREF(lpReserved); + return TRUE; +} + + +/** + * RPC entry point returning info about the proxy. + */ +void RPC_ENTRY GetProxyDllInfo(const ProxyFileInfo ***ppapInfo, const CLSID **ppClsid) +{ + *ppapInfo = &g_apProxyFiles[0]; + *ppClsid = &g_ProxyClsId; + Log12(("VBoxProxyStub[%u]/GetProxyDllInfo:\n", GetCurrentProcessId())); +} + + +/** + * Instantiate the proxy stub class object. + * + * @returns COM status code + * @param rclsid Reference to the ID of the call to instantiate (our + * g_ProxyClsId). + * @param riid The interface ID to return (IID_IPSFactoryBuffer). + * @param ppv Where to return the interface pointer on success. + */ +HRESULT STDAPICALLTYPE DllGetClassObject(REFCLSID rclsid, REFIID riid, void **ppv) +{ + HRESULT hrc; + Assert(memcmp(rclsid, &g_ProxyClsId, sizeof(g_ProxyClsId)) == 0); + + hrc = NdrDllGetClassObject(rclsid, riid, ppv, /* see DLLGETCLASSOBJECTROUTINE in RpcProxy.h */ + g_apProxyFiles, &g_ProxyClsId, &g_ProxyStubFactory); + + /* + * This may fail if the IDL compiler generates code that is incompatible + * with older windows releases. Like for instance 64-bit W2K8 SP1 not + * liking the output of MIDL 7.00.0555 (from the v7.1 SDK), despite + * /target being set to NT51. + */ + AssertLogRelMsg(hrc == S_OK, ("%Rhrc\n", hrc)); + Log12(("VBoxProxyStub[%u]/DllGetClassObject(%RTuuid, %RTuuid, %p): %#x + *ppv=%p\n", + GetCurrentProcessId(), rclsid, riid, ppv, hrc, ppv ? *ppv : NULL)); + return hrc; +} + + +/** + * Checks whether the DLL can be unloaded or not. + * + * @returns S_OK if it can be unloaded, S_FALSE if not. + */ +HRESULT STDAPICALLTYPE DllCanUnloadNow(void) +{ + HRESULT hrc = NdrDllCanUnloadNow(&g_ProxyStubFactory); /* see DLLCANUNLOADNOW in RpcProxy.h */ + Log12(("VBoxProxyStub[%u]/DllCanUnloadNow: %Rhrc\n", GetCurrentProcessId(), hrc)); + return hrc; +} + + + +/** + * Release call that could be referenced by VirtualBox_p.c via + * CStdStubBuffer_METHODS. + * + * @returns New reference count. + * @param pThis Buffer to release. + */ +ULONG STDMETHODCALLTYPE CStdStubBuffer_Release(IRpcStubBuffer *pThis) /* see CSTDSTUBBUFFERRELEASE in RpcProxy.h */ +{ + ULONG cRefs = NdrCStdStubBuffer_Release(pThis, (IPSFactoryBuffer *)&g_ProxyStubFactory); + Log12(("VBoxProxyStub[%u]/CStdStubBuffer_Release: %p -> %#x\n", GetCurrentProcessId(), pThis, cRefs)); + return cRefs; +} + + +/** + * Release call referenced by VirtualBox_p.c via + * CStdStubBuffer_DELEGATING_METHODS. + * + * @returns New reference count. + * @param pThis Buffer to release. + */ +ULONG WINAPI CStdStubBuffer2_Release(IRpcStubBuffer *pThis) /* see CSTDSTUBBUFFER2RELEASE in RpcProxy.h */ +{ + ULONG cRefs = NdrCStdStubBuffer2_Release(pThis, (IPSFactoryBuffer *)&g_ProxyStubFactory); + Log12(("VBoxProxyStub[%u]/CStdStubBuffer2_Release: %p -> %#x\n", GetCurrentProcessId(), pThis, cRefs)); + return cRefs; +} + + +/** + * Pure virtual method implementation referenced by VirtualBox_p.c + */ +void __cdecl _purecall(void) /* see DLLDUMMYPURECALL in RpcProxy.h */ +{ + AssertFailed(); +} + + +#ifdef VBSP_LOG_ENABLED +# include <iprt/asm.h> + +/** For logging full key names. */ +static PCRTUTF16 vbpsDebugKeyToWSZ(HKEY hKey) +{ + static union + { + KEY_NAME_INFORMATION NameInfo; + WCHAR awchPadding[260]; + } s_aBufs[4]; + static uint32_t volatile iNext = 0; + uint32_t i = ASMAtomicIncU32(&iNext) % RT_ELEMENTS(s_aBufs); + ULONG cbRet = 0; + NTSTATUS rcNt; + + memset(&s_aBufs[i], 0, sizeof(s_aBufs[i])); + rcNt = NtQueryKey(hKey, KeyNameInformation, &s_aBufs[i], sizeof(s_aBufs[i]) - sizeof(WCHAR), &cbRet); + if (!NT_SUCCESS(rcNt)) + s_aBufs[i].NameInfo.NameLength = 0; + s_aBufs[i].NameInfo.Name[s_aBufs[i].NameInfo.NameLength] = '\0'; + return s_aBufs[i].NameInfo.Name; +} +#endif + +/** + * Registry modifier state. + */ +typedef struct VBPSREGSTATE +{ + /** Where the classes and stuff are to be registered. */ + HKEY hkeyClassesRootDst; + /** The handle to the CLSID key under hkeyClassesRootDst. */ + HKEY hkeyClsidRootDst; + /** The handle to the Interface key under hkeyClassesRootDst. */ + HKEY hkeyInterfaceRootDst; + + /** Alternative locations where data needs to be deleted, but never updated. */ + struct + { + /** The classes root key handle. */ + HKEY hkeyClasses; + /** The classes/CLSID key handle. */ + HKEY hkeyClsid; + /** The classes/Interface key handle. */ + HKEY hkeyInterface; + } aAltDeletes[3]; + /** Alternative delete locations. */ + uint32_t cAltDeletes; + + /** The current total result. */ + LSTATUS rc; + + /** KEY_WOW64_32KEY, KEY_WOW64_64KEY or 0 (for default). Allows doing all + * almost the work from one process (at least W7+ due to aliases). */ + DWORD fSamWow; + /** Desired key access when only deleting. */ + DWORD fSamDelete; + /** Desired key access when only doing updates. */ + DWORD fSamUpdate; + /** Desired key access when both deleting and updating. */ + DWORD fSamBoth; + /** Whether to delete registrations first. */ + bool fDelete; + /** Whether to update registry value and keys. */ + bool fUpdate; + +} VBPSREGSTATE; + + +/** + * Initializes a registry modification job state. + * + * Always call vbpsRegTerm! + * + * @returns Windows error code (ERROR_SUCCESS on success). + * @param pState The state to init. + * @param hkeyRoot The registry root tree constant. + * @param pszSubRoot The path to the where the classes are registered, + * NULL if @a hkeyRoot. + * @param fDelete Whether to delete registrations first. + * @param fUpdate Whether to update registrations. + * @param fSamWow KEY_WOW64_32KEY or 0. + */ +static LSTATUS vbpsRegInit(VBPSREGSTATE *pState, HKEY hkeyRoot, const char *pszSubRoot, bool fDelete, bool fUpdate, DWORD fSamWow) +{ + LSTATUS rc; + unsigned i = 0; + + /* + * Initialize the whole structure first so we can safely call vbpsRegTerm on failure. + */ + pState->hkeyClassesRootDst = NULL; + pState->hkeyClsidRootDst = NULL; + pState->hkeyInterfaceRootDst = NULL; + for (i = 0; i < RT_ELEMENTS(pState->aAltDeletes); i++) + { + pState->aAltDeletes[i].hkeyClasses = NULL; + pState->aAltDeletes[i].hkeyClsid = NULL; + pState->aAltDeletes[i].hkeyInterface = NULL; + } + pState->cAltDeletes = 0; + pState->rc = ERROR_SUCCESS; + pState->fDelete = fDelete; + pState->fUpdate = fUpdate; + pState->fSamWow = fSamWow; + pState->fSamDelete = 0; + if (fDelete) + pState->fSamDelete = pState->fSamWow | DELETE | KEY_ENUMERATE_SUB_KEYS | KEY_QUERY_VALUE + | STANDARD_RIGHTS_READ | STANDARD_RIGHTS_WRITE; + pState->fSamUpdate = 0; + if (fUpdate) + pState->fSamUpdate = pState->fSamWow | KEY_ENUMERATE_SUB_KEYS | KEY_QUERY_VALUE | KEY_SET_VALUE | KEY_CREATE_SUB_KEY + | STANDARD_RIGHTS_READ | STANDARD_RIGHTS_WRITE; + pState->fSamBoth = pState->fSamDelete | pState->fSamUpdate; + + /* + * Open the root keys. + */ + rc = RegOpenKeyExA(hkeyRoot, pszSubRoot, 0 /*fOptions*/, pState->fSamBoth, &pState->hkeyClassesRootDst); + if (rc == ERROR_SUCCESS) + { + rc = RegCreateKeyExW(pState->hkeyClassesRootDst, L"CLSID", 0 /*Reserved*/, NULL /*pszClass*/, 0 /*fOptions*/, + pState->fSamBoth, NULL /*pSecAttr*/, &pState->hkeyClsidRootDst, NULL /*pdwDisposition*/); + if (rc == ERROR_SUCCESS) + return ERROR_SUCCESS; + + /* Ignore access denied errors as these may easily happen for + non-admin users. Just give up when this happens */ + AssertLogRelMsgReturn(rc == ERROR_ACCESS_DENIED, ("%u\n", rc), pState->rc = rc); + } + else + AssertLogRelMsgReturn(rc == ERROR_ACCESS_DENIED, ("%u\n", rc), pState->rc = rc); + return pState->rc = rc; +} + + +/** + * Terminates the state, closing all open keys. + * + * @param pState The state to clean up. + */ +static void vbpsRegTerm(VBPSREGSTATE *pState) +{ + LSTATUS rc; + if (pState->hkeyClassesRootDst) + { + rc = RegCloseKey(pState->hkeyClassesRootDst); + Assert(rc == ERROR_SUCCESS); + pState->hkeyClassesRootDst = NULL; + } + if (pState->hkeyClsidRootDst) + { + rc = RegCloseKey(pState->hkeyClsidRootDst); + Assert(rc == ERROR_SUCCESS); + pState->hkeyClsidRootDst = NULL; + } + if (pState->hkeyInterfaceRootDst) + { + rc = RegCloseKey(pState->hkeyInterfaceRootDst); + Assert(rc == ERROR_SUCCESS); + pState->hkeyInterfaceRootDst = NULL; + } + + while (pState->cAltDeletes > 0 && pState->cAltDeletes <= RT_ELEMENTS(pState->aAltDeletes)) + { + unsigned i = --pState->cAltDeletes; + if (pState->aAltDeletes[i].hkeyClasses) + { + rc = RegCloseKey(pState->aAltDeletes[i].hkeyClasses); + Assert(rc == ERROR_SUCCESS); + pState->aAltDeletes[i].hkeyClasses = NULL; + } + if (pState->aAltDeletes[i].hkeyClsid) + { + rc = RegCloseKey(pState->aAltDeletes[i].hkeyClsid); + Assert(rc == ERROR_SUCCESS); + pState->aAltDeletes[i].hkeyClsid = NULL; + } + if (pState->aAltDeletes[i].hkeyInterface) + { + rc = RegCloseKey(pState->aAltDeletes[i].hkeyInterface); + Assert(rc == ERROR_SUCCESS); + pState->aAltDeletes[i].hkeyInterface = NULL; + } + } +} + + +/** + * Add an alternative registry classes tree from which to remove keys. + * + * @returns ERROR_SUCCESS if we successfully opened the destination root, other + * wise windows error code (remebered). + * @param pState The registry modifier state. + * @param hkeyAltRoot The root of the alternate registry classes + * location. + * @param pszAltSubRoot The path to the 'classes' sub-key, or NULL if + * hkeyAltRoot is it. + */ +static LSTATUS vbpsRegAddAltDelete(VBPSREGSTATE *pState, HKEY hkeyAltRoot, const char *pszAltSubRoot) +{ + unsigned i; + LSTATUS rc; + + /* Ignore call if not in delete mode. */ + if (!pState->fDelete) + return ERROR_SUCCESS; + + /* Check that there is space in the state. */ + i = pState->cAltDeletes; + AssertReturn(i < RT_ELEMENTS(pState->aAltDeletes), pState->rc = ERROR_TOO_MANY_NAMES); + + + /* Open the root. */ + rc = RegOpenKeyExA(hkeyAltRoot, pszAltSubRoot, 0 /*fOptions*/, pState->fSamDelete, + &pState->aAltDeletes[i].hkeyClasses); + if (rc == ERROR_SUCCESS) + { + /* Try open the CLSID subkey, it's fine if it doesn't exists. */ + rc = RegOpenKeyExW(pState->aAltDeletes[i].hkeyClasses, L"CLSID", 0 /*fOptions*/, pState->fSamDelete, + &pState->aAltDeletes[i].hkeyClsid); + if (rc == ERROR_SUCCESS || rc == ERROR_FILE_NOT_FOUND) + { + if (rc == ERROR_FILE_NOT_FOUND) + pState->aAltDeletes[i].hkeyClsid = NULL; + pState->cAltDeletes = i + 1; + return ERROR_SUCCESS; + } + AssertLogRelMsgFailed(("%u\n", rc)); + RegCloseKey(pState->aAltDeletes[i].hkeyClasses); + } + /* No need to add non-existing alternative roots, nothing to delete in the void. */ + else if (rc == ERROR_FILE_NOT_FOUND) + rc = ERROR_SUCCESS; + else + { + AssertLogRelMsgFailed(("%u (%#x %s)\n", rc)); + pState->rc = rc; + } + + pState->aAltDeletes[i].hkeyClasses = NULL; + pState->aAltDeletes[i].hkeyClsid = NULL; + return rc; +} + + +/** + * Open the 'Interface' keys under the current classes roots. + * + * We don't do this during vbpsRegInit as it's only needed for updating. + * + * @returns ERROR_SUCCESS if we successfully opened the destination root, other + * wise windows error code (remebered). + * @param pState The registry modifier state. + */ +static LSTATUS vbpsRegOpenInterfaceKeys(VBPSREGSTATE *pState) +{ + unsigned i; + LSTATUS rc; + + /* + * Under the root destination. + */ + if (pState->hkeyInterfaceRootDst == NULL) + { + if (pState->fSamUpdate) + rc = RegCreateKeyExW(pState->hkeyClassesRootDst, L"Interface", 0 /*Reserved*/, NULL /*pszClass*/, 0 /*fOptions*/, + pState->fSamBoth, NULL /*pSecAttr*/, &pState->hkeyInterfaceRootDst, NULL /*pdwDisposition*/); + else + rc = RegOpenKeyExW(pState->hkeyClassesRootDst, L"Interface", 0 /*fOptions*/, pState->fSamBoth, + &pState->hkeyClsidRootDst); + if (rc == ERROR_ACCESS_DENIED) + { + pState->hkeyInterfaceRootDst = NULL; + return pState->rc = rc; + } + AssertLogRelMsgReturnStmt(rc == ERROR_SUCCESS, ("%u\n", rc), pState->hkeyInterfaceRootDst = NULL, pState->rc = rc); + } + + /* + * Under the alternative delete locations. + */ + i = pState->cAltDeletes; + while (i-- > 0) + if (pState->aAltDeletes[i].hkeyInterface == NULL) + { + rc = RegOpenKeyExW(pState->aAltDeletes[i].hkeyClasses, L"Interface", 0 /*fOptions*/, pState->fSamDelete, + &pState->aAltDeletes[i].hkeyInterface); + if (rc != ERROR_SUCCESS) + { + AssertMsgStmt(rc == ERROR_FILE_NOT_FOUND || rc == ERROR_ACCESS_DENIED, ("%u\n", rc), pState->rc = rc); + pState->aAltDeletes[i].hkeyInterface = NULL; + } + } + + return ERROR_SUCCESS; +} + + +/** The destination buffer size required by vbpsFormatUuidInCurly. */ +#define CURLY_UUID_STR_BUF_SIZE 40 + +/** + * Formats a UUID to a string, inside curly braces. + * + * @returns @a pszString + * @param pszString Output buffer of size CURLY_UUID_STR_BUF_SIZE. + * @param pUuidIn The UUID to format. + */ +static const char *vbpsFormatUuidInCurly(char pszString[CURLY_UUID_STR_BUF_SIZE], const CLSID *pUuidIn) +{ + static const char s_achDigits[17] = "0123456789abcdef"; + PCRTUUID pUuid = (PCRTUUID)pUuidIn; + uint32_t u32TimeLow; + unsigned u; + + pszString[ 0] = '{'; + u32TimeLow = RT_H2LE_U32(pUuid->Gen.u32TimeLow); + pszString[ 1] = s_achDigits[(u32TimeLow >> 28)/*& 0xf*/]; + pszString[ 2] = s_achDigits[(u32TimeLow >> 24) & 0xf]; + pszString[ 3] = s_achDigits[(u32TimeLow >> 20) & 0xf]; + pszString[ 4] = s_achDigits[(u32TimeLow >> 16) & 0xf]; + pszString[ 5] = s_achDigits[(u32TimeLow >> 12) & 0xf]; + pszString[ 6] = s_achDigits[(u32TimeLow >> 8) & 0xf]; + pszString[ 7] = s_achDigits[(u32TimeLow >> 4) & 0xf]; + pszString[ 8] = s_achDigits[(u32TimeLow/*>>0*/)& 0xf]; + pszString[ 9] = '-'; + u = RT_H2LE_U16(pUuid->Gen.u16TimeMid); + pszString[10] = s_achDigits[(u >> 12)/*& 0xf*/]; + pszString[11] = s_achDigits[(u >> 8) & 0xf]; + pszString[12] = s_achDigits[(u >> 4) & 0xf]; + pszString[13] = s_achDigits[(u/*>>0*/)& 0xf]; + pszString[14] = '-'; + u = RT_H2LE_U16(pUuid->Gen.u16TimeHiAndVersion); + pszString[15] = s_achDigits[(u >> 12)/*& 0xf*/]; + pszString[16] = s_achDigits[(u >> 8) & 0xf]; + pszString[17] = s_achDigits[(u >> 4) & 0xf]; + pszString[18] = s_achDigits[(u/*>>0*/)& 0xf]; + pszString[19] = '-'; + pszString[20] = s_achDigits[pUuid->Gen.u8ClockSeqHiAndReserved >> 4]; + pszString[21] = s_achDigits[pUuid->Gen.u8ClockSeqHiAndReserved & 0xf]; + pszString[22] = s_achDigits[pUuid->Gen.u8ClockSeqLow >> 4]; + pszString[23] = s_achDigits[pUuid->Gen.u8ClockSeqLow & 0xf]; + pszString[24] = '-'; + pszString[25] = s_achDigits[pUuid->Gen.au8Node[0] >> 4]; + pszString[26] = s_achDigits[pUuid->Gen.au8Node[0] & 0xf]; + pszString[27] = s_achDigits[pUuid->Gen.au8Node[1] >> 4]; + pszString[28] = s_achDigits[pUuid->Gen.au8Node[1] & 0xf]; + pszString[29] = s_achDigits[pUuid->Gen.au8Node[2] >> 4]; + pszString[30] = s_achDigits[pUuid->Gen.au8Node[2] & 0xf]; + pszString[31] = s_achDigits[pUuid->Gen.au8Node[3] >> 4]; + pszString[32] = s_achDigits[pUuid->Gen.au8Node[3] & 0xf]; + pszString[33] = s_achDigits[pUuid->Gen.au8Node[4] >> 4]; + pszString[34] = s_achDigits[pUuid->Gen.au8Node[4] & 0xf]; + pszString[35] = s_achDigits[pUuid->Gen.au8Node[5] >> 4]; + pszString[36] = s_achDigits[pUuid->Gen.au8Node[5] & 0xf]; + pszString[37] = '}'; + pszString[38] = '\0'; + + return pszString; + +} + + +/** + * Sets a registry string value, wide char variant. + * + * @returns See RegSetValueExA (errors are remembered in the state). + * @param pState The registry modifier state. + * @param hkey The key to add the value to. + * @param pwszValueNm The value name. NULL for setting the default. + * @param pwszValue The value string. + * @param uLine The line we're called from. + */ +static LSTATUS vbpsSetRegValueWW(VBPSREGSTATE *pState, HKEY hkey, PCRTUTF16 pwszValueNm, PCRTUTF16 pwszValue, unsigned uLine) +{ + DWORD const cbValue = (DWORD)((RTUtf16Len(pwszValue) + 1) * sizeof(RTUTF16)); + LSTATUS rc; + Assert(pState->fUpdate); + + /* + * If we're not deleting the key prior to updating, we're in gentle update + * mode where we will query if the existing value matches the incoming one. + */ + if (!pState->fDelete) + { + DWORD cbExistingData = cbValue + 128; + PRTUTF16 pwszExistingData = (PRTUTF16)alloca(cbExistingData); + DWORD dwExistingType; + rc = RegQueryValueExW(hkey, pwszValueNm, 0 /*Reserved*/, &dwExistingType, (BYTE *)pwszExistingData, &cbExistingData); + if (rc == ERROR_SUCCESS) + { + if ( dwExistingType == REG_SZ + && cbExistingData == cbValue) + { + if (memcmp(pwszValue, pwszExistingData, cbValue) == 0) + return ERROR_SUCCESS; + } + VBSP_LOG_VALUE_CHANGE(("vbpsSetRegValueWW: Value difference: dwExistingType=%d cbExistingData=%#x cbValue=%#x\n" + " hkey=%#x %ls; value name=%ls\n" + "existing: %.*Rhxs (%.*ls)\n" + " new: %.*Rhxs (%ls)\n", + dwExistingType, cbExistingData, cbValue, + hkey, vbpsDebugKeyToWSZ(hkey), pwszValueNm ? pwszValueNm : L"(default)", + cbExistingData, pwszExistingData, cbExistingData / sizeof(RTUTF16), pwszExistingData, + cbValue, pwszValue, pwszValue)); + } + else + Assert(rc == ERROR_FILE_NOT_FOUND || rc == ERROR_MORE_DATA); + } + + /* + * Set the value. + */ + rc = RegSetValueExW(hkey, pwszValueNm, 0 /*Reserved*/, REG_SZ, (const BYTE *)pwszValue, cbValue); + if (rc == ERROR_SUCCESS) + { + VBSP_LOG_SET_VALUE(("vbpsSetRegValueWW: %ls/%ls=%ls (at %d)\n", + vbpsDebugKeyToWSZ(hkey), pwszValueNm ? pwszValueNm : L"(Default)", pwszValue, uLine)); + return ERROR_SUCCESS; + } + + AssertLogRelMsg(VBPS_LOGREL_NO_ASSERT(rc == ERROR_ACCESS_DENIED), + ("%d: '%ls'='%ls' -> %u\n", uLine, pwszValueNm, pwszValue, rc)); + pState->rc = rc; + return rc; +} + + +/** + * Sets a registry string value. + * + * @returns See RegSetValueExA (errors are remembered in the state). + * @param pState The registry modifier state. + * @param hkey The key to add the value to. + * @param pszValueNm The value name. NULL for setting the default. + * @param pszValue The value string. + * @param uLine The line we're called from. + */ +static LSTATUS vbpsSetRegValueAA(VBPSREGSTATE *pState, HKEY hkey, const char *pszValueNm, const char *pszValue, unsigned uLine) +{ + DWORD const cbValue = (DWORD)strlen(pszValue) + 1; + LSTATUS rc; + Assert(pState->fUpdate); + + /* + * If we're not deleting the key prior to updating, we're in gentle update + * mode where we will query if the existing value matches the incoming one. + */ + if (!pState->fDelete) + { + DWORD cbExistingData = cbValue + 128; + char *pszExistingData = alloca(cbExistingData); + DWORD dwExistingType; + rc = RegQueryValueExA(hkey, pszValueNm, 0 /*Reserved*/, &dwExistingType, (PBYTE)pszExistingData, &cbExistingData); + if (rc == ERROR_SUCCESS) + { + if ( dwExistingType == REG_SZ + && cbExistingData == cbValue) + { + if (memcmp(pszValue, pszExistingData, cbValue) == 0) + return ERROR_SUCCESS; + if (memicmp(pszValue, pszExistingData, cbValue) == 0) + return ERROR_SUCCESS; + } + VBSP_LOG_VALUE_CHANGE(("vbpsSetRegValueAA: Value difference: dwExistingType=%d cbExistingData=%#x cbValue=%#x\n" + " hkey=%#x %ls; value name=%s\n" + "existing: %.*Rhxs (%.*s)\n" + " new: %.*Rhxs (%s)\n", + dwExistingType, cbExistingData, cbValue, + hkey, vbpsDebugKeyToWSZ(hkey), pszValueNm ? pszValueNm : "(default)", + cbExistingData, pszExistingData, cbExistingData, pszExistingData, + cbValue, pszValue, pszValue)); + } + else + Assert(rc == ERROR_FILE_NOT_FOUND || rc == ERROR_MORE_DATA); + } + + /* + * Set the value. + */ + rc = RegSetValueExA(hkey, pszValueNm, 0 /*Reserved*/, REG_SZ, (PBYTE)pszValue, cbValue); + if (rc == ERROR_SUCCESS) + { + VBSP_LOG_SET_VALUE(("vbpsSetRegValueAA: %ls/%s=%s (at %d)\n", + vbpsDebugKeyToWSZ(hkey), pszValueNm ? pszValueNm : "(Default)", pszValue, uLine)); + return ERROR_SUCCESS; + } + + AssertLogRelMsg(VBPS_LOGREL_NO_ASSERT(rc == ERROR_ACCESS_DENIED), + ("%d: '%s'='%s' -> %u\n", uLine, pszValueNm, pszValue, rc)); + pState->rc = rc; + return rc; +} + + +/** + * Closes a registry key. + * + * @returns See RegCloseKey (errors are remembered in the state). + * @param pState The registry modifier state. + * @param hkey The key to close. + * @param uLine The line we're called from. + */ +static LSTATUS vbpsCloseKey(VBPSREGSTATE *pState, HKEY hkey, unsigned uLine) +{ + LSTATUS rc = RegCloseKey(hkey); + if (rc == ERROR_SUCCESS) + return ERROR_SUCCESS; + + AssertLogRelMsgFailed(("%d: close key -> %u\n", uLine, rc)); + pState->rc = rc; + return rc; +} + + +/** + * Creates a registry key. + * + * @returns See RegCreateKeyA and RegSetValueExA (errors are remembered in the + * state). + * @param pState The registry modifier state. + * @param hkeyParent The parent key. + * @param pszKey The new key under @a hkeyParent. + * @param phkey Where to return the handle to the new key. + * @param uLine The line we're called from. + */ +static LSTATUS vbpsCreateRegKeyA(VBPSREGSTATE *pState, HKEY hkeyParent, const char *pszKey, PHKEY phkey, unsigned uLine) +{ + /* + * This will open if it exists and create if new, which is exactly what we want. + */ + HKEY hNewKey; + DWORD dwDisposition = 0; + LSTATUS rc = RegCreateKeyExA(hkeyParent, pszKey, 0 /*Reserved*/, NULL /*pszClass*/, 0 /*fOptions*/, + pState->fSamBoth, NULL /*pSecAttr*/, &hNewKey, &dwDisposition); + if (rc == ERROR_SUCCESS) + { + *phkey = hNewKey; + if (dwDisposition == REG_CREATED_NEW_KEY) + VBSP_LOG_NEW_KEY(("vbpsCreateRegKeyA: %ls/%s (at %d)\n", vbpsDebugKeyToWSZ(hkeyParent), pszKey, uLine)); + } + else + { + AssertLogRelMsg(VBPS_LOGREL_NO_ASSERT(rc == ERROR_ACCESS_DENIED), + ("%d: create key '%s' -> %u\n", uLine, pszKey, rc)); + pState->rc = rc; + *phkey = NULL; + } + return rc; +} + + +/** + * Creates a registry key with a default string value. + * + * @returns See RegCreateKeyA and RegSetValueExA (errors are remembered in the + * state). + * @param pState The registry modifier state. + * @param hkeyParent The parent key. + * @param pszKey The new key under @a hkeyParent. + * @param pszValue The value string. + * @param uLine The line we're called from. + */ +static LSTATUS vbpsCreateRegKeyWithDefaultValueAA(VBPSREGSTATE *pState, HKEY hkeyParent, const char *pszKey, + const char *pszValue, unsigned uLine) +{ + HKEY hNewKey; + LSTATUS rc = vbpsCreateRegKeyA(pState, hkeyParent, pszKey, &hNewKey, uLine); + if (rc == ERROR_SUCCESS) + { + rc = vbpsSetRegValueAA(pState, hNewKey, NULL /*pszValueNm*/, pszValue, uLine); + vbpsCloseKey(pState, hNewKey, uLine); + } + else + { + AssertLogRelMsg(VBPS_LOGREL_NO_ASSERT(rc == ERROR_ACCESS_DENIED), + ("%d: create key '%s'(/Default='%s') -> %u\n", uLine, pszKey, pszValue, rc)); + pState->rc = rc; + } + return rc; +} + + +/** + * Creates a registry key with a default wide string value. + * + * @returns See RegCreateKeyA and RegSetValueExA (errors are remembered in the + * state). + * @param pState The registry modifier state. + * @param hkeyParent The parent key. + * @param pszKey The new key under @a hkeyParent. + * @param pwszValue The value string. + * @param uLine The line we're called from. + */ +static LSTATUS vbpsCreateRegKeyWithDefaultValueAW(VBPSREGSTATE *pState, HKEY hkeyParent, const char *pszKey, + PCRTUTF16 pwszValue, unsigned uLine) +{ + HKEY hNewKey; + LSTATUS rc = vbpsCreateRegKeyA(pState, hkeyParent, pszKey, &hNewKey, uLine); + if (rc == ERROR_SUCCESS) + { + rc = vbpsSetRegValueWW(pState, hNewKey, NULL /*pwszValueNm*/, pwszValue, uLine); + vbpsCloseKey(pState, hNewKey, uLine); + } + else + { + AssertLogRelMsg(VBPS_LOGREL_NO_ASSERT(rc == ERROR_ACCESS_DENIED), + ("%d: create key '%s'(/Default='%ls') -> %u\n", uLine, pszKey, pwszValue, rc)); + pState->rc = rc; + } + return rc; +} + + +/** + * Creates a registry key with a default string value, return the key. + * + * @returns See RegCreateKeyA and RegSetValueExA (errors are remembered in the + * state). + * @param pState The registry modifier state. + * @param hkeyParent The parent key. + * @param pszKey The new key under @a hkeyParent. + * @param pszValue The value string. + * @param phkey Where to return the handle to the new key. + * @param uLine The line we're called from. + */ +static LSTATUS vbpsCreateRegKeyWithDefaultValueAAEx(VBPSREGSTATE *pState, HKEY hkeyParent, const char *pszKey, + const char *pszValue, PHKEY phkey, unsigned uLine) +{ + HKEY hNewKey; + LSTATUS rc = vbpsCreateRegKeyA(pState, hkeyParent, pszKey, &hNewKey, uLine); + if (rc == ERROR_SUCCESS) + { + rc = vbpsSetRegValueAA(pState, hNewKey, NULL /*pszValueNm*/, pszValue, uLine); + *phkey = hNewKey; + } + else + { + AssertLogRelMsg(VBPS_LOGREL_NO_ASSERT(rc == ERROR_ACCESS_DENIED), + ("%d: create key '%s'(/Default='%s') -> %u\n", uLine, pszKey, pszValue, rc)); + pState->rc = rc; + *phkey = NULL; + } + return rc; +} + + +/** + * Recursively deletes a registry key. + * + * @returns See SHDeleteKeyA (errors are remembered in the state). + * @param pState The registry modifier state. + * @param hkeyParent The parent key. + * @param pszKey The key under @a hkeyParent that should be + * deleted. + * @param uLine The line we're called from. + */ +static LSTATUS vbpsDeleteKeyRecursiveA(VBPSREGSTATE *pState, HKEY hkeyParent, const char *pszKey, unsigned uLine) +{ + LSTATUS rc; + + Assert(pState->fDelete); + Assert(pszKey); + AssertReturn(*pszKey != '\0', pState->rc = ERROR_INVALID_PARAMETER); + +#ifdef VBSP_LOG_ENABLED + { + HKEY hkeyLog; + rc = RegOpenKeyExA(hkeyParent, pszKey, 0 /*fOptions*/, pState->fSamDelete, &hkeyLog); + if (rc != ERROR_FILE_NOT_FOUND) + VBSP_LOG_DEL_KEY(("vbpsDeleteKeyRecursiveA: %ls/%s (at %d)\n", vbpsDebugKeyToWSZ(hkeyParent), pszKey, uLine)); + if (rc == ERROR_SUCCESS) + RegCloseKey(hkeyLog); + } +#endif + + rc = SHDeleteKeyA(hkeyParent, pszKey); + if (rc == ERROR_SUCCESS || rc == ERROR_FILE_NOT_FOUND) + return ERROR_SUCCESS; + + AssertLogRelMsg(VBPS_LOGREL_NO_ASSERT(rc == ERROR_ACCESS_DENIED), + ("%d: delete key '%s' -> %u\n", uLine, pszKey, rc)); + pState->rc = rc; + return rc; +} + + +/** + * Recursively deletes a registry key, wide char version. + * + * @returns See SHDeleteKeyW (errors are remembered in the state). + * @param pState The registry modifier state. + * @param hkeyParent The parent key. + * @param pwszKey The key under @a hkeyParent that should be + * deleted. + * @param uLine The line we're called from. + */ +static LSTATUS vbpsDeleteKeyRecursiveW(VBPSREGSTATE *pState, HKEY hkeyParent, PCRTUTF16 pwszKey, unsigned uLine) +{ + LSTATUS rc; + + Assert(pState->fDelete); + Assert(pwszKey); + AssertReturn(*pwszKey != '\0', pState->rc = ERROR_INVALID_PARAMETER); + +#ifdef VBSP_LOG_ENABLED + { + HKEY hkeyLog; + rc = RegOpenKeyExW(hkeyParent, pwszKey, 0 /*fOptions*/, pState->fSamDelete, &hkeyLog); + if (rc != ERROR_FILE_NOT_FOUND) + VBSP_LOG_DEL_KEY(("vbpsDeleteKeyRecursiveW: %ls/%ls (at %d)\n", vbpsDebugKeyToWSZ(hkeyParent), pwszKey, uLine)); + if (rc == ERROR_SUCCESS) + RegCloseKey(hkeyLog); + } +#endif + + rc = SHDeleteKeyW(hkeyParent, pwszKey); + if (rc == ERROR_SUCCESS || rc == ERROR_FILE_NOT_FOUND) + return ERROR_SUCCESS; + + AssertLogRelMsg(VBPS_LOGREL_NO_ASSERT(rc == ERROR_ACCESS_DENIED), + ("%d: delete key '%ls' -> %u\n", uLine, pwszKey, rc)); + pState->rc = rc; + return rc; +} + + +/** + * Register an application id. + * + * @returns Windows error code (errors are rememberd in the state). + * @param pState The registry modifier state. + * @param pszModuleName The module name. + * @param pszAppId The application UUID string. + * @param pszDescription The description string. + * @param pszServiceName The window service name if the application is a + * service, otherwise this must be NULL. + */ +LSTATUS VbpsRegisterAppId(VBPSREGSTATE *pState, const char *pszModuleName, const char *pszAppId, + const char *pszDescription, const char *pszServiceName) +{ + LSTATUS rc; + HKEY hkeyAppIds; + Assert(*pszAppId == '{'); + + /* + * Delete. + */ + if (pState->fDelete) + { + unsigned i = pState->cAltDeletes; + while (i-- > 0) + { + rc = RegOpenKeyExW(pState->aAltDeletes[i].hkeyClasses, L"AppID", 0 /*fOptions*/, pState->fSamDelete, &hkeyAppIds); + AssertLogRelMsgStmt(rc == ERROR_SUCCESS || rc == ERROR_FILE_NOT_FOUND, ("%u\n", rc), pState->rc = rc); + if (rc == ERROR_SUCCESS) + { + vbpsDeleteKeyRecursiveA(pState, hkeyAppIds, pszAppId, __LINE__); + vbpsCloseKey(pState, hkeyAppIds, __LINE__); + } + } + } + + if (pState->fUpdate) + { + rc = RegCreateKeyExW(pState->hkeyClassesRootDst, L"AppID", 0 /*Reserved*/, NULL /*pszClass*/, 0 /*fOptions*/, + pState->fSamBoth, NULL /*pSecAttr*/, &hkeyAppIds, NULL /*pdwDisposition*/); + if (rc == ERROR_ACCESS_DENIED) + return ERROR_SUCCESS; + } + else + { + rc = RegOpenKeyExW(pState->hkeyClassesRootDst, L"AppID", 0 /*fOptions*/, pState->fSamBoth, &hkeyAppIds); + if (rc == ERROR_FILE_NOT_FOUND || rc == ERROR_ACCESS_DENIED) + return ERROR_SUCCESS; + } + if (rc == ERROR_ACCESS_DENIED) + return pState->rc = rc; + AssertLogRelMsgReturn(rc == ERROR_SUCCESS, ("%u\n", rc), pState->rc = rc); + + if (pState->fDelete) + { + vbpsDeleteKeyRecursiveA(pState, hkeyAppIds, pszAppId, __LINE__); + vbpsDeleteKeyRecursiveA(pState, hkeyAppIds, pszModuleName, __LINE__); + } + + /* + * Register / update. + */ + if (pState->fUpdate) + { + HKEY hkey; + rc = vbpsCreateRegKeyA(pState, hkeyAppIds, pszAppId, &hkey, __LINE__); + if (rc == ERROR_SUCCESS) + { + vbpsSetRegValueAA(pState, hkey, NULL /*pszValueNm*/, pszDescription, __LINE__); + if (pszServiceName) + vbpsSetRegValueAA(pState, hkey, "LocalService", pszServiceName, __LINE__); + vbpsCloseKey(pState, hkey, __LINE__); + } + + rc = vbpsCreateRegKeyA(pState, hkeyAppIds, pszModuleName, &hkey, __LINE__); + if (rc == ERROR_SUCCESS) + { + vbpsSetRegValueAA(pState, hkey, NULL /*pszValueNm*/, "", __LINE__); + vbpsSetRegValueAA(pState, hkey, "AppID", pszAppId, __LINE__); + vbpsCloseKey(pState, hkey, __LINE__); + } + } + + vbpsCloseKey(pState, hkeyAppIds, __LINE__); + + return pState->rc; +} + + +/** + * Register an class name. + * + * @returns Windows error code (errors are rememberd in the state). + * @param pState The registry modifier state. + * @param pszClassName The name of the class. + * @param pszDescription The description string + * @param pClsId The UUID for the class. + * @param pszCurVerSuffIfRootName This is the current version suffix to + * append to @a pszClassName when + * registering the version idependent name. + */ +LSTATUS VbpsRegisterClassName(VBPSREGSTATE *pState, const char *pszClassName, const char *pszDescription, + const CLSID *pClsId, const char *pszCurVerSuffIfRootName) +{ + LSTATUS rc; + + /* + * Delete. + */ + if (pState->fDelete) + { + unsigned i = pState->cAltDeletes; + while (i-- > 0) + vbpsDeleteKeyRecursiveA(pState, pState->aAltDeletes[i].hkeyClasses, pszClassName, __LINE__); + vbpsDeleteKeyRecursiveA(pState, pState->hkeyClassesRootDst, pszClassName, __LINE__); + } + + /* + * Update. + */ + if (pState->fUpdate) + { + /* pszClassName/Default = description. */ + HKEY hkeyClass; + rc = vbpsCreateRegKeyWithDefaultValueAAEx(pState, pState->hkeyClassesRootDst, pszClassName, pszDescription, + &hkeyClass, __LINE__); + if (rc == ERROR_SUCCESS) + { + char szClsId[CURLY_UUID_STR_BUF_SIZE]; + + /* CLSID/Default = pClsId. */ + vbpsCreateRegKeyWithDefaultValueAA(pState, hkeyClass, "CLSID", vbpsFormatUuidInCurly(szClsId, pClsId), __LINE__); + + /* CurVer/Default = pszClassName+Suffix. */ + if (pszCurVerSuffIfRootName != NULL) + { + char szCurClassNameVer[128]; + rc = RTStrCopy(szCurClassNameVer, sizeof(szCurClassNameVer), pszClassName); + if (RT_SUCCESS(rc)) + rc = RTStrCat(szCurClassNameVer, sizeof(szCurClassNameVer), pszCurVerSuffIfRootName); + AssertStmt(RT_SUCCESS(rc), pState->rc = rc = ERROR_INVALID_DATA); + if (rc == ERROR_SUCCESS) + vbpsCreateRegKeyWithDefaultValueAA(pState, hkeyClass, "CurVer", szCurClassNameVer, __LINE__); + } + + vbpsCloseKey(pState, hkeyClass, __LINE__); + } + } + + return pState->rc; +} + + +/** + * Registers a class ID. + * + * @returns Windows error code (errors are rememberd in the state). + * @param pState The registry modifier state. + * @param pClsId The UUID for the class. + * @param pszDescription The description string. + * @param pszAppId The application ID. + * @param pszClassName The version idependent class name. + * @param pszCurClassNameVerSuffix The suffix to add to @a pszClassName for + * the current version. + * @param pTypeLibId The UUID for the typelib this class + * belongs to. + * @param pszServerType The server type (InprocServer32 or + * LocalServer32). + * @param pwszVBoxDir The VirtualBox install directory + * (unicode), trailing slash. + * @param pszServerSubPath What to append to @a pwszVBoxDir to + * construct the server module name. + * @param pszThreadingModel The threading model for inproc servers, + * NULL for local servers. + */ +LSTATUS VbpsRegisterClassId(VBPSREGSTATE *pState, const CLSID *pClsId, const char *pszDescription, const char *pszAppId, + const char *pszClassName, const char *pszCurClassNameVerSuffix, const CLSID *pTypeLibId, + const char *pszServerType, PCRTUTF16 pwszVBoxDir, const char *pszServerSubPath, + const char *pszThreadingModel) +{ + LSTATUS rc; + char szClsId[CURLY_UUID_STR_BUF_SIZE]; + RT_NOREF(pszAppId); + + Assert(!pszAppId || *pszAppId == '{'); + Assert((pwszVBoxDir == NULL && !pState->fUpdate) || pwszVBoxDir[RTUtf16Len(pwszVBoxDir) - 1] == '\\'); + + /* + * We need this, whatever we end up having to do. + */ + vbpsFormatUuidInCurly(szClsId, pClsId); + + /* + * Delete. + */ + if (pState->fDelete) + { + unsigned i = pState->cAltDeletes; + while (i-- > 0) + if (pState->aAltDeletes[i].hkeyClsid != NULL) + vbpsDeleteKeyRecursiveA(pState, pState->aAltDeletes[i].hkeyClsid, szClsId, __LINE__); + vbpsDeleteKeyRecursiveA(pState, pState->hkeyClsidRootDst, szClsId, __LINE__); + } + + /* + * Update. + */ + if (pState->fUpdate) + { + HKEY hkeyClass; + rc = vbpsCreateRegKeyWithDefaultValueAAEx(pState, pState->hkeyClsidRootDst, szClsId, pszDescription, + &hkeyClass, __LINE__); + if (rc == ERROR_SUCCESS) + { + bool const fIsLocalServer32 = strcmp(pszServerType, "LocalServer32") == 0; + HKEY hkeyServerType; + char szCurClassNameVer[128]; + + /* pszServerType/Default = module. */ + rc = vbpsCreateRegKeyA(pState, hkeyClass, pszServerType, &hkeyServerType, __LINE__); + if (rc == ERROR_SUCCESS) + { + RTUTF16 wszModule[MAX_PATH * 2]; + PRTUTF16 pwszCur = wszModule; + if (fIsLocalServer32) + *pwszCur++ = '"'; + + rc = RTUtf16Copy(pwszCur, MAX_PATH, pwszVBoxDir); AssertRC(rc); + pwszCur += RTUtf16Len(pwszCur); + rc = RTUtf16CopyAscii(pwszCur, MAX_PATH - 3, pszServerSubPath); AssertRC(rc); + pwszCur += RTUtf16Len(pwszCur); + + if (fIsLocalServer32) + *pwszCur++ = '"'; + *pwszCur++ = '\0'; /* included, so ++. */ + + vbpsSetRegValueWW(pState, hkeyServerType, NULL /*pszValueNm*/, wszModule, __LINE__); + + /* pszServerType/ThreadingModel = pszThreading Model. */ + if (pszThreadingModel) + vbpsSetRegValueAA(pState, hkeyServerType, "ThreadingModel", pszThreadingModel, __LINE__); + + vbpsCloseKey(pState, hkeyServerType, __LINE__); + } + + /* ProgId/Default = pszClassName + pszCurClassNameVerSuffix. */ + if (pszClassName) + { + rc = RTStrCopy(szCurClassNameVer, sizeof(szCurClassNameVer), pszClassName); + if (RT_SUCCESS(rc)) + rc = RTStrCat(szCurClassNameVer, sizeof(szCurClassNameVer), pszCurClassNameVerSuffix); + AssertStmt(RT_SUCCESS(rc), pState->rc = rc = ERROR_INVALID_DATA); + if (rc == ERROR_SUCCESS) + vbpsCreateRegKeyWithDefaultValueAA(pState, hkeyClass, "ProgId", szCurClassNameVer, __LINE__); + + /* VersionIndependentProgID/Default = pszClassName. */ + vbpsCreateRegKeyWithDefaultValueAA(pState, hkeyClass, "VersionIndependentProgID", pszClassName, __LINE__); + } + + /* TypeLib/Default = pTypeLibId. */ + if (pTypeLibId) + { + char szTypeLibId[CURLY_UUID_STR_BUF_SIZE]; + vbpsCreateRegKeyWithDefaultValueAA(pState, hkeyClass, "TypeLib", + vbpsFormatUuidInCurly(szTypeLibId, pTypeLibId), __LINE__); + } + + /* AppID = pszAppId */ + if (pszAppId && fIsLocalServer32) + vbpsSetRegValueAA(pState, hkeyClass, "AppID", pszAppId, __LINE__); + + vbpsCloseKey(pState, hkeyClass, __LINE__); + } + } + + return pState->rc; +} + + +/** + * Register modules and classes from the VirtualBox.xidl file. + * + * @returns COM status code. + * @param pState + * @param pwszVBoxDir The VirtualBox application directory. + * @param fIs32On64 Set if this is the 32-bit on 64-bit component. + * + * @todo convert to XSLT. + */ +void RegisterXidlModulesAndClassesGenerated(VBPSREGSTATE *pState, PCRTUTF16 pwszVBoxDir, bool fIs32On64) +{ + const char *pszAppId = "{819B4D85-9CEE-493C-B6FC-64FFE759B3C9}"; + const char *pszInprocDll = !fIs32On64 ? "VBoxC.dll" : "x86\\VBoxClient-x86.dll"; + const char *pszLocalServer = "VBoxSVC.exe"; +#ifdef VBOX_WITH_SDS + const char *pszSdsAppId = "{EC0E78E8-FA43-43E8-AC0A-02C784C4A4FA}"; + const char *pszSdsExe = "VBoxSDS.exe"; + const char *pszSdsServiceName = "VBoxSDS"; +#endif + + /* VBoxSVC */ + VbpsRegisterAppId(pState, pszLocalServer, pszAppId, "VirtualBox Application", NULL); + VbpsRegisterClassName(pState, "VirtualBox.VirtualBox.1", "VirtualBox Class", &CLSID_VirtualBox, NULL); + VbpsRegisterClassName(pState, "VirtualBox.VirtualBox", "VirtualBox Class", &CLSID_VirtualBox, ".1"); + VbpsRegisterClassId(pState, &CLSID_VirtualBox, "VirtualBox Class", pszAppId, "VirtualBox.VirtualBox", ".1", + &LIBID_VirtualBox, "LocalServer32", pwszVBoxDir, pszLocalServer, NULL /*N/A*/); + /* VBoxC */ + VbpsRegisterClassName(pState, "VirtualBox.Session.1", "Session Class", &CLSID_Session, NULL); + VbpsRegisterClassName(pState, "VirtualBox.Session", "Session Class", &CLSID_Session, ".1"); + VbpsRegisterClassId(pState, &CLSID_Session, "Session Class", pszAppId, "VirtualBox.Session", ".1", + &LIBID_VirtualBox, "InprocServer32", pwszVBoxDir, pszInprocDll, "Free"); + + VbpsRegisterClassName(pState, "VirtualBox.VirtualBoxClient.1", "VirtualBoxClient Class", &CLSID_VirtualBoxClient, NULL); + VbpsRegisterClassName(pState, "VirtualBox.VirtualBoxClient", "VirtualBoxClient Class", &CLSID_VirtualBoxClient, ".1"); + VbpsRegisterClassId(pState, &CLSID_VirtualBoxClient, "VirtualBoxClient Class", pszAppId, + "VirtualBox.VirtualBoxClient", ".1", + &LIBID_VirtualBox, "InprocServer32", pwszVBoxDir, pszInprocDll, "Free"); + +#ifdef VBOX_WITH_SDS + /* VBoxSDS */ + VbpsRegisterAppId(pState, pszSdsExe, pszSdsAppId, "VirtualBox System Service", pszSdsServiceName); + VbpsRegisterClassName(pState, "VirtualBox.VirtualBoxSDS.1", "VirtualBoxSDS Class", &CLSID_VirtualBoxSDS, NULL); + VbpsRegisterClassName(pState, "VirtualBox.VirtualBoxSDS", "VirtualBoxSDS Class", &CLSID_VirtualBoxSDS, ".1"); + VbpsRegisterClassId(pState, &CLSID_VirtualBoxSDS, "VirtualBoxSDS Class", pszSdsAppId, "VirtualBox.VirtualBoxSDS", ".1", + &LIBID_VirtualBox, "LocalServer32", pwszVBoxDir, pszSdsExe, NULL /*N/A*/); +#endif +} + + +/** + * Updates the VBox type lib registration. + * + * This is only used when updating COM registrations during com::Initialize. + * For normal registration and unregistrations we use the RegisterTypeLib and + * UnRegisterTypeLib APIs. + * + * @param pState The registry modifier state. + * @param pwszVBoxDir The VirtualBox install directory (unicode), + * trailing slash. + * @param fIs32On64 Set if we're registering the 32-bit proxy stub + * on a 64-bit system. + */ +static void vbpsUpdateTypeLibRegistration(VBPSREGSTATE *pState, PCRTUTF16 pwszVBoxDir, bool fIs32On64) +{ + const char * const pszTypeLibDll = VBPS_PROXY_STUB_FILE(fIs32On64); +#if ARCH_BITS == 32 && !defined(VBOX_IN_32_ON_64_MAIN_API) + const char * const pszWinXx = "win32"; +#else + const char * const pszWinXx = !fIs32On64 ? "win64" : "win32"; +#endif + const char * const pszDescription = "VirtualBox Type Library"; + + char szTypeLibId[CURLY_UUID_STR_BUF_SIZE]; + HKEY hkeyTypeLibs; + HKEY hkeyTypeLibId; + LSTATUS rc; + RT_NOREF(fIs32On64); + + Assert(pState->fUpdate && !pState->fDelete); + + /* + * Type library registration (w/o interfaces). + */ + + /* Open Classes/TypeLib/. */ + rc = vbpsCreateRegKeyA(pState, pState->hkeyClassesRootDst, "TypeLib", &hkeyTypeLibs, __LINE__); + if (rc != ERROR_SUCCESS) + return; + + /* Create TypeLib/{UUID}. */ + rc = vbpsCreateRegKeyA(pState, hkeyTypeLibs, vbpsFormatUuidInCurly(szTypeLibId, &LIBID_VirtualBox), &hkeyTypeLibId, __LINE__); + if (rc == ERROR_SUCCESS) + { + /* {UUID}/Major.Minor/Default = pszDescription. */ + HKEY hkeyMajMin; + char szMajMin[64]; + sprintf(szMajMin, "%u.%u", kTypeLibraryMajorVersion, kTypeLibraryMinorVersion); + rc = vbpsCreateRegKeyWithDefaultValueAAEx(pState, hkeyTypeLibId, szMajMin, pszDescription, &hkeyMajMin, __LINE__); + if (rc == ERROR_SUCCESS) + { + RTUTF16 wszBuf[MAX_PATH * 2]; + + /* {UUID}/Major.Minor/0. */ + HKEY hkey0; + rc = vbpsCreateRegKeyA(pState, hkeyMajMin, "0", &hkey0, __LINE__); + if (rc == ERROR_SUCCESS) + { + /* {UUID}/Major.Minor/0/winXX/Default = VBoxProxyStub. */ + rc = RTUtf16Copy(wszBuf, MAX_PATH, pwszVBoxDir); AssertRC(rc); + rc = RTUtf16CatAscii(wszBuf, MAX_PATH * 2, pszTypeLibDll); AssertRC(rc); + + vbpsCreateRegKeyWithDefaultValueAW(pState, hkey0, pszWinXx, wszBuf, __LINE__); + vbpsCloseKey(pState, hkey0, __LINE__); + } + + /* {UUID}/Major.Minor/FLAGS */ + vbpsCreateRegKeyWithDefaultValueAA(pState, hkeyMajMin, "FLAGS", "0", __LINE__); + + /* {UUID}/Major.Minor/HELPDIR */ + rc = RTUtf16Copy(wszBuf, MAX_PATH, pwszVBoxDir); AssertRC(rc); +#if 0 /* MSI: trailing slash; regsvr32/comregister: strip unnecessary trailing slash. Go with MSI to avoid user issues. */ + { + size_t off = RTUtf16Len(wszBuf); + while (off > 2 && wszBuf[off - 2] != ':' && RTPATH_IS_SLASH(wszBuf[off - 1])) + off--; + wszBuf[off] = '\0'; + } +#endif + vbpsCreateRegKeyWithDefaultValueAW(pState, hkeyMajMin, "HELPDIR", wszBuf, __LINE__); + + vbpsCloseKey(pState, hkeyMajMin, __LINE__); + } + vbpsCloseKey(pState, hkeyTypeLibId, __LINE__); + } + vbpsCloseKey(pState, hkeyTypeLibs, __LINE__); +} + + +/** + * Update the VBox proxy stub registration. + * + * This is only used when updating COM registrations during com::Initialize. + * For normal registration and unregistrations we use the NdrDllRegisterProxy + * and NdrDllUnregisterProxy. + * + * @param pState The registry modifier state. + * @param pwszVBoxDir The VirtualBox install directory (unicode), + * trailing slash. + * @param fIs32On64 Set if we're registering the 32-bit proxy stub + * on a 64-bit system. + */ +static void vbpsUpdateProxyStubRegistration(VBPSREGSTATE *pState, PCRTUTF16 pwszVBoxDir, bool fIs32On64) +{ + /* + * Register the proxy stub factory class ID. + * It's simple compared to the VBox classes, thus all the NULL parameters. + */ + const char *pszPsDll = VBPS_PROXY_STUB_FILE(fIs32On64); + RT_NOREF(fIs32On64); + Assert(pState->fUpdate && !pState->fDelete); + VbpsRegisterClassId(pState, &g_ProxyClsId, "PSFactoryBuffer", NULL /*pszAppId*/, + NULL /*pszClassName*/, NULL /*pszCurClassNameVerSuffix*/, NULL /*pTypeLibId*/, + "InprocServer32", pwszVBoxDir, pszPsDll, "Both"); +} + + +/** + * Updates the VBox interface registrations. + * + * This is only used when updating COM registrations during com::Initialize. + * For normal registration and unregistrations we use the NdrDllRegisterProxy + * and NdrDllUnregisterProxy. + * + * @param pState The registry modifier state. + */ +static void vbpsUpdateInterfaceRegistrations(VBPSREGSTATE *pState) +{ + const ProxyFileInfo **ppProxyFile = &g_apProxyFiles[0]; + const ProxyFileInfo *pProxyFile; + LSTATUS rc; + char szProxyClsId[CURLY_UUID_STR_BUF_SIZE]; + char szTypeLibId[CURLY_UUID_STR_BUF_SIZE]; + char szTypeLibVersion[64]; + + vbpsFormatUuidInCurly(szProxyClsId, &g_ProxyClsId); + vbpsFormatUuidInCurly(szTypeLibId, &LIBID_VirtualBox); + sprintf(szTypeLibVersion, "%u.%u", kTypeLibraryMajorVersion, kTypeLibraryMinorVersion); + + Assert(pState->fUpdate && !pState->fDelete); + rc = vbpsRegOpenInterfaceKeys(pState); + if (rc != ERROR_SUCCESS) + return; + + /* + * We walk the proxy file list (even if we only have one). + */ + while ((pProxyFile = *ppProxyFile++) != NULL) + { + const PCInterfaceStubVtblList * const papStubVtbls = pProxyFile->pStubVtblList; + const char * const *papszNames = pProxyFile->pNamesArray; + unsigned iIf = pProxyFile->TableSize; + AssertStmt(iIf < 1024, iIf = 0); + Assert(pProxyFile->TableVersion == 2); + + /* + * Walk the interfaces in that file, picking data from the various tables. + */ + while (iIf-- > 0) + { + char szIfId[CURLY_UUID_STR_BUF_SIZE]; + const char * const pszIfNm = papszNames[iIf]; + size_t const cchIfNm = RT_VALID_PTR(pszIfNm) ? strlen(pszIfNm) : 0; + char szMethods[32]; + uint32_t const cMethods = papStubVtbls[iIf]->header.DispatchTableCount; + HKEY hkeyIfId; + + AssertReturnVoidStmt(cchIfNm >= 3 && cchIfNm <= 72, pState->rc = ERROR_INVALID_DATA); + + AssertReturnVoidStmt(cMethods >= 3 && cMethods < 1024, pState->rc = ERROR_INVALID_DATA); + sprintf(szMethods, "%u", cMethods); + + rc = vbpsCreateRegKeyWithDefaultValueAAEx(pState, pState->hkeyInterfaceRootDst, + vbpsFormatUuidInCurly(szIfId, papStubVtbls[iIf]->header.piid), + pszIfNm, &hkeyIfId, __LINE__); + if (rc == ERROR_SUCCESS) + { + HKEY hkeyTypeLib; + vbpsCreateRegKeyWithDefaultValueAA(pState, hkeyIfId, "ProxyStubClsid32", szProxyClsId, __LINE__); + vbpsCreateRegKeyWithDefaultValueAA(pState, hkeyIfId, "NumMethods", szMethods, __LINE__); + + /* The MSI seems to still be putting TypeLib keys here. So, let's do that too. */ + rc = vbpsCreateRegKeyWithDefaultValueAAEx(pState, hkeyIfId, "TypeLib", szTypeLibId, &hkeyTypeLib, __LINE__); + if (rc == ERROR_SUCCESS) + { + vbpsSetRegValueAA(pState, hkeyTypeLib, "Version", szTypeLibVersion, __LINE__); + vbpsCloseKey(pState, hkeyTypeLib, __LINE__); + } + + vbpsCloseKey(pState, hkeyIfId, __LINE__); + } + } + } +} + + +static bool vbpsIsUpToDate(VBPSREGSTATE *pState) +{ + /** @todo read some registry key and */ + NOREF(pState); + return false; +} + +static bool vbpsMarkUpToDate(VBPSREGSTATE *pState) +{ + /** @todo write the key vbpsIsUpToDate uses, if pState indicates success. */ + NOREF(pState); + return false; +} + + + +/** + * Strips the stub dll name and any x86 subdir off the full DLL path to get a + * path to the VirtualBox application directory. + * + * @param pwszDllPath The path to strip, returns will end with a slash. + */ +static void vbpsDllPathToVBoxDir(PRTUTF16 pwszDllPath) +{ + RTUTF16 wc; + size_t off = RTUtf16Len(pwszDllPath); + while ( off > 0 + && ( (wc = pwszDllPath[off - 1]) >= 127U + || !RTPATH_IS_SEP((unsigned char)wc))) + off--; + +#ifdef VBOX_IN_32_ON_64_MAIN_API + /* + * The -x86 variant is in a x86 subdirectory, drop it. + */ + while ( off > 0 + && ( (wc = pwszDllPath[off - 1]) < 127U + && RTPATH_IS_SEP((unsigned char)wc))) + off--; + while ( off > 0 + && ( (wc = pwszDllPath[off - 1]) >= 127U + || !RTPATH_IS_SEP((unsigned char)wc))) + off--; +#endif + pwszDllPath[off] = '\0'; +} + + +/** + * Wrapper around RegisterXidlModulesAndClassesGenerated for the convenience of + * the standard registration entry points. + * + * @returns COM status code. + * @param pwszVBoxDir The VirtualBox install directory (unicode), + * trailing slash. + * @param fDelete Whether to delete registration keys and values. + * @param fUpdate Whether to update registration keys and values. + */ +HRESULT RegisterXidlModulesAndClasses(PRTUTF16 pwszVBoxDir, bool fDelete, bool fUpdate) +{ +#ifdef VBOX_IN_32_ON_64_MAIN_API + bool const fIs32On64 = true; +#else + bool const fIs32On64 = false; +#endif + VBPSREGSTATE State; + LSTATUS rc; + + /* + * Do registration for the current execution mode of the DLL. + */ + rc = vbpsRegInit(&State, HKEY_CLASSES_ROOT, NULL /* Alt: HKEY_LOCAL_MACHINE, "Software\\Classes", */, fDelete, fUpdate, 0); + if (rc == ERROR_SUCCESS) + { + if (!fUpdate) + { + /* When only unregistering, really purge everything twice or trice. :-) */ + vbpsRegAddAltDelete(&State, HKEY_LOCAL_MACHINE, "Software\\Classes"); + vbpsRegAddAltDelete(&State, HKEY_CURRENT_USER, "Software\\Classes"); + vbpsRegAddAltDelete(&State, HKEY_CLASSES_ROOT, NULL); + } + + RegisterXidlModulesAndClassesGenerated(&State, pwszVBoxDir, fIs32On64); + rc = State.rc; + } + vbpsRegTerm(&State); + + /* + * Translate error code? Return. + */ + if (rc == ERROR_SUCCESS) + return S_OK; + return E_FAIL; +} + + +/** + * Checks if the string matches any of our type library versions. + * + * @returns true on match, false on mismatch. + * @param pwszTypeLibVersion The type library version string. + */ +DECLINLINE(bool) vbpsIsTypeLibVersionToRemove(PCRTUTF16 pwszTypeLibVersion) +{ + AssertCompile(RT_ELEMENTS(g_apwszTypelibVersions) == 2); + + /* ASSUMES: 1.x version strings and that the input buffer is at least 3 wchars long. */ + if ( g_apwszTypelibVersions[0][3] == pwszTypeLibVersion[3] + && RTUtf16Cmp(g_apwszTypelibVersions[0], pwszTypeLibVersion) == 0) + return true; + if ( g_apwszTypelibVersions[1][3] == pwszTypeLibVersion[3] + && RTUtf16Cmp(g_apwszTypelibVersions[1], pwszTypeLibVersion) == 0) + return true; + + return false; +} + + +/** + * Quick check whether the given string looks like a UUID in braces. + * + * This does not check the whole string, just do a quick sweep. + * + * @returns true if possible UUID, false if definitely not. + * @param pwszUuid Alleged UUID in braces. + */ +DECLINLINE(bool) vbpsIsUuidInBracesQuickW(PCRTUTF16 pwszUuid) +{ + return pwszUuid[ 0] == '{' + && pwszUuid[ 9] == '-' + && pwszUuid[14] == '-' + && pwszUuid[19] == '-' + && pwszUuid[24] == '-' + && pwszUuid[37] == '}' + && pwszUuid[38] == '\0' + && RT_C_IS_XDIGIT(pwszUuid[1]); +} + + +/** + * Compares two UUIDs (in braces). + * + * @returns true on match, false if no match. + * @param pwszUuid1 The first UUID. + * @param pwszUuid2 The second UUID. + */ +static bool vbpsCompareUuidW(PCRTUTF16 pwszUuid1, PCRTUTF16 pwszUuid2) +{ +#define COMPARE_EXACT_RET(a_wch1, a_wch2) \ + if ((a_wch1) == (a_wch2)) { } else return false + +#define COMPARE_XDIGITS_RET(a_wch1, a_wch2) \ + if ((a_wch1) == (a_wch2)) { } \ + else if (RT_C_TO_UPPER(a_wch1) != RT_C_TO_UPPER(a_wch2) || (a_wch1) >= 127U || (a_wch2) >= 127U) \ + return false + COMPARE_EXACT_RET( pwszUuid1[ 0], pwszUuid2[ 0]); /* { */ + COMPARE_XDIGITS_RET(pwszUuid1[ 1], pwszUuid2[ 1]); /* 5 */ + COMPARE_XDIGITS_RET(pwszUuid1[ 2], pwszUuid2[ 2]); /* e */ + COMPARE_XDIGITS_RET(pwszUuid1[ 3], pwszUuid2[ 3]); /* 5 */ + COMPARE_XDIGITS_RET(pwszUuid1[ 4], pwszUuid2[ 4]); /* e */ + COMPARE_XDIGITS_RET(pwszUuid1[ 5], pwszUuid2[ 5]); /* 3 */ + COMPARE_XDIGITS_RET(pwszUuid1[ 6], pwszUuid2[ 6]); /* 6 */ + COMPARE_XDIGITS_RET(pwszUuid1[ 7], pwszUuid2[ 7]); /* 4 */ + COMPARE_XDIGITS_RET(pwszUuid1[ 8], pwszUuid2[ 8]); /* 0 */ + COMPARE_EXACT_RET( pwszUuid1[ 9], pwszUuid2[ 9]); /* - */ + COMPARE_XDIGITS_RET(pwszUuid1[10], pwszUuid2[10]); /* 7 */ + COMPARE_XDIGITS_RET(pwszUuid1[11], pwszUuid2[11]); /* 4 */ + COMPARE_XDIGITS_RET(pwszUuid1[12], pwszUuid2[12]); /* f */ + COMPARE_XDIGITS_RET(pwszUuid1[13], pwszUuid2[13]); /* 3 */ + COMPARE_EXACT_RET( pwszUuid1[14], pwszUuid2[14]); /* - */ + COMPARE_XDIGITS_RET(pwszUuid1[15], pwszUuid2[15]); /* 4 */ + COMPARE_XDIGITS_RET(pwszUuid1[16], pwszUuid2[16]); /* 6 */ + COMPARE_XDIGITS_RET(pwszUuid1[17], pwszUuid2[17]); /* 8 */ + COMPARE_XDIGITS_RET(pwszUuid1[18], pwszUuid2[18]); /* 9 */ + COMPARE_EXACT_RET( pwszUuid1[19], pwszUuid2[19]); /* - */ + COMPARE_XDIGITS_RET(pwszUuid1[20], pwszUuid2[20]); /* 9 */ + COMPARE_XDIGITS_RET(pwszUuid1[21], pwszUuid2[21]); /* 7 */ + COMPARE_XDIGITS_RET(pwszUuid1[22], pwszUuid2[22]); /* 9 */ + COMPARE_XDIGITS_RET(pwszUuid1[23], pwszUuid2[23]); /* f */ + COMPARE_EXACT_RET( pwszUuid1[24], pwszUuid2[24]); /* - */ + COMPARE_XDIGITS_RET(pwszUuid1[25], pwszUuid2[25]); /* 6 */ + COMPARE_XDIGITS_RET(pwszUuid1[26], pwszUuid2[26]); /* b */ + COMPARE_XDIGITS_RET(pwszUuid1[27], pwszUuid2[27]); /* 1 */ + COMPARE_XDIGITS_RET(pwszUuid1[28], pwszUuid2[28]); /* b */ + COMPARE_XDIGITS_RET(pwszUuid1[29], pwszUuid2[29]); /* 8 */ + COMPARE_XDIGITS_RET(pwszUuid1[30], pwszUuid2[30]); /* d */ + COMPARE_XDIGITS_RET(pwszUuid1[31], pwszUuid2[31]); /* 7 */ + COMPARE_XDIGITS_RET(pwszUuid1[32], pwszUuid2[32]); /* 6 */ + COMPARE_XDIGITS_RET(pwszUuid1[33], pwszUuid2[33]); /* 0 */ + COMPARE_XDIGITS_RET(pwszUuid1[34], pwszUuid2[34]); /* 9 */ + COMPARE_XDIGITS_RET(pwszUuid1[35], pwszUuid2[35]); /* a */ + COMPARE_XDIGITS_RET(pwszUuid1[36], pwszUuid2[36]); /* 5 */ + COMPARE_EXACT_RET( pwszUuid1[37], pwszUuid2[37]); /* } */ + COMPARE_EXACT_RET( pwszUuid1[38], pwszUuid2[38]); /* \0 */ +#undef COMPARE_EXACT_RET +#undef COMPARE_XDIGITS_RET + return true; +} + + +/** + * Checks if the type library ID is one of the ones we wish to clean up. + * + * @returns true if it should be cleaned up, false if not. + * @param pwszTypeLibId The type library ID as a bracketed string. + */ +DECLINLINE(bool) vbpsIsTypeLibIdToRemove(PRTUTF16 pwszTypeLibId) +{ + AssertCompile(RT_ELEMENTS(g_apwszTypeLibIds) == 2); +#ifdef VBOX_STRICT + static bool s_fDoneStrict = false; + if (s_fDoneStrict) { } + else + { + Assert(RT_ELEMENTS(g_apwszTypeLibIds) == 2); + Assert(g_apwszTypeLibIds[0][0] == '{'); + Assert(g_apwszTypeLibIds[1][0] == '{'); + Assert(RT_C_IS_XDIGIT(g_apwszTypeLibIds[0][1])); + Assert(RT_C_IS_XDIGIT(g_apwszTypeLibIds[1][1])); + Assert(RT_C_IS_UPPER(g_apwszTypeLibIds[0][1]) || RT_C_IS_DIGIT(g_apwszTypeLibIds[0][1])); + Assert(RT_C_IS_UPPER(g_apwszTypeLibIds[1][1]) || RT_C_IS_DIGIT(g_apwszTypeLibIds[1][1])); + s_fDoneStrict = true; + } +#endif + + /* + * Rolled out matching with inlined check of the opening braces + * and first two digits. + * + * ASSUMES input buffer is at least 3 wchars big and uppercased UUID in + * our matching array. + */ + if (pwszTypeLibId[0] == '{') + { + RTUTF16 const wcFirstDigit = RT_C_TO_UPPER(pwszTypeLibId[1]); + RTUTF16 const wcSecondDigit = RT_C_TO_UPPER(pwszTypeLibId[2]); + PCRTUTF16 pwsz2 = g_apwszTypeLibIds[0]; + if ( wcFirstDigit == pwsz2[1] + && wcSecondDigit == pwsz2[2] + && vbpsCompareUuidW(pwszTypeLibId, pwsz2)) + return true; + pwsz2 = g_apwszTypeLibIds[1]; + if ( wcFirstDigit == pwsz2[1] + && wcSecondDigit == pwsz2[2] + && vbpsCompareUuidW(pwszTypeLibId, pwsz2)) + return true; + } + return false; +} + + +/** + * Checks if the proxy stub class ID is one of the ones we wish to clean up. + * + * @returns true if it should be cleaned up, false if not. + * @param pwszProxyStubId The proxy stub class ID. + */ +DECLINLINE(bool) vbpsIsProxyStubClsIdToRemove(PRTUTF16 pwszProxyStubId) +{ + AssertCompile(RT_ELEMENTS(g_apwszProxyStubClsIds) == 2); +#ifdef VBOX_STRICT + static bool s_fDoneStrict = false; + if (s_fDoneStrict) { } + else + { + Assert(RT_ELEMENTS(g_apwszProxyStubClsIds) == 2); + Assert(g_apwszProxyStubClsIds[0][0] == '{'); + Assert(g_apwszProxyStubClsIds[1][0] == '{'); + Assert(RT_C_IS_XDIGIT(g_apwszProxyStubClsIds[0][1])); + Assert(RT_C_IS_XDIGIT(g_apwszProxyStubClsIds[1][1])); + Assert(RT_C_IS_UPPER(g_apwszProxyStubClsIds[0][1]) || RT_C_IS_DIGIT(g_apwszProxyStubClsIds[0][1])); + Assert(RT_C_IS_UPPER(g_apwszProxyStubClsIds[1][1]) || RT_C_IS_DIGIT(g_apwszProxyStubClsIds[1][1])); + s_fDoneStrict = true; + } +#endif + + /* + * Rolled out matching with inlined check of the opening braces + * and first two digits. + * + * ASSUMES input buffer is at least 3 wchars big and uppercased UUID in + * our matching array. + */ + if (pwszProxyStubId[0] == '{') + { + RTUTF16 const wcFirstDigit = RT_C_TO_UPPER(pwszProxyStubId[1]); + RTUTF16 const wcSecondDigit = RT_C_TO_UPPER(pwszProxyStubId[2]); + PCRTUTF16 pwsz2 = g_apwszProxyStubClsIds[0]; + if ( wcFirstDigit == pwsz2[1] + && wcSecondDigit == pwsz2[2] + && vbpsCompareUuidW(pwszProxyStubId, pwsz2)) + return true; + pwsz2 = g_apwszProxyStubClsIds[1]; + if ( wcFirstDigit == pwsz2[1] + && wcSecondDigit == pwsz2[2] + && vbpsCompareUuidW(pwszProxyStubId, pwsz2)) + return true; + } + return false; +} + + +/** + * Hack to clean out the interfaces belonging to obsolete typelibs on + * development boxes and such likes. + */ +static void vbpsRemoveOldInterfaces(VBPSREGSTATE *pState) +{ + unsigned iAlt = pState->cAltDeletes; + while (iAlt-- > 0) + { + /* + * Open the interface root key. Not using the vbpsRegOpenInterfaceKeys feature + * here in case it messes things up by keeping the special HKEY_CLASSES_ROOT key + * open with possibly pending deletes in parent views or other weird stuff. + */ + HKEY hkeyInterfaces; + LRESULT rc = RegOpenKeyExW(pState->aAltDeletes[iAlt].hkeyClasses, L"Interface", + 0 /*fOptions*/, pState->fSamDelete, &hkeyInterfaces); + if (rc == ERROR_SUCCESS) + { + /* + * This is kind of expensive, but we have to check all registered interfaces. + * Only use wide APIs to avoid wasting time on string conversion. + */ + DWORD idxKey; + for (idxKey = 0;; idxKey++) + { + RTUTF16 wszCurNm[128 + 48]; + DWORD cwcCurNm = 128; + rc = RegEnumKeyExW(hkeyInterfaces, idxKey, wszCurNm, &cwcCurNm, + NULL /*pdwReserved*/, NULL /*pwszClass*/, NULL /*pcwcClass*/, NULL /*pLastWriteTime*/); + if (rc == ERROR_SUCCESS) + { + /* + * We match the interface by type library ID or proxy stub class ID. + * + * We have to check the proxy ID last, as it is almost always there + * and we can safely skip it if there is a mismatching type lib + * associated with the interface. + */ + static RTUTF16 const s_wszTypeLib[] = L"\\TypeLib"; + bool fDeleteMe = false; + HKEY hkeySub; + RTUTF16 wszValue[128]; + DWORD cbValue; + DWORD dwType; + + /* Skip this entry if it doesn't look like a braced UUID. */ + wszCurNm[cwcCurNm] = '\0'; /* paranoia */ + if (vbpsIsUuidInBracesQuickW(wszCurNm)) { } + else continue; + + /* Try the TypeLib sub-key. */ + memcpy(&wszCurNm[cwcCurNm], s_wszTypeLib, sizeof(s_wszTypeLib)); + rc = RegOpenKeyExW(hkeyInterfaces, wszCurNm, 0 /*fOptions*/, KEY_QUERY_VALUE, &hkeySub); + if (rc == ERROR_SUCCESS) + { + cbValue = sizeof(wszValue) - sizeof(RTUTF16); + rc = RegQueryValueExW(hkeySub, NULL /*pszValueNm*/, NULL /*pdwReserved*/, + &dwType, (PBYTE)&wszValue[0], &cbValue); + if (rc != ERROR_SUCCESS || dwType != REG_SZ) + cbValue = 0; + wszValue[cbValue / sizeof(RTUTF16)] = '\0'; + + if ( rc == ERROR_SUCCESS + && vbpsIsTypeLibIdToRemove(wszValue)) + { + /* Check the TypeLib/Version value to make sure. */ + cbValue = sizeof(wszValue) - sizeof(RTUTF16); + rc = RegQueryValueExW(hkeySub, L"Version", 0 /*pdwReserved*/, &dwType, (PBYTE)&wszValue[0], &cbValue); + if (rc != ERROR_SUCCESS) + cbValue = 0; + wszValue[cbValue] = '\0'; + + if ( rc == ERROR_SUCCESS + && vbpsIsTypeLibVersionToRemove(wszValue)) + fDeleteMe = true; + } + vbpsCloseKey(pState, hkeySub, __LINE__); + } + else if (rc == ERROR_FILE_NOT_FOUND) + { + /* No TypeLib, try the ProxyStubClsid32 sub-key next. */ + static RTUTF16 const s_wszProxyStubClsid32[] = L"\\ProxyStubClsid32"; + memcpy(&wszCurNm[cwcCurNm], s_wszProxyStubClsid32, sizeof(s_wszProxyStubClsid32)); + rc = RegOpenKeyExW(hkeyInterfaces, wszCurNm, 0 /*fOptions*/, KEY_QUERY_VALUE, &hkeySub); + if (rc == ERROR_SUCCESS) + { + cbValue = sizeof(wszValue) - sizeof(RTUTF16); + rc = RegQueryValueExW(hkeySub, NULL /*pszValueNm*/, NULL /*pdwReserved*/, + &dwType, (PBYTE)&wszValue[0], &cbValue); + if (rc != ERROR_SUCCESS || dwType != REG_SZ) + cbValue = 0; + wszValue[cbValue / sizeof(RTUTF16)] = '\0'; + + if ( rc == ERROR_SUCCESS + && vbpsIsProxyStubClsIdToRemove(wszValue)) + fDeleteMe = true; + + vbpsCloseKey(pState, hkeySub, __LINE__); + } + } + + if (fDeleteMe) + { + /* + * Ok, it's an orphaned VirtualBox interface. Delete it. + */ + wszCurNm[cwcCurNm] = '\0'; + vbpsDeleteKeyRecursiveW(pState, hkeyInterfaces, wszCurNm, __LINE__); + } + } + else + { + Assert(rc == ERROR_NO_MORE_ITEMS); + break; + } + } + + vbpsCloseKey(pState, hkeyInterfaces, __LINE__); + } + } +} + + +/** + * Hack to clean out the class IDs belonging to obsolete typelibs on development + * boxes and such likes. + */ +static void vbpsRemoveOldClassIDs(VBPSREGSTATE *pState) +{ + unsigned iAlt = pState->cAltDeletes; + while (iAlt-- > 0) + { + /* + * Open the CLSID key if it exists. + * We don't use the hKeyClsid member for the same paranoid reasons as + * already stated in vbpsRemoveOldInterfaces. + */ + HKEY hkeyClsIds; + LRESULT rc; + rc = RegOpenKeyExW(pState->aAltDeletes[iAlt].hkeyClasses, L"CLSID", 0 /*fOptions*/, pState->fSamDelete, &hkeyClsIds); + if (rc == ERROR_SUCCESS) + { + /* + * This is kind of expensive, but we have to check all registered interfaces. + * Only use wide APIs to avoid wasting time on string conversion. + */ + DWORD idxKey; + for (idxKey = 0;; idxKey++) + { + RTUTF16 wszCurNm[128 + 48]; + DWORD cwcCurNm = 128; + rc = RegEnumKeyExW(hkeyClsIds, idxKey, wszCurNm, &cwcCurNm, + NULL /*pdwReserved*/, NULL /*pwszClass*/, NULL /*pcwcClass*/, NULL /*pLastWriteTime*/); + if (rc == ERROR_SUCCESS) + { + /* + * Match both the type library ID and the program ID. + */ + static RTUTF16 const s_wszTypeLib[] = L"\\TypeLib"; + HKEY hkeySub; + RTUTF16 wszValue[128]; + DWORD cbValue; + DWORD dwType; + + + /* Skip this entry if it doesn't look like a braced UUID. (Microsoft + has one two malformed ones plus a hack.) */ + wszCurNm[cwcCurNm] = '\0'; /* paranoia */ + if (vbpsIsUuidInBracesQuickW(wszCurNm)) { } + else continue; + + /* The TypeLib sub-key. */ + memcpy(&wszCurNm[cwcCurNm], s_wszTypeLib, sizeof(s_wszTypeLib)); + rc = RegOpenKeyExW(hkeyClsIds, wszCurNm, 0 /*fOptions*/, KEY_QUERY_VALUE, &hkeySub); + if (rc == ERROR_SUCCESS) + { + bool fDeleteMe = false; + + cbValue = sizeof(wszValue) - sizeof(RTUTF16); + rc = RegQueryValueExW(hkeySub, NULL /*pszValueNm*/, NULL /*pdwReserved*/, + &dwType, (PBYTE)&wszValue[0], &cbValue); + if (rc != ERROR_SUCCESS || dwType != REG_SZ) + cbValue = 0; + wszValue[cbValue / sizeof(RTUTF16)] = '\0'; + + if ( rc == ERROR_SUCCESS + && vbpsIsTypeLibIdToRemove(wszValue)) + fDeleteMe = true; + + vbpsCloseKey(pState, hkeySub, __LINE__); + + if (fDeleteMe) + { + /* The ProgId sub-key. */ + static RTUTF16 const s_wszProgId[] = L"\\ProgId"; + memcpy(&wszCurNm[cwcCurNm], s_wszProgId, sizeof(s_wszProgId)); + rc = RegOpenKeyExW(hkeyClsIds, wszCurNm, 0 /*fOptions*/, KEY_QUERY_VALUE, &hkeySub); + if (rc == ERROR_SUCCESS) + { + static RTUTF16 const s_wszProgIdPrefix[] = L"VirtualBox."; + + cbValue = sizeof(wszValue) - sizeof(RTUTF16); + rc = RegQueryValueExW(hkeySub, NULL /*pszValueNm*/, NULL /*pdwReserved*/, + &dwType, (PBYTE)&wszValue[0], &cbValue); + if (rc != ERROR_SUCCESS || dwType != REG_SZ) + cbValue = 0; + wszValue[cbValue / sizeof(RTUTF16)] = '\0'; + + if ( cbValue < sizeof(s_wszProgIdPrefix) + || memcmp(wszValue, s_wszProgIdPrefix, sizeof(s_wszProgIdPrefix) - sizeof(RTUTF16)) != 0) + fDeleteMe = false; + + vbpsCloseKey(pState, hkeySub, __LINE__); + } + else + AssertStmt(rc == ERROR_FILE_NOT_FOUND, fDeleteMe = false); + + if (fDeleteMe) + { + /* + * Ok, it's an orphaned VirtualBox interface. Delete it. + */ + wszCurNm[cwcCurNm] = '\0'; + vbpsDeleteKeyRecursiveW(pState, hkeyClsIds, wszCurNm, __LINE__); + } + } + } + else + Assert(rc == ERROR_FILE_NOT_FOUND); + } + else + { + Assert(rc == ERROR_NO_MORE_ITEMS); + break; + } + } + + vbpsCloseKey(pState, hkeyClsIds, __LINE__); + } + else + Assert(rc == ERROR_FILE_NOT_FOUND); + } +} + + +/** + * Hack to clean obsolete typelibs on development boxes and such. + */ +static void vbpsRemoveOldTypeLibs(VBPSREGSTATE *pState) +{ + unsigned iAlt = pState->cAltDeletes; + while (iAlt-- > 0) + { + /* + * Open the TypeLib key, if it exists. + */ + HKEY hkeyTypeLibs; + LSTATUS rc; + rc = RegOpenKeyExW(pState->aAltDeletes[iAlt].hkeyClasses, L"TypeLib", 0 /*fOptions*/, pState->fSamDelete, &hkeyTypeLibs); + if (rc == ERROR_SUCCESS) + { + /* + * Look for our type library IDs. + */ + unsigned iTlb = RT_ELEMENTS(g_apwszTypeLibIds); + while (iTlb-- > 0) + { + HKEY hkeyTypeLibId; + rc = RegOpenKeyExW(hkeyTypeLibs, g_apwszTypeLibIds[iTlb], 0 /*fOptions*/, pState->fSamDelete, &hkeyTypeLibId); + if (rc == ERROR_SUCCESS) + { + unsigned iVer = RT_ELEMENTS(g_apwszTypelibVersions); + while (iVer-- > 0) + { + HKEY hkeyVer; + rc = RegOpenKeyExW(hkeyTypeLibId, g_apwszTypelibVersions[iVer], 0, KEY_READ, &hkeyVer); + if (rc == ERROR_SUCCESS) + { + char szValue[128]; + DWORD cbValue = sizeof(szValue) - 1; + rc = RegQueryValueExA(hkeyVer, NULL, NULL, NULL, (PBYTE)&szValue[0], &cbValue); + vbpsCloseKey(pState, hkeyVer, __LINE__); + if (rc == ERROR_SUCCESS) + { + szValue[cbValue] = '\0'; + if (!strcmp(szValue, "VirtualBox Type Library")) + { + /* + * Delete the type library version. + * We do not delete the whole type library ID, just this version of it. + */ + vbpsDeleteKeyRecursiveW(pState, hkeyTypeLibId, g_apwszTypelibVersions[iVer], __LINE__); + } + } + } + } + vbpsCloseKey(pState, hkeyTypeLibId, __LINE__); + + /* + * The type library ID key should be empty now, so we can try remove it (non-recursively). + */ + rc = RegDeleteKeyW(hkeyTypeLibs, g_apwszTypeLibIds[iTlb]); + Assert(rc == ERROR_SUCCESS); + } + } + } + else + Assert(rc == ERROR_FILE_NOT_FOUND); + } +} + + +/** + * Hack to clean out obsolete typelibs on development boxes and such. + */ +static void vbpsRemoveOldMessSub(REGSAM fSamWow) +{ + /* + * Note! The worker procedures does not use the default destination, + * because it's much much simpler to enumerate alternative locations. + */ + VBPSREGSTATE State; + LRESULT rc = vbpsRegInit(&State, HKEY_CLASSES_ROOT, NULL, true /*fDelete*/, false /*fUpdate*/, fSamWow); + if (rc == ERROR_SUCCESS) + { + vbpsRegAddAltDelete(&State, HKEY_CURRENT_USER, "Software\\Classes"); + vbpsRegAddAltDelete(&State, HKEY_LOCAL_MACHINE, "Software\\Classes"); + vbpsRegAddAltDelete(&State, HKEY_CLASSES_ROOT, NULL); + + vbpsRemoveOldInterfaces(&State); + vbpsRemoveOldClassIDs(&State); + vbpsRemoveOldTypeLibs(&State); + } + vbpsRegTerm(&State); +} + + +/** + * Hack to clean out obsolete typelibs on development boxes and such. + */ +static void removeOldMess(void) +{ + vbpsRemoveOldMessSub(0 /*fSamWow*/); +#if ARCH_BITS == 64 || defined(VBOX_IN_32_ON_64_MAIN_API) + vbpsRemoveOldMessSub(KEY_WOW64_32KEY); +#endif +} + + + +/** + * Register the interfaces proxied by this DLL, and to avoid duplication and + * minimize work the VBox type library, classes and servers are also registered. + * + * This is normally only used by developers via comregister.cmd and the heat.exe + * tool during MSI creation. The only situation where users may end up here is + * if they're playing around or we recommend it as a solution to COM problems. + * So, no problem if this approach is less gentle, though we leave the cleaning + * up of orphaned interfaces to DllUnregisterServer. + * + * @returns COM status code. + */ +HRESULT STDAPICALLTYPE DllRegisterServer(void) +{ + HRESULT hrc; + + /* + * Register the type library first. + */ + ITypeLib *pITypeLib; + WCHAR wszDllName[MAX_PATH]; + DWORD cwcRet = GetModuleFileNameW(g_hDllSelf, wszDllName, RT_ELEMENTS(wszDllName)); + AssertReturn(cwcRet > 0 && cwcRet < RT_ELEMENTS(wszDllName), CO_E_PATHTOOLONG); + + hrc = LoadTypeLib(wszDllName, &pITypeLib); + AssertMsgReturn(SUCCEEDED(hrc), ("%Rhrc\n", hrc), hrc); + hrc = RegisterTypeLib(pITypeLib, wszDllName, NULL /*pszHelpDir*/); + pITypeLib->lpVtbl->Release(pITypeLib); + AssertMsgReturn(SUCCEEDED(hrc), ("%Rhrc\n", hrc), hrc); + + /* + * Register proxy stub. + */ + hrc = NdrDllRegisterProxy(g_hDllSelf, &g_apProxyFiles[0], &g_ProxyClsId); /* see DLLREGISTRY_ROUTINES in RpcProxy.h */ + AssertMsgReturn(SUCCEEDED(hrc), ("%Rhrc\n", hrc), hrc); + + /* + * Register the VBox modules and classes. + */ + vbpsDllPathToVBoxDir(wszDllName); + hrc = RegisterXidlModulesAndClasses(wszDllName, true /*fDelete*/, true /*fUpdate*/); + AssertMsgReturn(SUCCEEDED(hrc), ("%Rhrc\n", hrc), hrc); + + return S_OK; +} + + +/** + * Reverse of DllRegisterServer. + * + * This is normally only used by developers via comregister.cmd. Users may be + * asked to perform it in order to fix some COM issue. So, it's OK if we spend + * some extra time and clean up orphaned interfaces, because developer boxes + * will end up with a bunch of those as interface UUIDs changes. + * + * @returns COM status code. + */ +HRESULT STDAPICALLTYPE DllUnregisterServer(void) +{ + HRESULT hrc = S_OK; + HRESULT hrc2; + + /* + * Unregister the type library. + * + * We ignore TYPE_E_REGISTRYACCESS as that is what is returned if the + * type lib hasn't been registered (W10). + */ + hrc2 = UnRegisterTypeLib(&LIBID_VirtualBox, kTypeLibraryMajorVersion, kTypeLibraryMinorVersion, + 0 /*LCid*/, RT_CONCAT(SYS_WIN, ARCH_BITS)); + AssertMsgStmt(SUCCEEDED(hrc2) || hrc2 == TYPE_E_REGISTRYACCESS, ("%Rhrc\n", hrc2), if (SUCCEEDED(hrc)) hrc = hrc2); + + /* + * Unregister the proxy stub. + * + * We ignore ERROR_FILE_NOT_FOUND as that is returned if not registered (W10). + */ + hrc2 = NdrDllUnregisterProxy(g_hDllSelf, &g_apProxyFiles[0], &g_ProxyClsId); /* see DLLREGISTRY_ROUTINES in RpcProxy.h */ + AssertMsgStmt( SUCCEEDED(hrc2) + || hrc2 == MAKE_HRESULT(SEVERITY_ERROR, FACILITY_WIN32, ERROR_FILE_NOT_FOUND) + || hrc2 == REGDB_E_INVALIDVALUE, + ("%Rhrc\n", hrc2), if (SUCCEEDED(hrc)) hrc = hrc2); + + /* + * Register the VBox modules and classes. + */ + hrc2 = RegisterXidlModulesAndClasses(NULL, true /*fDelete*/, false /*fUpdate*/); + AssertMsgStmt(SUCCEEDED(hrc2), ("%Rhrc\n", hrc2), if (SUCCEEDED(hrc)) hrc = hrc2); + + /* + * Purge old mess. + */ + removeOldMess(); + + return hrc; +} + + +#ifdef VBOX_WITH_SDS +/** + * Update a SCM service. + * + * @param pState The state. + * @param pwszVBoxDir The VirtualBox install directory (unicode), + * trailing slash. + * @param pwszModule The service module. + * @param pwszServiceName The service name. + * @param pwszDisplayName The service display name. + * @param pwszDescription The service description. + */ +static void vbpsUpdateWindowsService(VBPSREGSTATE *pState, const WCHAR *pwszVBoxDir, const WCHAR *pwszModule, + const WCHAR *pwszServiceName, const WCHAR *pwszDisplayName, const WCHAR *pwszDescription) +{ + SC_HANDLE hSCM; + + /* Configuration options that are currently standard. */ + uint32_t const uServiceType = SERVICE_WIN32_OWN_PROCESS; + uint32_t const uStartType = SERVICE_DEMAND_START; + uint32_t const uErrorControl = SERVICE_ERROR_NORMAL; + WCHAR const * const pwszServiceStartName = L"LocalSystem"; + static WCHAR const wszzDependencies[] = L"RPCSS\0"; + + /* + * Make double quoted executable file path. ASSUMES pwszVBoxDir ends with a slash! + */ + WCHAR wszFilePath[MAX_PATH + 2]; + int rc = RTUtf16CopyAscii(wszFilePath, RT_ELEMENTS(wszFilePath), "\""); + if (RT_SUCCESS(rc)) + rc = RTUtf16Cat(wszFilePath, RT_ELEMENTS(wszFilePath), pwszVBoxDir); + if (RT_SUCCESS(rc)) + rc = RTUtf16Cat(wszFilePath, RT_ELEMENTS(wszFilePath), pwszModule); + if (RT_SUCCESS(rc)) + rc = RTUtf16CatAscii(wszFilePath, RT_ELEMENTS(wszFilePath), "\""); + AssertLogRelRCReturnVoid(rc); + + /* + * Open the service manager for the purpose of checking the configuration. + */ + hSCM = OpenSCManagerW(NULL, NULL, SC_MANAGER_CONNECT); + if (hSCM != NULL) + { + union + { + QUERY_SERVICE_CONFIGW Config; + SERVICE_STATUS Status; + SERVICE_DESCRIPTIONW Desc; + uint8_t abPadding[sizeof(QUERY_SERVICE_CONFIGW) + 5 * _1K]; + } uBuf; + SC_HANDLE hService; + bool fCreateIt = pState->fUpdate; + bool fDeleteIt = true; + + /* + * Step #1: Open the service and validate the configuration. + */ + if (pState->fUpdate) + { + hService = OpenServiceW(hSCM, pwszServiceName, SERVICE_QUERY_CONFIG); + if (hService != NULL) + { + DWORD cbNeeded = 0; + if (QueryServiceConfigW(hService, &uBuf.Config, sizeof(uBuf), &cbNeeded)) + { + if (uBuf.Config.dwErrorControl) + { + uint32_t cErrors = 0; + if (uBuf.Config.dwServiceType != uServiceType) + { + LogRel(("update service '%ls': dwServiceType %u, expected %u\n", + pwszServiceName, uBuf.Config.dwServiceType, uServiceType)); + cErrors++; + } + if (uBuf.Config.dwStartType != uStartType) + { + LogRel(("update service '%ls': dwStartType %u, expected %u\n", + pwszServiceName, uBuf.Config.dwStartType, uStartType)); + cErrors++; + } + if (uBuf.Config.dwErrorControl != uErrorControl) + { + LogRel(("update service '%ls': dwErrorControl %u, expected %u\n", + pwszServiceName, uBuf.Config.dwErrorControl, uErrorControl)); + cErrors++; + } + if (RTUtf16ICmp(uBuf.Config.lpBinaryPathName, wszFilePath) != 0) + { + LogRel(("update service '%ls': lpBinaryPathName '%ls', expected '%ls'\n", + pwszServiceName, uBuf.Config.lpBinaryPathName, wszFilePath)); + cErrors++; + } + if ( uBuf.Config.lpServiceStartName != NULL + && *uBuf.Config.lpServiceStartName != L'\0' + && RTUtf16ICmp(uBuf.Config.lpServiceStartName, pwszServiceStartName) != 0) + { + LogRel(("update service '%ls': lpServiceStartName '%ls', expected '%ls'\n", + pwszServiceName, uBuf.Config.lpBinaryPathName, pwszServiceStartName)); + cErrors++; + } + + fDeleteIt = fCreateIt = cErrors > 0; + } + } + else + AssertLogRelMsgFailed(("QueryServiceConfigW returned %u (cbNeeded=%u vs %zu)\n", + GetLastError(), cbNeeded, sizeof(uBuf))); + } + else + { + DWORD dwErr = GetLastError(); + fDeleteIt = dwErr != ERROR_SERVICE_DOES_NOT_EXIST; + AssertLogRelMsg(dwErr == ERROR_SERVICE_DOES_NOT_EXIST, ("OpenServiceW('%ls') -> %u\n", pwszServiceName, dwErr)); + } + CloseServiceHandle(hService); + } + + /* + * Step #2: Stop and delete the service if needed. + * We can do this without reopening the service manager. + */ + if (fDeleteIt) + { + hService = OpenServiceW(hSCM, pwszServiceName, SERVICE_STOP | DELETE); + if (hService) + { + BOOL fRet; + DWORD dwErr; + RT_ZERO(uBuf.Status); + SetLastError(ERROR_SERVICE_NOT_ACTIVE); + fRet = ControlService(hService, SERVICE_CONTROL_STOP, &uBuf.Status); + dwErr = GetLastError(); + if ( fRet + || dwErr == ERROR_SERVICE_NOT_ACTIVE + || ( dwErr == ERROR_SERVICE_CANNOT_ACCEPT_CTRL + && uBuf.Status.dwCurrentState == SERVICE_STOP_PENDING) ) + { + if (DeleteService(hService)) + LogRel(("update service '%ls': deleted\n", pwszServiceName)); + else + AssertLogRelMsgFailed(("Failed to not delete service %ls: %u\n", pwszServiceName, GetLastError())); + } + else + AssertMsg(dwErr == ERROR_ACCESS_DENIED, + ("Failed to stop service %ls: %u (state=%u)\n", pwszServiceName, dwErr, uBuf.Status.dwCurrentState)); + CloseServiceHandle(hService); + } + else + { + pState->rc = GetLastError(); + LogRel(("Failed to not open service %ls for stop+delete: %u\n", pwszServiceName, pState->rc)); + hService = OpenServiceW(hSCM, pwszServiceName, SERVICE_CHANGE_CONFIG); + } + CloseServiceHandle(hService); + } + + CloseServiceHandle(hSCM); + + /* + * Step #3: Create the service (if requested). + * Need to have the SC_MANAGER_CREATE_SERVICE access right for this. + */ + if (fCreateIt) + { + Assert(pState->fUpdate); + hSCM = OpenSCManagerW(NULL, NULL, SC_MANAGER_CREATE_SERVICE); + if (hSCM) + { + hService = CreateServiceW(hSCM, + pwszServiceName, + pwszDisplayName, + SERVICE_CHANGE_CONFIG /* dwDesiredAccess */, + uServiceType, + uStartType, + uErrorControl, + wszFilePath, + NULL /* pwszLoadOrderGroup */, + NULL /* pdwTagId */, + wszzDependencies, + NULL /* pwszServiceStartName */, + NULL /* pwszPassword */); + if (hService != NULL) + { + uBuf.Desc.lpDescription = (WCHAR *)pwszDescription; + if (ChangeServiceConfig2W(hService, SERVICE_CONFIG_DESCRIPTION, &uBuf.Desc)) + LogRel(("update service '%ls': created\n", pwszServiceName)); + else + AssertMsgFailed(("Failed to set service description for %ls: %u\n", pwszServiceName, GetLastError())); + CloseServiceHandle(hService); + } + else + { + pState->rc = GetLastError(); + AssertMsgFailed(("Failed to create service '%ls': %u\n", pwszServiceName, pState->rc)); + } + CloseServiceHandle(hSCM); + } + else + { + pState->rc = GetLastError(); + LogRel(("Failed to open service manager with create service access: %u\n", pState->rc)); + } + } + } + else + AssertLogRelMsgFailed(("OpenSCManagerW failed: %u\n", GetLastError())); +} +#endif /* VBOX_WITH_SDS */ + + + +/** + * Gently update the COM registrations for VirtualBox. + * + * API that com::Initialize (VBoxCOM/initterm.cpp) calls the first time COM is + * initialized in a process. ASSUMES that the caller has initialized IPRT. + * + * @returns Windows error code. + */ +DECLEXPORT(uint32_t) VbpsUpdateRegistrations(void) +{ + LSTATUS rc; + VBPSREGSTATE State; +#ifdef VBOX_IN_32_ON_64_MAIN_API + bool const fIs32On64 = true; +#else + bool const fIs32On64 = false; +#endif + + /** @todo Should probably skip this when VBoxSVC is already running... Use + * some mutex or something for checking. */ + + /* + * Find the VirtualBox application directory first. + */ + WCHAR wszVBoxDir[MAX_PATH]; + DWORD cwcRet = GetModuleFileNameW(g_hDllSelf, wszVBoxDir, RT_ELEMENTS(wszVBoxDir)); + AssertReturn(cwcRet > 0 && cwcRet < RT_ELEMENTS(wszVBoxDir), ERROR_BUFFER_OVERFLOW); + vbpsDllPathToVBoxDir(wszVBoxDir); + + /* + * Update registry entries for the current CPU bitness. + */ + rc = vbpsRegInit(&State, HKEY_CLASSES_ROOT, NULL, false /*fDelete*/, true /*fUpdate*/, 0); + if (rc == ERROR_SUCCESS && !vbpsIsUpToDate(&State)) + { + +#ifdef VBOX_WITH_SDS + vbpsUpdateWindowsService(&State, wszVBoxDir, L"VBoxSDS.exe", L"VBoxSDS", + L"VirtualBox system service", L"Used as a COM server for VirtualBox API."); +#endif + vbpsUpdateTypeLibRegistration(&State, wszVBoxDir, fIs32On64); + vbpsUpdateProxyStubRegistration(&State, wszVBoxDir, fIs32On64); + vbpsUpdateInterfaceRegistrations(&State); + RegisterXidlModulesAndClassesGenerated(&State, wszVBoxDir, fIs32On64); + vbpsMarkUpToDate(&State); + rc = State.rc; + } + vbpsRegTerm(&State); + + +#if (ARCH_BITS == 64 && defined(VBOX_WITH_32_ON_64_MAIN_API)) /*|| defined(VBOX_IN_32_ON_64_MAIN_API) ??*/ + /* + * Update registry entries for the other CPU bitness. + */ + if (rc == ERROR_SUCCESS) + { + rc = vbpsRegInit(&State, HKEY_CLASSES_ROOT, NULL, false /*fDelete*/, true /*fUpdate*/, + !fIs32On64 ? KEY_WOW64_32KEY : KEY_WOW64_64KEY); + if (rc == ERROR_SUCCESS && !vbpsIsUpToDate(&State)) + { + vbpsUpdateTypeLibRegistration(&State, wszVBoxDir, !fIs32On64); + vbpsUpdateProxyStubRegistration(&State, wszVBoxDir, !fIs32On64); + vbpsUpdateInterfaceRegistrations(&State); + RegisterXidlModulesAndClassesGenerated(&State, wszVBoxDir, !fIs32On64); + vbpsMarkUpToDate(&State); + rc = State.rc; + } + vbpsRegTerm(&State); + } +#endif + + return VINF_SUCCESS; +} |