Leaked source code of windows server 2003
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

1270 lines
38 KiB

/*++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
Copyright (c) 1994-2000 Microsoft Corporation. All rights reserved.
Module Name:
registry.cxx
Abstract:
Registers the interfaces contained in the proxy DLL.
Public Functions:
DllRegisterServer
NdrDllRegisterProxy
NdrDllUnregisterProxy
Private Functions:
NdrpGetClassID
NdrpRegisterClass
NdrpRegisterInterface
NdrpUnregisterClass
NdrpUnregisterInterface
Author:
ShannonC 12-Oct-1994
Environment:
Windows NT and Windows 95.
Revision History:
RyszardK Nov 1997 Changes for async registration.
--------------------------------------------------------------------*/
#define USE_STUBLESS_PROXY
#define CINTERFACE
#include <ndrp.h>
#include <ndrole.h>
#include <rpcproxy.h>
#include <stdlib.h>
EXTERN_C HINSTANCE g_hRpcrt4 = 0;
HRESULT NdrpGetClassID(
OUT LPSTR pszClassID,
IN const CLSID * pclsid,
IN const ProxyFileInfo ** pProxyFileList);
HRESULT NdrpRegisterClass(
IN LPCSTR pszClassID,
IN LPCTSTR pszClassName,
IN LPCTSTR pszDllFileName,
IN LPCTSTR pszThreadingModel);
HRESULT NdrpRegisterInterface(
IN HKEY hKeyInterface,
IN REFIID riid,
IN LPCSTR pszInterfaceName,
IN LPCSTR pszClassID,
IN long NumMethods,
IN const IID * riidAsync
);
HRESULT NdrpRegisterAsyncInterface(
IN HKEY hKeyInterface,
IN REFIID riid,
IN LPCSTR pszSyncInterfaceName,
IN long SyncNumMethods,
IN REFIID riidAsync
);
HRESULT NdrpUnregisterClass(
IN LPCSTR pszClassID,
IN LPCTSTR pszDllFileName);
HRESULT NdrpUnregisterInterface(
IN HKEY hKeyInterface,
IN REFIID riid,
IN LPCSTR pszClassID,
IN const IID * riidAsync );
HRESULT RPC_ENTRY NdrDllRegisterProxy (
IN HMODULE hDll,
IN const ProxyFileInfo ** pProxyFileList,
IN const CLSID * pclsid OPTIONAL)
/*++
Routine Description:
Creates registry entries for the interfaces contained in the proxy DLL.
Arguments:
hDll - Supplies a handle to the proxy DLL.
pProxyFileList - Supplies a list of proxy files to be registered.
pclsid - Supplies the classid for the proxy DLL. May be zero.
Return Value:
S_OK
See Also:
DllRegisterServer
NdrDllUnregisterProxy
--*/
{
HRESULT hr;
long i, j;
HKEY hKeyInterface;
DWORD dwDisposition;
TCHAR szDllFileName[MAX_PATH+1];
long error;
ULONG length;
char szClassID[39];
if(hDll != 0)
{
//Get the proxy dll name.
length = GetModuleFileName(hDll,
szDllFileName,
MAX_PATH);
if( (length > 0) && ( length != MAX_PATH ) )
{
szDllFileName[MAX_PATH]='\0';
NDR_ASSERT(GetLastError() == NO_ERROR, "incorrect GetModuleFileName return" );
hr = S_OK;
}
else
{
NDR_ASSERT(GetLastError() != NO_ERROR, "incorrect GetModuleFileName return" );
hr = HRESULT_FROM_WIN32(GetLastError());
}
}
else
{
//The proxy DLL's DLL_PROCESS_ATTACH did not initialize hProxyDll.
hr = E_HANDLE;
}
if(SUCCEEDED(hr))
{
//Convert the class ID to to a registry key name.
hr = NdrpGetClassID(szClassID, pclsid, pProxyFileList);
}
if(SUCCEEDED(hr))
{
//Register the class
hr = NdrpRegisterClass(szClassID,
TEXT("PSFactoryBuffer"),
szDllFileName,
TEXT("Both"));
}
if(SUCCEEDED(hr))
{
//Create the Interface key.
error = RegCreateKeyEx(HKEY_CLASSES_ROOT,
TEXT("Interface"),
0,
TEXT("REG_SZ"),
REG_OPTION_NON_VOLATILE,
KEY_WRITE,
0,
&hKeyInterface,
&dwDisposition);
if(!error)
{
HRESULT hr2;
// the only purpose of this call is to setup gfRpcVerifierEnabled flag.
// this call can be called later again disregard the function succeeded or not here.
NdrpPerformRpcInitialization();
//iterate over the list of proxy files in the proxy DLL.
for(i = 0;
pProxyFileList[i] != 0;
i++)
{
if ( pProxyFileList[i]->TableVersion & NDR_PROXY_FILE_ASYNC_UUID)
{
// Iterate through sync and async interfaces.
for(j = 0;
pProxyFileList[i]->pProxyVtblList[j] != 0;
j++)
{
if ( gfRPCVerifierEnabledWithBreaks )
{
if ( NdrpCheckMIDLRobust( pProxyFileList[i]->pStubVtblList[j]->header.pServerInfo,
pProxyFileList[i]->pStubVtblList[j]->header.DispatchTableCount,
TRUE ) )
{
RPC_VERIFIER_WARNING_MSG("Possible security threat: Server registers an interface compiled without /robust option",
RPC_VERIFIER_REGISTERING_NONROBUST_IF);
DbgPrint("RPC: Unsecure interface UUID: ");
PrintUUID((UUID *)pProxyFileList[i]->pStubVtblList[j]->header.piid);
DbgPrint("\n");
RPC_VERIFIER_PRINT_OFFENDING_STACK(2, 4);
}
}
if ( pProxyFileList[i]->pAsyncIIDLookup[j] == 0)
{
// just a sync interface, no async counterpart.
hr2 = NdrpRegisterInterface(hKeyInterface,
*pProxyFileList[i]->pStubVtblList[j]->header.piid,
pProxyFileList[i]->pNamesArray[j],
szClassID,
pProxyFileList[i]->pStubVtblList[j]->header.DispatchTableCount,
0 /* no async */);
if(FAILED(hr2) && SUCCEEDED(hr))
hr = hr2;
}
else if ( (ULONG_PTR) pProxyFileList[i]->pAsyncIIDLookup[j] != -1 )
{
// Register an sync-async pair of interfaces.
hr2 = NdrpRegisterInterface(hKeyInterface,
*pProxyFileList[i]->pStubVtblList[j]->header.piid,
pProxyFileList[i]->pNamesArray[j],
szClassID,
pProxyFileList[i]->pStubVtblList[j]->header.DispatchTableCount,
pProxyFileList[i]->pAsyncIIDLookup[j]);
if(FAILED(hr2) && SUCCEEDED(hr))
hr = hr2;
hr2 = NdrpRegisterAsyncInterface(hKeyInterface,
*pProxyFileList[i]->pStubVtblList[j]->header.piid,
pProxyFileList[i]->pNamesArray[j],
pProxyFileList[i]->pStubVtblList[j]->header.DispatchTableCount,
*pProxyFileList[i]->pAsyncIIDLookup[j] );
if(FAILED(hr2) && SUCCEEDED(hr))
hr = hr2;
}
}
}
else
{
// Plain old style sync interfaces only.
// iterate over the list of interfaces in the proxy file.
for(j = 0;
pProxyFileList[i]->pProxyVtblList[j] != 0;
j++)
{
if ( gfRPCVerifierEnabledWithBreaks )
{
if ( NdrpCheckMIDLRobust( pProxyFileList[i]->pStubVtblList[j]->header.pServerInfo,
pProxyFileList[i]->pStubVtblList[j]->header.DispatchTableCount,
TRUE ) )
{
RPC_VERIFIER_WARNING_MSG("Possible security threat: Server registers an interface compiled without /robust option",
RPC_VERIFIER_REGISTERING_NONROBUST_IF);
DbgPrint("RPC: Unsecure interface UUID: ");
PrintUUID((UUID *)pProxyFileList[i]->pStubVtblList[j]->header.piid);
DbgPrint("\n");
RPC_VERIFIER_PRINT_OFFENDING_STACK(2, 4);
}
}
hr2 = NdrpRegisterInterface(hKeyInterface,
*pProxyFileList[i]->pStubVtblList[j]->header.piid,
pProxyFileList[i]->pNamesArray[j],
szClassID,
pProxyFileList[i]->pStubVtblList[j]->header.DispatchTableCount,
0 /* no async */);
if(FAILED(hr2) && SUCCEEDED(hr))
hr = hr2;
}
}
}
RegCloseKey(hKeyInterface);
}
else
{
hr = HRESULT_FROM_WIN32(error);
}
}
return hr;
}
HRESULT CheckInprocServer32(
IN HKEY hKeyIID,
IN LPCTSTR pszDllFileName)
{
HRESULT hr;
HKEY hKey;
TCHAR szDll[MAX_PATH];
long cbData = sizeof(szDll);
long error;
DWORD dwType;
//Open the InprocServer32 key.
error = RegOpenKeyEx(hKeyIID,
TEXT("InprocServer32"),
0,
KEY_READ,
&hKey);
if(!error)
{
error = RegQueryValueEx(hKey,
TEXT(""),
0,
&dwType,
(BYTE*)szDll,
(ulong*)&cbData);
if(!error)
{
if(0 == lstrcmpi(pszDllFileName,
szDll))
hr = S_OK;
else
hr = REGDB_E_INVALIDVALUE;
}
else
{
hr = HRESULT_FROM_WIN32(error);
}
RegCloseKey(hKey);
}
else
{
hr = HRESULT_FROM_WIN32(error);
}
return hr;
}
HRESULT NdrpCheckClass(
IN LPCSTR pszClassID,
IN LPCTSTR pszDllFileName)
{
HRESULT hr;
long error;
HKEY hKeyCLSID;
//open the CLSID key
error = RegOpenKeyEx(HKEY_CLASSES_ROOT,
TEXT("CLSID"),
0,
KEY_WRITE,
&hKeyCLSID);
if(!error)
{
HKEY hKeyClassID;
//open registry key for class ID string
error = RegOpenKeyExA(hKeyCLSID,
pszClassID,
0,
KEY_WRITE,
&hKeyClassID);
if(!error)
{
hr = CheckInprocServer32(hKeyClassID,
pszDllFileName);
RegCloseKey(hKeyClassID);
}
else
{
hr = HRESULT_FROM_WIN32(error);
}
RegCloseKey(hKeyCLSID);
}
else
{
hr = HRESULT_FROM_WIN32(error);
}
return hr;
}
HRESULT RPC_ENTRY NdrDllUnregisterProxy (
IN HMODULE hDll,
IN const ProxyFileInfo ** pProxyFileList,
IN const CLSID * pclsid OPTIONAL)
/*++
Routine Description:
Removes registry entries for the interfaces contained in the proxy DLL.
Arguments:
hDll - Supplies a handle to the proxy DLL.
pProxyFileList - Supplies a list of proxy files to be unregistered.
pclsid - Supplies the classid for the proxy DLL. May be zero.
Return Value:
S_OK
See Also:
DllUnregisterServer
NdrDllRegisterProxy
--*/
{
HRESULT hr;
HKEY hKeyInterface;
long i, j;
long error;
TCHAR szDllFileName[MAX_PATH];
ULONG length;
char szClassID[39];
if(hDll != 0)
{
//Get the proxy dll name.
length = GetModuleFileName(hDll, szDllFileName, sizeof(szDllFileName));
if(length > 0)
{
hr = S_OK;
}
else
{
hr = HRESULT_FROM_WIN32(GetLastError());
}
}
else
{
//The DLL_PROCESS_ATTACH in the proxy DLL failed to initialize hProxyDll.
hr = E_HANDLE;
}
if(SUCCEEDED(hr))
{
//Convert the class ID to a registry key name.
hr = NdrpGetClassID(szClassID, pclsid, pProxyFileList);
}
if(SUCCEEDED(hr))
{
//Check the class
hr = NdrpCheckClass(szClassID, szDllFileName);
}
if(SUCCEEDED(hr))
{
HRESULT hr2;
//Open the Interface key.
error = RegOpenKeyEx(HKEY_CLASSES_ROOT,
TEXT("Interface"),
0,
KEY_WRITE,
&hKeyInterface);
if (!error)
{
//iterate over the list of proxy files in the proxy DLL.
for(i = 0;
pProxyFileList[i] != 0;
i++)
{
if ( pProxyFileList[i]->TableVersion & NDR_PROXY_FILE_ASYNC_UUID)
{
// Iterate through sync and async interfaces.
for(j = 0;
pProxyFileList[i]->pProxyVtblList[j] != 0;
j++)
{
if ( (ULONG_PTR) pProxyFileList[i]->pAsyncIIDLookup[j] != -1 )
{
// Unegister a single sync only interface or an sync-async
// pair of interfaces. Skip async interfaces.
NdrpUnregisterInterface( hKeyInterface,
*pProxyFileList[i]->pStubVtblList[j]->header.piid,
szClassID,
pProxyFileList[i]->pAsyncIIDLookup[j]);
}
}
}
else
{
//iterate over the list of interfaces in the proxy file.
for(j = 0;
pProxyFileList[i]->pProxyVtblList[j] != 0;
j++)
{
NdrpUnregisterInterface(hKeyInterface,
*pProxyFileList[i]->pStubVtblList[j]->header.piid,
szClassID,
0 /* no async */);
}
}
}
RegCloseKey(hKeyInterface);
}
else
{
hr = HRESULT_FROM_WIN32(error);
}
//Unregister the class
hr2 = NdrpUnregisterClass(szClassID, szDllFileName);
if(FAILED(hr2) && SUCCEEDED(hr))
hr = hr2;
}
return hr;
}
HRESULT NdrpGetClassID(
OUT LPSTR pszClassID,
IN const CLSID * pclsid,
IN const ProxyFileInfo ** pProxyFileList)
/*++
Routine Description:
Gets a string specifying the Class ID for the PSFactoryBuffer.
If pclsid is NULL, then this function will use the IID of the
first interface as the class ID.
Arguments:
pszClassID - The Class ID string is returned in this buffer.
pclsid - Specifies the class ID. May be zero.
pProxyFileList - Points to a list of ProxyFiles.
Return Value:
S_OK
E_NOINTERFACE
--*/
{
HRESULT hr;
long i, j;
//If necessary, use the IID of the first interface as the CLSID.
for(i = 0;
(pProxyFileList[i] != 0) && (!pclsid);
i++)
{
for(j = 0;
(pProxyFileList[i]->pProxyVtblList[j] != 0) && (!pclsid);
j++)
{
pclsid = pProxyFileList[i]->pStubVtblList[j]->header.piid;
}
}
if(pclsid != 0)
{
hr = NdrStringFromIID( *pclsid, pszClassID );
}
else
{
hr = E_NOINTERFACE;
}
return hr;
}
HRESULT NdrpRegisterClass(
IN LPCSTR pszClassID,
IN LPCTSTR pszClassName OPTIONAL,
IN LPCTSTR pszDllFileName,
IN LPCTSTR pszThreadingModel OPTIONAL)
/*++
Routine Description:
Creates a registry entry for an in-process server class.
Arguments:
pszClassID - Supplies the class ID.
pszClassName - Supplies the class name. May be NULL.
pszDllFileName - Supplies the DLL file name.
pszThreadingModel - Supplies the threading model. May be NULL.
The threading model should be one of the following:
"Apartment", "Both", "Free".
Return Value:
S_OK
See Also:
NdrDllRegisterProxy
NdrpUnregisterClass
--*/
{
HRESULT hr;
long error;
HKEY hKeyCLSID;
HKEY hKeyClassID;
HKEY hKey;
DWORD dwDisposition;
//create the CLSID key
error = RegCreateKeyEx(HKEY_CLASSES_ROOT,
TEXT("CLSID"),
0,
TEXT("REG_SZ"),
REG_OPTION_NON_VOLATILE,
KEY_WRITE,
0,
&hKeyCLSID,
&dwDisposition);
if(!error)
{
//Create registry key for class ID
error = RegCreateKeyExA(hKeyCLSID,
pszClassID,
0,
"REG_SZ",
REG_OPTION_NON_VOLATILE,
KEY_WRITE,
0,
&hKeyClassID,
&dwDisposition);
if(!error)
{
//Create InProcServer32 key for the proxy dll
error = RegCreateKeyEx(hKeyClassID,
TEXT("InProcServer32"),
0,
TEXT("REG_SZ"),
REG_OPTION_NON_VOLATILE,
KEY_WRITE,
0,
&hKey,
&dwDisposition);
if(!error)
{
//register the proxy DLL filename
error = RegSetValueEx(hKey,
TEXT(""),
0,
REG_SZ,
(BYTE*)pszDllFileName,
strlen(pszDllFileName) + 1);
if((!error) && (pszThreadingModel != 0))
{
//register the threading model for the proxy DLL.
error = RegSetValueEx(hKey,
TEXT("ThreadingModel"),
0,
REG_SZ,
(BYTE*)pszThreadingModel,
strlen(pszThreadingModel) + 1);
}
RegCloseKey(hKey);
}
if((!error) && (pszClassName != 0))
{
// put the class name in an unnamed value
error = RegSetValueEx(hKeyClassID,
TEXT(""),
0,
REG_SZ,
(BYTE*)pszClassName,
strlen(pszClassName) + 1);
}
RegCloseKey(hKeyClassID);
}
RegCloseKey(hKeyCLSID);
}
if(!error)
hr = S_OK;
else
hr = HRESULT_FROM_WIN32(error);
return hr;
}
HRESULT NdrpRegisterInterface(
IN HKEY hKeyInterface,
IN REFIID riid,
IN LPCSTR pszInterfaceName,
IN LPCSTR pszClassID,
IN long NumMethods,
IN const IID * riidAsync )
/*++
Routine Description:
Creates a registry entry for an interface proxy.
Arguments:
hKeyInterface
riid
pszInterfaceName
pszClassID
NumMethods
riidAsync - async iid, may be null.
Return Value:
S_OK
See Also:
NdrDllRegisterProxy
NdrpUnregisterInterface
--*/
{
HRESULT hr;
long error;
char szIID[39];
char szNumMethods[6];
DWORD dwDisposition;
HKEY hKey;
HKEY hKeyIID;
//convert the IID to a registry key name.
NdrStringFromIID( riid, szIID );
//create registry key for the interface
error = RegCreateKeyExA(hKeyInterface,
szIID,
0,
"REG_SZ",
REG_OPTION_NON_VOLATILE,
KEY_WRITE,
0,
&hKeyIID,
&dwDisposition);
if (!error)
{
//create ProxyStubClsid32 key.
error = RegCreateKeyEx(hKeyIID,
TEXT("ProxyStubClsid32"),
0,
TEXT("REG_SZ"),
REG_OPTION_NON_VOLATILE,
KEY_WRITE,
0,
&hKey,
&dwDisposition);
if (!error)
{
//Set the class id for the PSFactoryBuffer.
error = RegSetValueExA(hKey,
"",
0,
REG_SZ,
(BYTE*)pszClassID,
strlen(pszClassID) + 1);
RegCloseKey(hKey);
}
// put the interface name in the unnamed value
if(!error)
{
error = RegSetValueExA(hKeyIID,
"",
0,
REG_SZ,
(BYTE*)pszInterfaceName,
strlen(pszInterfaceName) + 1);
}
//create NumMethods key.
if(!error)
{
error = RegCreateKeyEx(hKeyIID,
TEXT("NumMethods"),
0,
TEXT("REG_SZ"),
REG_OPTION_NON_VOLATILE,
KEY_WRITE,
0,
&hKey,
&dwDisposition);
if(!error)
{
//Set the number of methods
RpcItoa( NumMethods, szNumMethods, 10 );
error = RegSetValueExA(hKey,
"",
0,
REG_SZ,
(UCHAR *) szNumMethods,
strlen(szNumMethods) + 1);
RegCloseKey(hKey);
}
}
if ( riidAsync )
{
//create AsynchronousInterface key under the interface.
if(!error)
{
error = RegCreateKeyEx( hKeyIID,
TEXT("AsynchronousInterface"),
0,
TEXT("REG_SZ"),
REG_OPTION_NON_VOLATILE,
KEY_WRITE,
0,
&hKey,
&dwDisposition);
if(!error)
{
// Set the iid as value for the string.
NdrStringFromIID( *riidAsync, szIID );
error = RegSetValueExA( hKey,
"",
0,
REG_SZ,
(UCHAR *) szIID,
strlen(szIID) + 1);
RegCloseKey(hKey);
}
}
}
RegCloseKey(hKeyIID);
}
if(!error)
hr = S_OK;
else
hr = HRESULT_FROM_WIN32(error);
return hr;
}
HRESULT NdrpRegisterAsyncInterface(
IN HKEY hKeyInterface,
IN REFIID riid,
IN LPCSTR pszSyncInterfaceName,
IN long SyncNumMethods,
IN REFIID riidAsync )
/*++
Routine Description:
Creates a registry entry for an async interface proxy.
Arguments:
hKeyInterface
riid
pszInterfaceName
pszClassID
NumMethods
riidAsync
Return Value:
S_OK
See Also:
NdrDllRegisterProxy
NdrpUnregisterInterface
--*/
{
HRESULT hr;
long error;
char szIID[39];
char szNumMethods[6];
DWORD dwDisposition;
HKEY hKey;
HKEY hKeyIID;
//convert the IID to a registry key name.
NdrStringFromIID( riidAsync, szIID );
//create registry key for the interface
error = RegCreateKeyExA(hKeyInterface,
szIID,
0,
"REG_SZ",
REG_OPTION_NON_VOLATILE,
KEY_WRITE,
0,
&hKeyIID,
&dwDisposition);
// By definition, for async interfaces do not create Clsid32 key.
// put the interface name in the unnamed value
if(!error)
{
char * pszAsyncInterfaceName;
int len;
len = 5 + strlen(pszSyncInterfaceName) + 1; /* 5 is strlen("Async") */
pszAsyncInterfaceName = (char*)alloca(len);
RpcpMemoryCopy( pszAsyncInterfaceName, "Async", 5 );
RpcpMemoryCopy( pszAsyncInterfaceName + 5, pszSyncInterfaceName, len - 5 );
error = RegSetValueExA(hKeyIID,
"",
0,
REG_SZ,
(BYTE*)pszAsyncInterfaceName,
len);
//create NumMethods key.
if(!error)
{
long AsyncNumMethods = 2 * SyncNumMethods - 3;
error = RegCreateKeyEx(hKeyIID,
TEXT("NumMethods"),
0,
TEXT("REG_SZ"),
REG_OPTION_NON_VOLATILE,
KEY_WRITE,
0,
&hKey,
&dwDisposition);
if(!error)
{
//Set the number of methods
RpcItoa( AsyncNumMethods, szNumMethods, 10 );
error = RegSetValueExA(hKey,
"",
0,
REG_SZ,
(UCHAR *) szNumMethods,
strlen(szNumMethods) + 1);
RegCloseKey(hKey);
}
}
//create SynchronousInterface key under the interface.
if(!error)
{
error = RegCreateKeyEx( hKeyIID,
TEXT("SynchronousInterface"),
0,
TEXT("REG_SZ"),
REG_OPTION_NON_VOLATILE,
KEY_WRITE,
0,
&hKey,
&dwDisposition);
if(!error)
{
// Set the iid as value for the string.
NdrStringFromIID( riid, szIID );
error = RegSetValueExA( hKey,
"",
0,
REG_SZ,
(UCHAR *) szIID,
strlen(szIID) + 1);
RegCloseKey(hKey);
}
}
RegCloseKey(hKeyIID);
}
if(!error)
hr = S_OK;
else
hr = HRESULT_FROM_WIN32(error);
return hr;
}
HRESULT NdrpUnregisterClass(
IN LPCSTR pszClassID,
IN LPCTSTR pszDllFileName)
/*++
Routine Description:
Removes an in-process server class from the registry.
Arguments:
pszClassID - Supplies the class ID.
Return Value:
S_OK
See Also:
NdrDllUnregisterProxy
NdrpRegisterClass
--*/
{
HRESULT hr;
HKEY hKeyCLSID;
HKEY hKeyClassID;
long error;
//open the CLSID key
error = RegOpenKeyEx(HKEY_CLASSES_ROOT,
TEXT("CLSID"),
0,
KEY_WRITE,
&hKeyCLSID);
if(!error)
{
//open registry key for class ID string
error = RegOpenKeyExA(hKeyCLSID,
pszClassID,
0,
KEY_WRITE,
&hKeyClassID);
if(!error)
{
hr = CheckInprocServer32(hKeyClassID,
pszDllFileName);
if(SUCCEEDED(hr))
{
//delete InProcServer32 key.
error = RegDeleteKey(hKeyClassID,
TEXT("InProcServer32"));
if(error != 0)
{
hr = HRESULT_FROM_WIN32(error);
}
}
RegCloseKey(hKeyClassID);
}
else
{
hr = HRESULT_FROM_WIN32(error);
}
if(SUCCEEDED(hr))
{
error = RegDeleteKeyA(hKeyCLSID, pszClassID);
if(error != 0)
{
hr = HRESULT_FROM_WIN32(error);
}
}
RegCloseKey(hKeyCLSID);
}
else
{
hr = HRESULT_FROM_WIN32(error);
}
return hr;
}
HRESULT CheckProxyStubClsid32(
IN HKEY hKeyIID,
IN LPCSTR pszClassID)
{
HRESULT hr;
HKEY hKey;
char szClassID[39];
long cbData = sizeof(szClassID);
long error;
DWORD dwType;
//Open the ProxyStubClsid32 key.
error = RegOpenKeyEx(hKeyIID,
TEXT("ProxyStubClsid32"),
0,
KEY_READ,
&hKey);
if(!error)
{
error = RegQueryValueExA(hKey,
"",
0,
&dwType,
(BYTE*)szClassID,
(ulong*)&cbData);
if(!error)
{
if(0 == memcmp(szClassID, pszClassID, cbData))
hr = S_OK;
else
hr = REGDB_E_INVALIDVALUE;
}
else
{
hr = HRESULT_FROM_WIN32(error);
}
RegCloseKey(hKey);
}
else
{
hr = HRESULT_FROM_WIN32(error);
}
return hr;
}
HRESULT NdrpUnregisterInterface(
IN HKEY hKeyInterface,
IN REFIID riid,
IN LPCSTR pszClassID,
IN const IID * riidAsync )
/*++
Routine Description:
Unregisters an interface proxy.
Arguments:
hKeyInterface
riid
Return Value:
S_OK
See Also:
NdrDllUnregisterProxy
NdrpRegisterInterface
--*/
{
HRESULT hr = S_OK;
long error;
char szIID[39];
HKEY hKeyIID;
//convert the IID to a registry key name.
NdrStringFromIID( riid, szIID );
//Open the IID key.
error = RegOpenKeyExA(hKeyInterface,
szIID,
0,
KEY_WRITE,
&hKeyIID);
if (!error)
{
// As we call for sync singles or sync pairs (sync-async),
// we always have the class id.
hr = CheckProxyStubClsid32(hKeyIID, pszClassID);
if(SUCCEEDED(hr))
{
// Once the class id matches, just attempt to delete
// every possible key that may happen under the sync entry.
// Note that additional key may be present due to oleauto
// registering a TLB.
RegDeleteKey(hKeyIID, TEXT("NumMethods"));
RegDeleteKey(hKeyIID, TEXT("ProxyStubClsid32"));
RegDeleteKey(hKeyIID, TEXT("AsynchronousInterface"));
// Now remove the matching async interface entry, if there is one.
if ( riidAsync )
{
char szAsyncIID[39];
HKEY hKeyAsyncIID;
//convert the IID to a registry key name.
NdrStringFromIID( *riidAsync, szAsyncIID );
//Open the IID key.
error = RegOpenKeyExA(hKeyInterface,
szAsyncIID,
0,
KEY_WRITE,
&hKeyAsyncIID);
if ( !error )
{
RegDeleteKey( hKeyAsyncIID, TEXT("NumMethods"));
RegDeleteKey( hKeyAsyncIID, TEXT("SynchronousInterface"));
RegCloseKey(hKeyAsyncIID);
RegDeleteKeyA(hKeyInterface, szAsyncIID);
}
}
}
else
hr = S_FALSE;
//Close the IID key.
RegCloseKey(hKeyIID);
RegDeleteKeyA(hKeyInterface, szIID);
}
return hr;
}
STDAPI DllRegisterServer(void)
/*++
Routine Description:
Creates registry entries for the classes contained in rpcrt4.dll.
Return Value:
S_OK
--*/
{
HRESULT hr;
TCHAR szDllFileName[MAX_PATH];
ULONG length;
if(!g_hRpcrt4)
return E_HANDLE;
//Get the proxy dll name.
length = GetModuleFileName(g_hRpcrt4,
szDllFileName,
sizeof(szDllFileName));
if(length > 0)
{
//Register the class
hr = NdrpRegisterClass(TEXT("{b5866878-bd99-11d0-b04b-00c04fd91550}"),
TEXT("TypeFactory"),
szDllFileName,
TEXT("Both"));
}
else
{
hr = HRESULT_FROM_WIN32(GetLastError());
}
return hr;
}