Source code of Windows XP (NT5)
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.
 
 
 
 
 
 

1237 lines
32 KiB

/*++
Copyright (C) Microsoft Corporation, 1997 - 1999
Module Name:
senssink.cxx
Abstract:
Main entry point for the Sample SENS Subscriber.
Author:
Gopal Parupudi <GopalP>
[Notes:]
optional-notes
Revision History:
GopalP 11/17/1997 Start.
--*/
// Define this once and only once per exe.
#define INITGUIDS
#include <common.hxx>
#include <objbase.h>
#include <windows.h>
#include <ole2ver.h>
#include <initguid.h>
#include <eventsys.h>
#include <sensevts.h>
#include <sens.h>
#include "sinkcomn.hxx"
#include "sinkguid.hxx"
#include "senssink.hxx"
#include "cfacnet.hxx"
#include "cfaclogn.hxx"
#include "cfacpwr.hxx"
#if defined(SENS_NT4)
#define SENS_TLB SENS_BSTR("SENS.EXE")
#else // SENS_NT4
#define SENS_TLB SENS_BSTR("SENS.DLL")
#endif // SENS_NT4
#define SUBSCRIBER SENS_STRING("SENS_SUBSCRIBER: ")
#define MAX_QUERY_SIZE 512
#define MAJOR_VER 1
#define MINOR_VER 0
#define DEFAULT_LCID 0x0
//
// Globals
//
IEventSystem *gpIEventSystem;
ITypeInfo *gpITypeInfoNetwork;
ITypeInfo *gpITypeInfoLogon;
ITypeInfo *gpITypeInfoLogon2;
ITypeInfo *gpITypeInfoOnNow;
DWORD gTid;
HANDLE ghEvent;
#if defined(SENS_CHICAGO)
#ifdef DBG
DWORD gdwDebugOutputLevel;
#endif // DBG
#endif // SENS_CHICAGO
int __cdecl
main(
int argc,
char ** argv
)
{
HRESULT hr;
DWORD dwVer;
DWORD dwRegCO;
DWORD dwWaitStatus;
BOOL fInitialized;
BOOL bUnregister;
BOOL bSetupPhase;
MSG msg;
LPCLASSFACTORY pNetCF;
LPCLASSFACTORY pLogonCF;
LPCLASSFACTORY pLogon2CF;
LPCLASSFACTORY pPowerCF;
ITypeLib *pITypeLib;
hr = S_OK;
dwRegCO = 0x0;
bUnregister = FALSE;
bSetupPhase = FALSE;
fInitialized = FALSE;
pNetCF = NULL;
pLogonCF = NULL;
pLogon2CF = NULL;
pPowerCF = NULL;
gpIEventSystem = NULL;
pITypeLib = NULL;
gpITypeInfoNetwork = NULL;
gpITypeInfoLogon = NULL;
gpITypeInfoLogon2 = NULL;
gpITypeInfoOnNow = NULL;
dwVer = CoBuildVersion();
if (rmm != HIWORD(dwVer))
{
SensPrint(SENS_ERR, (SUBSCRIBER SENS_STRING("CoBuildVersion() returned incompatible version.\n")));
return -1;
}
if (FAILED(hr = CoInitializeEx(NULL, COINIT_MULTITHREADED)))
{
SensPrint(SENS_ERR, (SUBSCRIBER SENS_STRING("CoInitializeEx() returned 0x%x.\n"), hr));
goto Cleanup;
}
fInitialized = TRUE;
//
// Check the command-line args
//
if ( ((argc == 2) && ((argv[1][0] != '-') && (argv[1][0] != '/')))
|| (argc > 2))
{
Usage();
return (-1);
}
if (argc == 2)
{
switch (argv[1][1])
{
case 'i':
case 'I':
bSetupPhase = TRUE;
bUnregister = FALSE;
break;
case 'u':
case 'U':
bSetupPhase = TRUE;
bUnregister = TRUE;
break;
case 'e':
case 'E':
// SCM calls me with /Embedding flag
break;
default:
Usage();
return (-1);
}
}
if (bSetupPhase == TRUE)
{
hr = RegisterWithES(bUnregister);
if (FAILED(hr))
{
goto Cleanup;
}
// Network Subscriber
hr = RegisterSubscriberCLSID(
CLSID_SensTestSubscriberNetwork,
TEST_SUBSCRIBER_NAME_NETWORK,
bUnregister
);
if (FAILED(hr))
{
SensPrint(SENS_ERR, (SUBSCRIBER SENS_STRING("Failed to %sregister Test Subscriber Network CLSID")
SENS_STRING(" - hr = <%x>\n"), bUnregister ? SENS_STRING("un") : SENS_STRING(""), hr));
}
else
{
SensPrint(SENS_INFO, (SUBSCRIBER SENS_STRING("%segistered Test Subscriber Network CLSID.\n"),
bUnregister ? SENS_STRING("Unr") : SENS_STRING("R")));
}
// Logon Subscriber
hr = RegisterSubscriberCLSID(
CLSID_SensTestSubscriberLogon,
TEST_SUBSCRIBER_NAME_LOGON,
bUnregister
);
if (FAILED(hr))
{
SensPrint(SENS_ERR, (SUBSCRIBER SENS_STRING("Failed to %sregister Test Subscriber Logon CLSID")
SENS_STRING(" - hr = <%x>\n"), bUnregister ? SENS_STRING("un") : SENS_STRING(""), hr));
}
else
{
SensPrint(SENS_INFO, (SUBSCRIBER SENS_STRING("%segistered Test Subscriber Logon CLSID.\n"),
bUnregister ? SENS_STRING("Unr") : SENS_STRING("R")));
}
// Logon2 Subscriber
hr = RegisterSubscriberCLSID(
CLSID_SensTestSubscriberLogon2,
TEST_SUBSCRIBER_NAME_LOGON2,
bUnregister
);
if (FAILED(hr))
{
SensPrint(SENS_ERR, (SUBSCRIBER SENS_STRING("Failed to %sregister Test Subscriber Logon2 CLSID")
SENS_STRING(" - hr = <%x>\n"), bUnregister ? SENS_STRING("un") : SENS_STRING(""), hr));
}
else
{
SensPrint(SENS_INFO, (SUBSCRIBER SENS_STRING("%segistered Test Subscriber Logon2 CLSID.\n"),
bUnregister ? SENS_STRING("Unr") : SENS_STRING("R")));
}
// Power Subscriber
hr = RegisterSubscriberCLSID(
CLSID_SensTestSubscriberOnNow,
TEST_SUBSCRIBER_NAME_POWER,
bUnregister
);
if (FAILED(hr))
{
SensPrint(SENS_ERR, (SUBSCRIBER SENS_STRING("Failed to %sregister Test Subscriber Power CLSID")
SENS_STRING(" - hr = <%x>\n"), bUnregister ? SENS_STRING("un") : SENS_STRING(""), hr));
}
else
{
SensPrint(SENS_INFO, (SUBSCRIBER SENS_STRING("%segistered Test Subscriber Power CLSID.\n"),
bUnregister ? SENS_STRING("Unr") : SENS_STRING("R")));
}
goto Cleanup;
}
//
// Get the ITypeInfo pointer.
//
hr = LoadRegTypeLib(
LIBID_SensEvents,
MAJOR_VER,
MINOR_VER,
DEFAULT_LCID,
&pITypeLib
);
SensPrint(SENS_INFO, (SUBSCRIBER SENS_STRING("LoadRegTypeLib() returned 0x%x\n"), hr));
if (FAILED(hr))
{
hr = LoadTypeLib(
SENS_TLB,
&pITypeLib
);
SensPrint(SENS_INFO, (SUBSCRIBER SENS_STRING("LoadTypeLib() returned 0x%x\n"), hr));
if (FAILED(hr))
{
SensPrint(SENS_ERR, (SUBSCRIBER SENS_STRING("Exiting...\n")));
goto Cleanup;
}
}
// Get type information for the ISensNetwork interface.
hr = pITypeLib->GetTypeInfoOfGuid(
IID_ISensNetwork,
&gpITypeInfoNetwork
);
SensPrint(SENS_INFO, (SUBSCRIBER SENS_STRING("ITypeLib::GetTypeInfoOfGuid(ISensNetwork) returned 0x%x\n"), hr));
if (FAILED(hr))
{
SensPrint(SENS_ERR, (SUBSCRIBER SENS_STRING("Exiting...\n")));
goto Cleanup;
}
// Get type information for the ISensLogon interface.
hr = pITypeLib->GetTypeInfoOfGuid(
IID_ISensLogon,
&gpITypeInfoLogon
);
SensPrint(SENS_INFO, (SUBSCRIBER SENS_STRING("ITypeLib::GetTypeInfoOfGuid(ISensLogon) returned 0x%x\n"), hr));
if (FAILED(hr))
{
SensPrint(SENS_ERR, (SUBSCRIBER SENS_STRING("Exiting...\n")));
goto Cleanup;
}
// Get type information for the ISensLogon2 interface.
hr = pITypeLib->GetTypeInfoOfGuid(
IID_ISensLogon2,
&gpITypeInfoLogon2
);
SensPrint(SENS_INFO, (SUBSCRIBER SENS_STRING("ITypeLib::GetTypeInfoOfGuid(ISensLogon2) returned 0x%x\n"), hr));
if (FAILED(hr))
{
SensPrint(SENS_ERR, (SUBSCRIBER SENS_STRING("Exiting...\n")));
goto Cleanup;
}
// Get type information for the ISensOnNow interface.
hr = pITypeLib->GetTypeInfoOfGuid(
IID_ISensOnNow,
&gpITypeInfoOnNow
);
SensPrint(SENS_INFO, (SUBSCRIBER SENS_STRING("ITypeLib::GetTypeInfoOfGuid(ISensOnNow) returned 0x%x\n"), hr));
if (FAILED(hr))
{
SensPrint(SENS_ERR, (SUBSCRIBER SENS_STRING("Exiting...\n")));
goto Cleanup;
}
//
// Create the event
//
ghEvent = CreateEvent(
NULL, // Security Attributes
FALSE, // bManualReset
FALSE, // Initial state
SENS_STRING("SENS Test Subscriber Quit Event")
);
if (ghEvent == NULL)
{
SensPrint(SENS_ERR, (SUBSCRIBER SENS_STRING("CreateEvent() failed.\n")));
goto Cleanup;
}
//
// Create the Network ClassFactory and register it with COM.
//
pNetCF = new CISensNetworkCF;
if (NULL == pNetCF)
{
goto Cleanup;
}
pNetCF->AddRef(); // Because we hold on to it.
SensPrint(SENS_INFO, (SUBSCRIBER SENS_STRING("ClassFactory created successfully.\n")));
// Register the CLSID
hr = CoRegisterClassObject(
CLSID_SensTestSubscriberNetwork,
(LPUNKNOWN) pNetCF,
CLSCTX_LOCAL_SERVER,
REGCLS_MULTIPLEUSE,
&dwRegCO
);
if (FAILED(hr))
{
SensPrint(SENS_ERR, (SUBSCRIBER SENS_STRING("CoRegisterClassObject(Network) returned 0x%x.\n"), hr));
}
else
{
SensPrint(SENS_INFO, (SUBSCRIBER SENS_STRING("Successfully registered the ISensNetwork Class Factory.\n")));
}
//
// Create the Logon ClassFactory and register it with COM.
//
pLogonCF = new CISensLogonCF;
if (NULL == pLogonCF)
{
goto Cleanup;
}
pLogonCF->AddRef(); // Because we hold on to it.
SensPrint(SENS_INFO, (SUBSCRIBER SENS_STRING("ClassFactory created successfully.\n")));
hr = CoRegisterClassObject(
CLSID_SensTestSubscriberLogon,
(LPUNKNOWN) pLogonCF,
CLSCTX_LOCAL_SERVER,
REGCLS_MULTIPLEUSE,
&dwRegCO
);
if (FAILED(hr))
{
SensPrint(SENS_ERR, (SUBSCRIBER SENS_STRING("CoRegisterClassObject(Logon) returned 0x%x.\n"), hr));
}
else
{
SensPrint(SENS_INFO, (SUBSCRIBER SENS_STRING("Successfully registered the ISensLogon Class Factory.\n")));
}
//
// Create the Logon2 ClassFactory and register it with COM.
//
pLogon2CF = new CISensLogon2CF;
if (NULL == pLogon2CF)
{
goto Cleanup;
}
pLogon2CF->AddRef(); // Because we hold on to it.
SensPrint(SENS_INFO, (SUBSCRIBER SENS_STRING("ClassFactory created successfully.\n")));
hr = CoRegisterClassObject(
CLSID_SensTestSubscriberLogon2,
(LPUNKNOWN) pLogon2CF,
CLSCTX_LOCAL_SERVER,
REGCLS_MULTIPLEUSE,
&dwRegCO
);
if (FAILED(hr))
{
SensPrint(SENS_ERR, (SUBSCRIBER SENS_STRING("CoRegisterClassObject(Logon2) returned 0x%x.\n"), hr));
}
else
{
SensPrint(SENS_INFO, (SUBSCRIBER SENS_STRING("Successfully registered the ISensLogon2 Class Factory.\n")));
}
//
// Create the Power ClassFactory and register it with COM.
//
pPowerCF = new CISensOnNowCF;
if (NULL == pPowerCF)
{
goto Cleanup;
}
pPowerCF->AddRef(); // Because we hold on to it.
SensPrint(SENS_INFO, (SUBSCRIBER SENS_STRING("ClassFactory created successfully.\n")));
hr = CoRegisterClassObject(
CLSID_SensTestSubscriberOnNow,
(LPUNKNOWN) pPowerCF,
CLSCTX_LOCAL_SERVER,
REGCLS_MULTIPLEUSE,
&dwRegCO
);
if (FAILED(hr))
{
SensPrint(SENS_ERR, (SUBSCRIBER SENS_STRING("CoRegisterClassObject(Power) returned 0x%x.\n"), hr));
}
else
{
SensPrint(SENS_INFO, (SUBSCRIBER SENS_STRING("Successfully registered the ISensOnNow Class Factory.\n")));
}
//
// Wait to quit.
//
dwWaitStatus = WaitForSingleObject(ghEvent, INFINITE);
SensPrint(SENS_INFO, (SUBSCRIBER SENS_STRING("WaitForSingleObject returned %d\n"), dwWaitStatus));
//
// Cleanup
//
Cleanup:
if ( (0L != dwRegCO)
&& (bSetupPhase == TRUE)
&& (bUnregister == TRUE))
{
CoRevokeClassObject(dwRegCO);
SensPrint(SENS_INFO, (SUBSCRIBER SENS_STRING("CoRevokeClassObject() returned 0x%x.\n"), hr));
}
if (bSetupPhase == TRUE)
{
SensPrint(SENS_INFO, (SENS_STRING("\n") SUBSCRIBER SENS_STRING("Sens Test Subscriber Configuration %s.\n"),
FAILED(hr) ? SENS_STRING("failed") : SENS_STRING("successful")));
}
if (NULL != pNetCF)
{
pNetCF->Release();
}
if (NULL != pLogonCF)
{
pLogonCF->Release();
}
if (NULL != pLogon2CF)
{
pLogon2CF->Release();
}
if (NULL != pPowerCF)
{
pPowerCF->Release();
}
if (NULL != pITypeLib)
{
pITypeLib->Release();
}
if (NULL != gpITypeInfoNetwork)
{
gpITypeInfoNetwork->Release();
}
if (NULL != gpITypeInfoLogon)
{
gpITypeInfoLogon->Release();
}
if (NULL != gpITypeInfoLogon2)
{
gpITypeInfoLogon2->Release();
}
if (NULL != gpITypeInfoOnNow)
{
gpITypeInfoOnNow->Release();
}
if (fInitialized)
{
CoUninitialize();
}
return 0;
}
HRESULT
RegisterWithES(
BOOL bUnregister
)
{
HRESULT hr;
hr = S_OK;
//
// Instantiate the Event System
//
hr = CoCreateInstance(
CLSID_CEventSystem,
NULL,
CLSCTX_SERVER,
IID_IEventSystem,
(LPVOID *) &gpIEventSystem
);
if (FAILED(hr))
{
SensPrint(SENS_ERR, (SUBSCRIBER SENS_STRING("Failed to create CEventSystem, HRESULT=%x\n"), hr));
goto Cleanup;
}
SensPrint(SENS_INFO, (SUBSCRIBER SENS_STRING("Successfully created CEventSystem\n")));
#if 0
//
// Test to see if we can get the MachineName
//
BSTR bstrMachineName;
hr = gpIEventQuery->get_MachineName(&bstrMachineName);
if (SUCCEEDED(hr))
SensPrint(SENS_INFO, (SUBSCRIBER SENS_STRING("Configuring Test Subscriber on %s...\n"), bstrMachineName));
else
SensPrint(SENS_ERR, (SUBSCRIBER SENS_STRING("get_MachineName failed, hr = %x\n"), hr));
#endif // 0
//
// Register my Subscriber's subscriptions with SENS.
//
hr = RegisterSubscriptions(bUnregister);
if (FAILED(hr))
{
SensPrint(SENS_ERR, (SUBSCRIBER SENS_STRING("Failed to %sregister SensSink Subscriptions")
SENS_STRING(" - hr = <%x>\n"), bUnregister ? SENS_STRING("un") : SENS_STRING(""), hr));
goto Cleanup;
}
SensPrint(SENS_INFO, (SUBSCRIBER SENS_STRING("%segistered SensSink Subscriptions.\n"),
bUnregister ? SENS_STRING("Unr") : SENS_STRING("R")));
Cleanup:
//
// Cleanup
//
if (gpIEventSystem)
{
gpIEventSystem->Release();
}
return hr;
}
HRESULT
RegisterSubscriptions(
BOOL bUnregister
)
{
int i;
int errorIndex;
HRESULT hr;
LPOLESTR strGuid;
LPOLESTR strSubscriptionID;
WCHAR szQuery[MAX_QUERY_SIZE];
BSTR bstrEventClassID;
BSTR bstrInterfaceID;
BSTR bstrSubscriberCLSID;
BSTR bstrPublisherID;
BSTR bstrSubscriptionID;
BSTR bstrSubscriptionName;
BSTR bstrMethodName;
BSTR bstrPropertyName;
BSTR bstrPropertyValue;
ULONG ulPropertyValue;
VARIANT variantPropertyValue;
BSTR bstrPROGID_EventSubscription;
IEventSubscription *pIEventSubscription;
hr = S_OK;
strGuid = NULL;
errorIndex = 0;
strSubscriptionID = NULL;
bstrEventClassID = NULL;
bstrInterfaceID = NULL;
bstrSubscriberCLSID = NULL;
bstrPublisherID = NULL;
bstrSubscriptionID = NULL;
bstrSubscriptionName = NULL;
bstrMethodName = NULL;
bstrPropertyName = NULL;
bstrPropertyValue = NULL;
ulPropertyValue = 0x0;
bstrPROGID_EventSubscription = NULL;
pIEventSubscription = NULL;
//
// Build a Subscriber
//
AllocateBstrFromGuid(bstrPublisherID, SENSGUID_PUBLISHER);
AllocateBstrFromString(bstrPROGID_EventSubscription, PROGID_EventSubscription);
for (i = 0; i < TEST_SUBSCRIPTIONS_COUNT; i++)
{
if (bUnregister)
{
// Form the query
wcscpy(szQuery, SENS_BSTR("SubscriptionID"));
wcscat(szQuery, SENS_BSTR("="));
AllocateStrFromGuid(strSubscriptionID, *(gTestSubscriptions[i].pSubscriptionID));
wcscat(szQuery, strSubscriptionID);
hr = gpIEventSystem->Remove(
PROGID_EventSubscription,
szQuery,
&errorIndex
);
FreeStr(strSubscriptionID);
if (FAILED(hr))
{
SensPrint(SENS_ERR, (SUBSCRIBER SENS_STRING("RegisterSubscriptions(%d) failed to unregister")
SENS_STRING(" - hr = <%x>\n"), i, hr));
goto Cleanup;
}
continue;
}
// Get a new IEventSubscription object to play with.
hr = CoCreateInstance(
CLSID_CEventSubscription,
NULL,
CLSCTX_SERVER,
IID_IEventSubscription,
(LPVOID *) &pIEventSubscription
);
if (FAILED(hr))
{
SensPrint(SENS_ERR, (SUBSCRIBER SENS_STRING("RegisterSubscriptions(%d) failed to create ")
SENS_STRING("IEventSubscriptions - hr = <%x>\n"), i, hr));
goto Cleanup;
}
AllocateBstrFromGuid(bstrSubscriptionID, *(gTestSubscriptions[i].pSubscriptionID));
hr = pIEventSubscription->put_SubscriptionID(bstrSubscriptionID);
ASSERT(SUCCEEDED(hr));
AllocateBstrFromGuid(bstrSubscriberCLSID, *(gTestSubscriptions[i].pSubscriberCLSID));
hr = pIEventSubscription->put_SubscriberCLSID(bstrSubscriberCLSID);
ASSERT(SUCCEEDED(hr));
hr = pIEventSubscription->put_PublisherID(bstrPublisherID);
ASSERT(SUCCEEDED(hr));
hr = pIEventSubscription->put_SubscriptionID(bstrSubscriptionID);
ASSERT(SUCCEEDED(hr));
AllocateBstrFromString(bstrSubscriptionName, gTestSubscriptions[i].strSubscriptionName);
hr = pIEventSubscription->put_SubscriptionName(bstrSubscriptionName);
ASSERT(SUCCEEDED(hr));
AllocateBstrFromString(bstrMethodName, gTestSubscriptions[i].strMethodName);
hr = pIEventSubscription->put_MethodName(bstrMethodName);
ASSERT(SUCCEEDED(hr));
AllocateBstrFromGuid(bstrEventClassID, *(gTestSubscriptions[i].pEventClassID));
hr = pIEventSubscription->put_EventClassID(bstrEventClassID);
ASSERT(SUCCEEDED(hr));
AllocateBstrFromGuid(bstrInterfaceID, *(gTestSubscriptions[i].pInterfaceID));
hr = pIEventSubscription->put_InterfaceID(bstrInterfaceID);
ASSERT(SUCCEEDED(hr));
if (wcscmp(gTestSubscriptions[i].strMethodName, SENS_BSTR("ConnectionMadeNoQOCInfo")) == 0)
{
AllocateBstrFromString(bstrPropertyName, SENS_BSTR("ulConnectionMadeTypeNoQOC"));
ulPropertyValue = CONNECTION_LAN;
InitializeDwordVariant(&variantPropertyValue, ulPropertyValue);
hr = pIEventSubscription->PutPublisherProperty(
bstrPropertyName,
&variantPropertyValue
);
ASSERT(SUCCEEDED(hr));
SensPrint(SENS_INFO, (SUBSCRIBER SENS_STRING("PutPublisherProperty(WAN/LAN) returned 0x%x\n"), hr));
FreeBstr(bstrPropertyName);
ulPropertyValue = 0x0;
}
else
if ( (wcscmp(gTestSubscriptions[i].strMethodName, SENS_BSTR("DestinationReachable")) == 0)
|| (wcscmp(gTestSubscriptions[i].strMethodName, SENS_BSTR("DestinationReachableNoQOCInfo")) == 0))
{
// Set the DestinationName for which we want to be notified for.
AllocateBstrFromString(bstrPropertyName, gTestSubscriptions[i].strPropertyMethodName);
AllocateBstrFromString(bstrPropertyValue, gTestSubscriptions[i].strPropertyMethodNameValue);
InitializeBstrVariant(&variantPropertyValue, bstrPropertyValue);
hr = pIEventSubscription->PutPublisherProperty(
bstrPropertyName,
&variantPropertyValue
);
ASSERT(SUCCEEDED(hr));
SensPrint(SENS_INFO, (SUBSCRIBER SENS_STRING("PutPublisherProperty(DestinationName) returned 0x%x\n"), hr));
FreeBstr(bstrPropertyName);
FreeBstr(bstrPropertyValue);
}
FreeBstr(bstrSubscriptionID);
FreeBstr(bstrSubscriberCLSID);
FreeBstr(bstrEventClassID);
FreeBstr(bstrInterfaceID);
FreeBstr(bstrSubscriptionName);
FreeBstr(bstrMethodName);
hr = gpIEventSystem->Store(bstrPROGID_EventSubscription, pIEventSubscription);
if (FAILED(hr))
{
SensPrint(SENS_ERR, (SUBSCRIBER SENS_STRING("RegisterSubscriptions(%d) failed to commit")
SENS_STRING(" - hr = <%x>\n"), i, hr));
goto Cleanup;
}
pIEventSubscription->Release();
pIEventSubscription = NULL;
} // for loop
Cleanup:
//
// Cleanup
//
if (pIEventSubscription)
{
pIEventSubscription->Release();
}
FreeBstr(bstrPublisherID);
FreeBstr(bstrSubscriberCLSID);
FreeStr(strGuid);
return (hr);
}
HRESULT
RegisterSubscriberCLSID(
REFIID clsid,
TCHAR* strSubscriberName,
BOOL bUnregister
)
/*++
Routine Description:
Register/Unregister the CLSID of the Test subscriber.
Arguments:
clsid - CLSID of the Subscriber.
strSubscriberName - Name of the Subscriber.
bUnregister - If TRUE, then unregister all subscriptions of SENS Test subscriber.
Return Value:
S_OK, if successful
hr, otherwise
--*/
{
HRESULT hr;
HMODULE hModule;
HKEY appidKey = 0;
HKEY clsidKey = 0;
HKEY serverKey = 0;
TCHAR szModule[MAX_PATH+1];
WCHAR *szCLSID = 0;
WCHAR *szLIBID = 0;
TCHAR *szCLSID_t = 0;
TCHAR *szLIBID_t = 0;
TCHAR *szFriendlyName = strSubscriberName;
hr = S_OK;
// Convert the CLSID into a TCHAR.
hr = StringFromCLSID(clsid, &szCLSID);
if (FAILED(hr))
{
goto Cleanup;
}
if (bUnregister == FALSE)
{
hr = StringFromCLSID(LIBID_SensEvents, &szLIBID);
if (FAILED(hr))
{
goto Cleanup;
}
hModule = GetModuleHandle(0);
// Get Subscriber location.
const size_t moduleBufSize = sizeof szModule / sizeof szModule[0];
DWORD dwResult = GetModuleFileName(hModule, szModule, moduleBufSize);
if (dwResult == 0)
{
hr = HRESULT_FROM_WIN32(ERROR_FILE_NOT_FOUND);
goto Cleanup;
}
}
//
// Convert UNICODE strings into ANSI, if necessary
//
#if !defined(SENS_CHICAGO)
szCLSID_t = szCLSID;
szLIBID_t = szLIBID;
#else // SENS_CHICAGO
szCLSID_t = SensUnicodeStringToAnsi(szCLSID);
szLIBID_t = SensUnicodeStringToAnsi(szLIBID);
if ( (NULL == szCLSID_t)
|| (NULL == szLIBID_t))
{
goto Cleanup;
}
#endif // SENS_CHICAGO
// Build the key CLSID\\{clsid}
TCHAR clsidKeyName[sizeof "CLSID\\{12345678-1234-1234-1234-123456789012}"];
_tcscpy(clsidKeyName, SENS_STRING("CLSID\\"));
_tcscat(clsidKeyName, szCLSID_t);
// Build the key AppID\\{clsid}
TCHAR appidKeyName[sizeof "AppID\\{12345678-1234-1234-1234-123456789012}"];
_tcscpy(appidKeyName, SENS_STRING("AppID\\"));
_tcscat(appidKeyName, szCLSID_t);
if (bUnregister)
{
hr = RecursiveDeleteKey(HKEY_CLASSES_ROOT, clsidKeyName);
if (FAILED(hr))
{
goto Cleanup;
}
hr = RecursiveDeleteKey(HKEY_CLASSES_ROOT, appidKeyName);
goto Cleanup;
}
// Create the CLSID\\{clsid} key
hr = CreateKey(
HKEY_CLASSES_ROOT,
clsidKeyName,
szFriendlyName,
&clsidKey
);
if (FAILED(hr))
{
goto Cleanup;
}
//
// Under the CLSID\\{clsid} key, create a named value
// AppID = {clsid}
hr = CreateNamedValue(clsidKey, SENS_STRING("AppID"), szCLSID_t);
if (FAILED(hr))
{
goto Cleanup;
}
//
// Create the appropriate server key beneath the clsid key.
// For servers, this is CLSID\\{clsid}\\LocalServer32.
// In both cases, the default value is the module path name.
//
hr = CreateKey(
clsidKey,
SENS_STRING("LocalServer32"),
szModule,
&serverKey
);
if (FAILED(hr))
{
goto Cleanup;
}
RegCloseKey(serverKey);
//
// Create CLSID\\{clsid}\\TypeLib subkey with a default value of
// the LIBID of the TypeLib
//
hr = CreateKey(
clsidKey,
SENS_STRING("TypeLib"),
szLIBID_t,
&serverKey
);
if (FAILED(hr))
{
goto Cleanup;
}
RegCloseKey(serverKey);
// We're finished with the CLSID\\{clsid} key
RegCloseKey(clsidKey);
// Register APPID.
hr = CreateKey(
HKEY_CLASSES_ROOT,
appidKeyName,
szFriendlyName,
&appidKey
);
if (FAILED(hr))
{
goto Cleanup;
}
// Under AppId\{clsid} key, create a named value [RunAs = "Interactive User"]
hr = CreateNamedValue(appidKey, SENS_STRING("RunAs"), SENS_STRING("Interactive User"));
if (FAILED(hr))
{
goto Cleanup;
}
RegCloseKey(appidKey);
Cleanup:
//
// Cleanup
//
CoTaskMemFree(szCLSID);
CoTaskMemFree(szLIBID);
#if defined(SENS_CHICAGO)
if (szCLSID_t != NULL)
{
delete szCLSID_t;
}
if (szLIBID_t != NULL)
{
delete szLIBID_t;
}
#endif // SENS_CHICAGO
return hr;
}
HRESULT
CreateKey(
HKEY hParentKey,
const TCHAR* KeyName,
const TCHAR* defaultValue,
HKEY* hKey
)
/*++
Routine Description:
Create a key (with an optional default value). The handle to the key is
returned as an [out] parameter. If NULL is passed as the key parameter,
the key is created in the registry, then closed.
Arguments:
hParentKey - Handle to the parent Key.
KeyName - Name of the key to create.
defaultValue - The default value for the key to create.
hKey - OUT Handle to key that was created.
Return Value:
S_OK, if successful
hr, otherwise
--*/
{
HKEY hTempKey;
LONG lResult;
hTempKey = NULL;
lResult = RegCreateKeyEx(
hParentKey, // Handle to open key
KeyName, // Subkey name
0, // Reserved
NULL, // Class string
REG_OPTION_NON_VOLATILE, // Options Flag
KEY_ALL_ACCESS, // Desired Security access
NULL, // Pointer to Security Attributes structure
&hTempKey, // Handle of the opened/created key
NULL // Disposition value
);
if (lResult != ERROR_SUCCESS)
{
return HRESULT_FROM_WIN32(lResult);
}
// Set the default value for the key
if (defaultValue != NULL)
{
lResult = RegSetValueEx(
hTempKey, // Key to set Value for.
NULL, // Value to set
0, // Reserved
REG_SZ, // Value Type
(BYTE*) defaultValue, // Address of Value data
sizeof(TCHAR) * (_tcslen(defaultValue)+1) // Size of Value
);
if (lResult != ERROR_SUCCESS)
{
RegCloseKey(hTempKey);
return HRESULT_FROM_WIN32(lResult);
}
}
if (hKey == NULL)
{
RegCloseKey(hTempKey);
}
else
{
*hKey = hTempKey;
}
return S_OK;
}
HRESULT
CreateNamedValue(
HKEY hKey,
const TCHAR* title,
const TCHAR* value
)
/*++
Routine Description:
Create a named value under a key
Arguments:
hKey - Handle to the parent Key.
title - Name of the Value to create.
value - The data for the Value under the Key.
Return Value:
S_OK, if successful
hr, otherwise
--*/
{
HRESULT hr;
LONG lResult;
hr = S_OK;
lResult = RegSetValueEx(
hKey, // Key to set Value for.
title, // Value to set
0, // Reserved
REG_SZ, // Value Type
(BYTE*) value, // Address of Value data
sizeof(TCHAR) * (_tcslen(value)+1) // Size of Value
);
if (lResult != ERROR_SUCCESS)
{
hr = HRESULT_FROM_WIN32(lResult);
}
return hr;
}
HRESULT
RecursiveDeleteKey(
HKEY hKeyParent,
const TCHAR* lpszKeyChild
)
/*++
Routine Description:
Delete a key and all of its descendents.
Arguments:
hKeyParent - Handle to the parent Key.
lpszKeyChild - The data for the Value under the Key.
Return Value:
S_OK, if successful
hr, otherwise
--*/
{
HKEY hKeyChild;
LONG lResult;
//
// Open the child.
//
lResult = RegOpenKeyEx(
hKeyParent, // Handle to the Parent
lpszKeyChild, // Name of the child key
0, // Reserved
KEY_ALL_ACCESS, // Security Access Mask
&hKeyChild // Handle to the opened key
);
if (lResult != ERROR_SUCCESS)
{
return HRESULT_FROM_WIN32(lResult);
}
//
// Enumerate all of the decendents of this child.
//
FILETIME time;
TCHAR szBuffer[MAX_PATH+1];
const DWORD bufSize = sizeof szBuffer / sizeof szBuffer[0];
DWORD dwSize = bufSize;
while (TRUE)
{
lResult = RegEnumKeyEx(
hKeyChild, // Handle of the key to enumerate
0, // Index of the subkey to retrieve
szBuffer, // OUT Name of the subkey
&dwSize, // OUT Size of the buffer for name of subkey
NULL, // Reserved
NULL, // OUT Class of the enumerated subkey
NULL, // OUT Size of the class of the subkey
&time // OUT Last time the subkey was written to
);
if (lResult != ERROR_SUCCESS)
{
break;
}
// Delete the decendents of this child.
lResult = RecursiveDeleteKey(hKeyChild, szBuffer);
if (lResult != ERROR_SUCCESS)
{
// Cleanup before exiting.
RegCloseKey(hKeyChild);
return HRESULT_FROM_WIN32(lResult);
}
dwSize = bufSize;
} // while
// Close the child.
RegCloseKey(hKeyChild);
// Delete this child.
lResult = RegDeleteKey(hKeyParent, lpszKeyChild);
return HRESULT_FROM_WIN32(lResult);
}