windows-nt/Source/XPSP1/NT/com/oleutest/accctrl/server/oactests.cxx
2020-09-26 16:20:57 +08:00

520 lines
15 KiB
C++

/****************************************************************************
FILE: oactests.cxx
****************************************************************************/
#include <windows.h>
#include <oleext.h>
#include <stdio.h>
#include "oactest.h" // header file generated by MIDL compiler
HRESULT RegisterClassFactory();
HRESULT RevokeClassFactory();
//Prototypes for local functions
HRESULT RegisterLocalServer(REFCLSID rclsid);
HRESULT RegisterProxyDll(const char *pszProxyDll);
HRESULT UnregisterLocalServer(REFCLSID rclsid);
HRESULT UnregisterProxyDll(const char *pszProxyDll);
extern "C" const CLSID CLSID_COAccessControlTest
= {0x20692b00,0xe710,0x11cf,{0xaf,0x0b,0x00,0xaa,0x00,0x44,0xfb,0x89}};
extern ULONG g_ulFrequency;
long g_LockCount = 0;
long g_ObjectCount = 0;
DWORD g_dwThreadID;
ULONG g_ulFrequency;
//+---------------------------------------------------------------------------
//
// Function: DecrementLockCount
//
// Synopsis: Decrements the lock count. The lock count controls the
// registration of the class factory in the class table.
// DecrementLockCount revokes the class factory when the
// lock count is zero.
//
// Called By: ~CAccessControlTest, IClassFactory::LockServer(FALSE).
//
// See Also: ObjectCreated, ObjectDestroyed, IncrementLockCount.
//
//----------------------------------------------------------------------------
void DecrementLockCount()
{
//Decrement the lock count.
if(InterlockedDecrement(&g_LockCount) == 0)
{
//When the lock count is decremented to zero,
//revoke the class factory.
RevokeClassFactory();
}
}
//+---------------------------------------------------------------------------
//
// Function: IncrementLockCount
//
// Synopsis: Increments the lock count. The lock count controls the
// registration of the class factory in the class table.
// If necessary, IncrementLockCount will reregister the class
// factory in the class table.
//
// Called By: CAccessControlTest, IClassFactory::LockServer(TRUE).
//
// See Also: DecrementLockCount, ObjectCreated, ObjectDestroyed.
//
//----------------------------------------------------------------------------
void IncrementLockCount()
{
if(g_LockCount == 0)
{
//Increment the lock count.
InterlockedIncrement(&g_LockCount);
//Reregister the class factory if necessary.
RegisterClassFactory();
}
else
{
//Increment the lock count.
InterlockedIncrement(&g_LockCount);
}
}
//+---------------------------------------------------------------------------
//
// Function: main
//
// Synopsis: Main entry point for the server application. This function
// initializes the server and processes the message loop.
//
//----------------------------------------------------------------------------
int _cdecl main(int argc, char *argv[])
{
HRESULT hr = S_OK;
int i;
BOOL fRegisterServer = FALSE;
BOOL fUnregisterServer = FALSE;
BOOL fEmbedding = FALSE;
BOOL fHelp = FALSE;
MSG msg;
// Parse each item, skip the command name
for (i = 1; i < argc; i++)
{
if (lstrcmpiA( argv[i], "-Embedding" ) == 0)
{
//This server has been activated by OLE.
fEmbedding = TRUE;
}
else if (lstrcmpiA( argv[i], "/REGSERVER" ) == 0)
{
fRegisterServer = TRUE;
}
else if (lstrcmpiA( argv[i], "/UNREGSERVER" ) == 0)
{
fUnregisterServer = TRUE;
}
else
{
fHelp = TRUE;
}
}
if(fHelp == TRUE)
{
printf("USAGE: server /REGSERVER - Installs server in the registry.\n");
printf(" server /UNREGSERVER - Removes server from the registry.\n");
printf(" server -Embedding - Server was auto-started by OLE.\n");
return 0;
}
else if(fUnregisterServer == TRUE)
{
hr = UnregisterLocalServer(CLSID_COAccessControlTest);
if(FAILED(hr))
{
printf("Failed to remove local server from the registry.\n");
}
hr = UnregisterProxyDll("oactest.dll");
if(FAILED(hr))
{
printf("Failed to remove proxy DLL from the registry.\n");
}
return 0;
}
//Install the local server in the registry.
hr = RegisterLocalServer(CLSID_COAccessControlTest);
if(FAILED(hr))
{
printf("Failed to install local server in the registry.\n");
}
//Install the proxy DLL in the registry.
hr = RegisterProxyDll("oactest.dll");
if(FAILED(hr))
{
printf("Failed to install proxy DLL in the registry.\n");
}
if(fRegisterServer == TRUE)
return 0;
//
// Initialize OLE before calling any other OLE functions.
//
hr = CoInitializeEx(NULL, COINIT_MULTITHREADED);
if(SUCCEEDED(hr))
{
hr = RegisterClassFactory();
if(SUCCEEDED(hr))
{
LARGE_INTEGER liFreq;
QueryPerformanceFrequency(&liFreq);
g_ulFrequency = liFreq.LowPart;
printf("Waiting for client to connect...\n");
g_dwThreadID = GetCurrentThreadId();
// Main message loop. We will remain in the message loop
// until we get a WM_QUIT message.
while (GetMessage(&msg, NULL, 0, 0))
{
TranslateMessage(&msg);
DispatchMessage(&msg);
}
//If the class factory is still registered in the
//class table, revoke it before we shut down.
RevokeClassFactory();
}
else
{
printf("Failed to register class factory.\n");
}
CoUninitialize();
}
else
{
printf("CoInitialize failed.\n");
}
return 0;
}
//+---------------------------------------------------------------------------
//
// Function: ObjectCreated
//
// Synopsis: Increments the object count controlling the main message loop
// of the server process.
//
// Called By: CAccessControlTest, CAccessControlTestFactory.
//
// See Also: DecrementLockCount, IncrementLockCount, ObjectDestroyed.
//
//----------------------------------------------------------------------------
void ObjectCreated()
{
InterlockedIncrement(&g_ObjectCount);
}
//+---------------------------------------------------------------------------
//
// Function: ObjectDestroyed
//
// Synopsis: Decrements the object count controlling the main message loop
// of the server process. When the object count is decremented
// to zero, we post a WM_QUIT message. The main message loop will
// exit and terminate the server process.
//
// Called By: ~CAccessControlTest, ~CAccessControlTestFactory.
//
// See Also: DecrementLockCount, IncrementLockCount, ObjectCreated.
//
//----------------------------------------------------------------------------
void ObjectDestroyed()
{
if(InterlockedDecrement(&g_ObjectCount) == 0)
{
//When the last object is released, post a quit message.
PostThreadMessage(g_dwThreadID, WM_QUIT, NULL, NULL);
}
}
//+---------------------------------------------------------------------------
//
// Function: RegisterLocalServer
//
// Synopsis: Creates a registry key for a local server.
//
// Parameters: rclsid - Supplies the class ID of the local server.
//
//----------------------------------------------------------------------------
HRESULT RegisterLocalServer(REFCLSID rclsid)
{
HRESULT hr;
LONG error;
HKEY hKeyCLSID; // key for ...\Classes\CLSID
HKEY hKeyClassID;
HKEY hKey; // current key
DWORD dwDisposition;
char szServer[MAX_PATH];
char szClassID[39];
unsigned long length;
length = GetModuleFileNameA(0, szServer, sizeof(szServer));
if(length == 0)
{
hr = HRESULT_FROM_WIN32(GetLastError());
return hr;
}
//create the CLSID key
error = RegCreateKeyExA(
HKEY_CLASSES_ROOT,
"CLSID",
0,
"REG_SZ",
REG_OPTION_NON_VOLATILE,
KEY_ALL_ACCESS,
0,
&hKeyCLSID,
&dwDisposition);
if(!error)
{
//convert the class ID to a registry key name.
sprintf(szClassID, "{%08lX-%04X-%04X-%02X%02X-%02X%02X%02X%02X%02X%02X}",
rclsid.Data1, rclsid.Data2, rclsid.Data3,
rclsid.Data4[0], rclsid.Data4[1],
rclsid.Data4[2], rclsid.Data4[3],
rclsid.Data4[4], rclsid.Data4[5],
rclsid.Data4[6], rclsid.Data4[7]);
//create key for the server class
error = RegCreateKeyExA(hKeyCLSID,
szClassID,
0,
"REG_SZ",
REG_OPTION_NON_VOLATILE,
KEY_ALL_ACCESS,
0,
&hKeyClassID,
&dwDisposition);
if(!error)
{
//create LocalServer32 key.
error = RegCreateKeyExA(hKeyClassID,
"LocalServer32",
0,
"REG_SZ",
REG_OPTION_NON_VOLATILE,
KEY_ALL_ACCESS,
0,
&hKey,
&dwDisposition);
if(!error)
{
//Set the server name.
error = RegSetValueExA(hKey,
"",
0,
REG_SZ,
(const unsigned char *)szServer,
lstrlenA(szServer) + 1);
// Specifies in the registry that this
// server supports both threading models
error = RegSetValueExA( hKey
, "ThreadingModel"
, 0
, REG_SZ
, (const unsigned char *)"Both"
, lstrlenA("Both") + 1);
RegFlushKey(hKey);
RegCloseKey(hKey);
}
RegCloseKey(hKeyClassID);
}
RegCloseKey(hKeyCLSID);
}
if(!error)
hr = S_OK;
else
hr = HRESULT_FROM_WIN32(error);
return hr;
}
//+---------------------------------------------------------------------------
//
// Function: RegisterProxyDll
//
// Synopsis: Creates registry entries for the interfaces contained in a
// proxy DLL.
//
// Parameters: pszProxyDll - Supplies the filename of the proxy DLL.
//
// Notes: RegisterProxyDll calls the DllRegisterServer function
// exported from the proxy DLL.
//
//----------------------------------------------------------------------------
HRESULT RegisterProxyDll(const char *pszProxyDll)
{
HRESULT hr;
HINSTANCE hProxyDll;
HRESULT (STDAPICALLTYPE *pfnDllRegisterServer)();
hProxyDll = LoadLibraryA(pszProxyDll);
if(hProxyDll != 0)
{
pfnDllRegisterServer = (HRESULT (STDAPICALLTYPE *)())
GetProcAddress(hProxyDll, "DllRegisterServer");
if(pfnDllRegisterServer != 0)
hr = (*pfnDllRegisterServer)();
else
hr = HRESULT_FROM_WIN32(GetLastError());
FreeLibrary(hProxyDll);
}
else
{
hr = HRESULT_FROM_WIN32(GetLastError());
}
return hr;
}
//+---------------------------------------------------------------------------
//
// Function: UnregisterLocalServer
//
// Synopsis: Removes the registry key for a local server.
//
// Parameters: rclsid - Supplies the class ID of the local server.
//
//----------------------------------------------------------------------------
HRESULT UnregisterLocalServer(REFCLSID rclsid)
{
HRESULT hr;
HKEY hKeyCLSID;
HKEY hKeyClassID;
long error;
char szClassID[39];
//open the CLSID key
error = RegOpenKeyExA(
HKEY_CLASSES_ROOT,
"CLSID",
0,
KEY_ALL_ACCESS,
&hKeyCLSID);
if(!error)
{
//convert the class ID to a registry key name.
sprintf(szClassID, "{%08lX-%04X-%04X-%02X%02X-%02X%02X%02X%02X%02X%02X}",
rclsid.Data1, rclsid.Data2, rclsid.Data3,
rclsid.Data4[0], rclsid.Data4[1],
rclsid.Data4[2], rclsid.Data4[3],
rclsid.Data4[4], rclsid.Data4[5],
rclsid.Data4[6], rclsid.Data4[7]);
//open registry key for class ID string
error = RegOpenKeyExA(
hKeyCLSID,
szClassID,
0,
KEY_ALL_ACCESS,
&hKeyClassID);
if(!error)
{
//delete LocalServer32 key.
error = RegDeleteKeyA(hKeyClassID, "LocalServer32");
RegCloseKey(hKeyClassID);
}
error = RegDeleteKeyA(hKeyCLSID, szClassID);
RegCloseKey(hKeyCLSID);
}
if(!error)
hr = S_OK;
else
hr = HRESULT_FROM_WIN32(error);
return hr;
}
//+---------------------------------------------------------------------------
//
// Function: UnregisterProxyDll
//
// Synopsis: Removes registry entries for the interfaces contained in a
// proxy DLL.
//
// Parameters: pszProxyDll - Supplies the filename of the proxy DLL.
//
// Notes: UnregisterProxyDll calls the DllUnregisterServer function
// exported from the proxy DLL.
//
//----------------------------------------------------------------------------
HRESULT UnregisterProxyDll(const char *pszProxyDll)
{
HRESULT hr;
HINSTANCE hProxyDll;
HRESULT (STDAPICALLTYPE *pfnDllUnregisterServer)();
hProxyDll = LoadLibraryA(pszProxyDll);
if(hProxyDll != 0)
{
pfnDllUnregisterServer = (HRESULT (STDAPICALLTYPE *)())
GetProcAddress(hProxyDll, "DllUnregisterServer");
if(pfnDllUnregisterServer != 0)
hr = (*pfnDllUnregisterServer)();
else
hr = HRESULT_FROM_WIN32(GetLastError());
FreeLibrary(hProxyDll);
}
else
{
hr = HRESULT_FROM_WIN32(GetLastError());
}
return hr;
}
/* end file actestss.c */