|
|
/***************************************************************************
* * Copyright (C) 2001-2002 Microsoft Corporation. All Rights Reserved. * * File: dpnhpastdllmain.cpp * * Content: DPNHPAST DLL entry points. * * History: * Date By Reason * ======== ======== ========= * 04/16/01 VanceO Split DPNATHLP into DPNHUPNP and DPNHPAST. * ***************************************************************************/
#include "dpnhpasti.h"
//=============================================================================
// External globals
//=============================================================================
volatile LONG g_lOutstandingInterfaceCount = 0; // number of outstanding interfaces
DNCRITICAL_SECTION g_csGlobalsLock; // lock protecting all of the following globals
CBilink g_blNATHelpPASTObjs; // bilink of all the NATHelpPAST interface objects
DWORD g_dwHoldRand; // current random number sequence
DWORD g_dwPASTICSMode; // whether using PAST ICS is enabled or not
DWORD g_dwPASTPFWMode; // whether using PAST PFW is enabled or not
INT g_iUnicastTTL = 1; // unicast TTL value, or 0 to use default set by OS; normally we use 1
DWORD g_dwDefaultGatewayV4 = INADDR_BROADCAST; // default to broadcasting when searching for PAST server
DWORD g_dwSubnetMaskV4 = 0x00FFFFFF; // = 0xFFFFFF00 in Intel order = 255.255.255.0, a class C network
DWORD g_dwNoActiveNotifyPollInterval = 25000; // start by polling every 25 seconds
DWORD g_dwMinUpdateServerStatusInterval = 1000; // don't hit the network more often than every 1 second
DWORD g_dwPollIntervalBackoff = 30000; // backoff an additional 0 to 30 seconds if no network changes occur
DWORD g_dwMaxPollInterval = 300000; // don't go more than 5 minutes without polling
BOOL g_fKeepPollingForRemoteGateway = FALSE; // whether to continue to search for new Internet gateway devices if none were found during startup
DWORD g_dwReusePortTime = 60000; // how long to keep using the same port for polling remote Internet gateway devices (default is 1 minute)
DWORD g_dwCacheLifeFound = 30000; // how long to cache QueryAddress results where the address was found
DWORD g_dwCacheLifeNotFound = 30000; // how long to cache QueryAddress results where the address was not found
//=============================================================================
// Defines
//=============================================================================
#define REGKEY_VALUE_GUID L"Guid"
#define REGKEY_VALUE_DIRECTPLAY8PRIORITY L"DirectPlay8Priority"
#define REGKEY_VALUE_DIRECTPLAY8INITFLAGS L"DirectPlay8InitFlags"
#define REGKEY_VALUE_PASTICSMODE L"PASTICSMode"
#define REGKEY_VALUE_PASTPFWMODE L"PASTPFWMode"
#define REGKEY_VALUE_UNICASTTTL L"UnicastTTL"
#define REGKEY_VALUE_DEFAULTGATEWAYV4 L"GatewayV4"
#define REGKEY_VALUE_SUBNETMASKV4 L"SubnetMaskV4"
#define REGKEY_VALUE_NOACTIVENOTIFYPOLLINTERVAL L"NoActiveNotifyPollInterval"
#define REGKEY_VALUE_MINUPDATESERVERSTATUSINTERVAL L"MinUpdateServerStatusInterval"
#define REGKEY_VALUE_POLLINTERVALBACKOFF L"PollIntervalBackoff"
#define REGKEY_VALUE_MAXPOLLINTERVAL L"MaxPollInterval"
#define REGKEY_VALUE_KEEPPOLLINGFORREMOTEGATEWAY L"KeepPollingForRemoteGateway"
#define REGKEY_VALUE_REUSEPORTTIME L"ReusePortTime"
#define REGKEY_VALUE_CACHELIFEFOUND L"CacheLifeFound"
#define REGKEY_VALUE_CACHELIFENOTFOUND L"CacheLifeNotFound"
#define DEFAULT_DIRECTPLAY8PRIORITY 2
#define DEFAULT_DIRECTPLAY8INITFLAGS DPNHINITIALIZE_DISABLELOCALFIREWALLSUPPORT
//=============================================================================
// Local prototypes
//=============================================================================
BOOL InitializeProcessGlobals(void); void CleanupProcessGlobals(void); void InitializeGlobalRand(const DWORD dwSeed);
#undef DPF_MODNAME
#define DPF_MODNAME "DllMain"
//=============================================================================
// DllMain
//-----------------------------------------------------------------------------
//
// Description: DLL entry point.
//
// Arguments:
// HANDLE hDllInst - Handle to this DLL module.
// DWORD dwReason - Reason for calling this function.
// LPVOID lpvReserved - Reserved.
//
// Returns: TRUE if all goes well, FALSE otherwise.
//=============================================================================
BOOL WINAPI DllMain(HANDLE hDllInst, DWORD dwReason, LPVOID lpvReserved) { BOOL fResult = TRUE;
switch (dwReason) { case DLL_PROCESS_ATTACH: { DPFX(DPFPREP, 2, "====> ENTER: DLLMAIN(%p): Process Attach: %08lx, tid=%08lx", DllMain, GetCurrentProcessId(), GetCurrentThreadId());
//
// Ignore thread attach/detach messages.
//
DisableThreadLibraryCalls((HMODULE) hDllInst);
//
// Attempt to initialize the OS abstraction layer.
//
if (DNOSIndirectionInit(0)) { //
// Attempt to initialize process-global items.
//
if (! InitializeProcessGlobals()) { DPFX(DPFPREP, 0, "Failed to initialize globals!"); DNOSIndirectionDeinit(); fResult = FALSE; } } else { DPFX(DPFPREP, 0, "Failed to initialize OS indirection layer!"); fResult = FALSE; } break; }
case DLL_THREAD_ATTACH: { //
// Ignore.
//
break; }
case DLL_THREAD_DETACH: { //
// Ignore.
//
break; }
case DLL_PROCESS_DETACH: { DPFX(DPFPREP, 2, "====> EXIT: DLLMAIN(%p): Process Detach %08lx, tid=%08lx", DllMain, GetCurrentProcessId(), GetCurrentThreadId());
CleanupProcessGlobals();
DNOSIndirectionDeinit(); break; }
default: { DNASSERT(FALSE); break; } }
return fResult; } // DllMain
#ifndef DPNBUILD_NOCOMREGISTER
#undef DPF_MODNAME
#define DPF_MODNAME "DllRegisterServer"
//=============================================================================
// DllRegisterServer
//-----------------------------------------------------------------------------
//
// Description: Registers the DirectPlay NAT Helper PAST COM object.
//
// Arguments: None.
//
// Returns: HRESULT
// S_OK - Successfully unregistered DirectPlay NAT Helper PAST.
// E_FAIL - Failed unregistering DirectPlay NAT Helper PAST.
//=============================================================================
HRESULT WINAPI DllRegisterServer(void) { CRegistry RegObject;
//
// Register this COM object CLSID.
//
if (! CRegistry::Register(L"DirectPlayNATHelperPAST.1", L"DirectPlay NAT Helper PAST Object", L"dpnhpast.dll", &CLSID_DirectPlayNATHelpPAST, L"DirectPlayNATHelperPAST")) { DPFX(DPFPREP, 0, "Could not register DirectPlay NAT Helper PAST object!"); return E_FAIL; }
//
// Write this object's GUID and DirectPlay8 availability to the registry.
//
if (! RegObject.Open(HKEY_LOCAL_MACHINE, DIRECTPLAYNATHELP_REGKEY L"\\" REGKEY_COMPONENTSUBKEY, FALSE, TRUE)) { DPFX(DPFPREP, 0, "Couldn't create DirectPlay NAT Helper PAST key!"); return E_FAIL; }
if (! RegObject.WriteGUID(REGKEY_VALUE_GUID, CLSID_DirectPlayNATHelpPAST)) { DPFX(DPFPREP, 0, "Couldn't write GUID to registry!"); return E_FAIL; }
if (! RegObject.WriteDWORD(REGKEY_VALUE_DIRECTPLAY8PRIORITY, DEFAULT_DIRECTPLAY8PRIORITY)) { DPFX(DPFPREP, 0, "Couldn't write DirectPlay8 priority to registry!"); return E_FAIL; }
if (! RegObject.WriteDWORD(REGKEY_VALUE_DIRECTPLAY8INITFLAGS, DEFAULT_DIRECTPLAY8INITFLAGS)) { DPFX(DPFPREP, 0, "Couldn't write DirectPlay8 init flags to registry!"); return E_FAIL; }
RegObject.Close();
return S_OK; } // DllRegisterServer
#undef DPF_MODNAME
#define DPF_MODNAME "DllUnregisterServer"
//=============================================================================
// DllUnregisterServer
//-----------------------------------------------------------------------------
//
// Description: Unregisters the DirectPlay NAT Helper PAST COM object.
//
// Arguments: None.
//
// Returns: HRESULT
// S_OK - Successfully unregistered DirectPlay NAT Helper PAST.
// E_FAIL - Failed unregistering DirectPlay NAT Helper PAST.
//=============================================================================
STDAPI DllUnregisterServer(void) { CRegistry RegObject;
//
// Unregister the class.
//
if (! CRegistry::UnRegister(&CLSID_DirectPlayNATHelpPAST)) { DPFX(DPFPREP, 0, "Failed to unregister DirectPlay NAT Helper PAST object!"); return E_FAIL; }
//
// Try removing all the subitems we registered.
//
if (! RegObject.Open(HKEY_LOCAL_MACHINE, DIRECTPLAYNATHELP_REGKEY L"\\" REGKEY_COMPONENTSUBKEY, FALSE, FALSE)) { DPFX(DPFPREP, 0, "Couldn't open DirectPlay NAT Helper key! Ignoring."); } else { if (! RegObject.DeleteValue(REGKEY_VALUE_GUID)) { DPFX(DPFPREP, 0, "Couldn't delete GUID registry value! Ignoring."); }
if (! RegObject.DeleteValue(REGKEY_VALUE_DIRECTPLAY8PRIORITY)) { DPFX(DPFPREP, 0, "Couldn't delete DirectPlay8 priority registry value! Ignoring."); }
if (! RegObject.DeleteValue(REGKEY_VALUE_DIRECTPLAY8INITFLAGS)) { DPFX(DPFPREP, 0, "Couldn't delete DirectPlay8 init flags registry value! Ignoring."); }
RegObject.Close(); }
return S_OK; } // DllUnregisterServer
#endif // !DPNBUILD_NOCOMREGISTER
#ifndef WINCE
#undef DPF_MODNAME
#define DPF_MODNAME "DirectPlayNATHelpCreate"
//=============================================================================
// DirectPlayNATHelpCreate
//-----------------------------------------------------------------------------
//
// Description: Creates an IDirectPlayNATHelp interface object.
//
// Arguments:
// GUID * pIID - Pointer to IDirectPlayNATHelp interface GUID.
// void ** ppvInterface - Place to store pointer to interface object
// created.
//
// Returns: HRESULT
// DPNH_OK - Creating the object was successful.
// DPNHERR_INVALIDPOINTER - The destination pointer is invalid.
// DPNHERR_OUTOFMEMORY - Not enough memory to create the object.
// E_NOINTERFACE - The requested interface was invalid.
//=============================================================================
HRESULT WINAPI DirectPlayNATHelpCreate(const GUID * pIID, void ** ppvInterface) { HRESULT hr;
DPFX(DPFPREP, 2, "Parameters: (0x%p, 0x%p)", pIID, ppvInterface);
hr = DoCreateInstance(NULL, // no class factory object necessary
NULL, // ?
CLSID_DirectPlayNATHelpPAST, // DirectPlayNATHelp class
(*pIID), // requested interface
ppvInterface); // place to store interface
DPFX(DPFPREP, 2, "Returning: [0x%lx]", hr);
return hr; } // DirectPlayNATHelpCreate
#endif // !WINCE
#undef DPF_MODNAME
#define DPF_MODNAME "InitializeProcessGlobals"
//=============================================================================
// InitializeProcessGlobals
//-----------------------------------------------------------------------------
//
// Description: Initialize global items needed for the DLL to operate.
//
// Arguments: None.
//
// Returns: TRUE if successful, FALSE if an error occurred.
//=============================================================================
BOOL InitializeProcessGlobals(void) { BOOL fReturn = TRUE;
if (! DNInitializeCriticalSection(&g_csGlobalsLock)) { fReturn = FALSE; }
//
// Don't allow critical section reentry.
//
DebugSetCriticalSectionRecursionCount(&g_csGlobalsLock, 0);
g_blNATHelpPASTObjs.Initialize();
//
// Seed the random number generator with the current time.
//
InitializeGlobalRand(GETTIMESTAMP());
return fReturn; } // InitializeProcessGlobals
#undef DPF_MODNAME
#define DPF_MODNAME "CleanupProcessGlobals"
//=============================================================================
// CleanupProcessGlobals
//-----------------------------------------------------------------------------
//
// Description: Releases global items used by DLL.
//
// Arguments: None.
//
// Returns: None.
//=============================================================================
void CleanupProcessGlobals(void) { CBilink * pBilink; CNATHelpPAST * pNATHelpPAST;
if (! g_blNATHelpPASTObjs.IsEmpty()) { //
// This assert is far more descriptive than hitting one of those in the
// cleanup code that complain about flags incorrectly being set.
//
DNASSERT(! "DPNHPAST.DLL unloading without all objects having been released! The caller's DirectPlayNATHelpPAST cleanup code needs to be fixed!");
//
// Force close all the objects still outstanding.
//
pBilink = g_blNATHelpPASTObjs.GetNext(); while (pBilink != &g_blNATHelpPASTObjs) { pNATHelpPAST = NATHELPPAST_FROM_BILINK(pBilink); pBilink = pBilink->GetNext();
DPFX(DPFPREP, 0, "Forcefully releasing object 0x%p!", pNATHelpPAST);
pNATHelpPAST->Close(0); // ignore error
//
// Forcefully remove it from the list and delete it instead of
// using pNATHelpPAST->Release().
//
pNATHelpPAST->m_blList.RemoveFromList(); pNATHelpPAST->UninitializeObject(); delete pNATHelpPAST; } }
DNDeleteCriticalSection(&g_csGlobalsLock); } // CleanupProcessGlobals
#undef DPF_MODNAME
#define DPF_MODNAME "ReadRegistrySettings"
//=============================================================================
// ReadRegistrySettings
//-----------------------------------------------------------------------------
//
// Description: Reads registry settings to override behavior of this DLL and to
// turn on some debugging features.
//
// Arguments: None.
//
// Returns: None.
//=============================================================================
void ReadRegistrySettings(void) { CRegistry RegObject; DWORD dwNewValue; BOOL fNewValue;
//
// Try opening the registry key.
//
if (RegObject.Open(HKEY_LOCAL_MACHINE, DIRECTPLAYNATHELP_REGKEY L"\\" REGKEY_COMPONENTSUBKEY) != FALSE) { //
// Lock out other interfaces from modifying the globals simultaneously.
//
DNEnterCriticalSection(&g_csGlobalsLock);
//
// If we successfully read a new mode, save it.
//
if (RegObject.ReadDWORD(REGKEY_VALUE_PASTICSMODE, &dwNewValue)) { g_dwPASTICSMode = dwNewValue; DPFX(DPFPREP, 1, "Using PAST ICS mode %u.", g_dwPASTICSMode); }
//
// If we successfully read a new mode, save it.
//
if (RegObject.ReadDWORD(REGKEY_VALUE_PASTPFWMODE, &dwNewValue)) { g_dwPASTPFWMode = dwNewValue; DPFX(DPFPREP, 1, "Using PAST PFW mode %u.", g_dwPASTPFWMode); }
//
// If we successfully read a new value, save it.
//
if (RegObject.ReadDWORD(REGKEY_VALUE_UNICASTTTL, &dwNewValue)) { g_iUnicastTTL = dwNewValue; if (g_iUnicastTTL != 0) { DPFX(DPFPREP, 1, "Using unicast TTL of %i.", g_iUnicastTTL); } else { DPFX(DPFPREP, 1, "Using OS default unicast TTL."); } }
//
// If we successfully read a new default gateway, save it.
//
if (RegObject.ReadDWORD(REGKEY_VALUE_DEFAULTGATEWAYV4, &dwNewValue)) { g_dwDefaultGatewayV4 = dwNewValue; DPFX(DPFPREP, 1, "Using default gateway 0x%08lx.", g_dwDefaultGatewayV4); }
//
// If we successfully read a new mask, save it.
//
if (RegObject.ReadDWORD(REGKEY_VALUE_SUBNETMASKV4, &dwNewValue)) { g_dwSubnetMaskV4 = dwNewValue; DPFX(DPFPREP, 1, "Using subnet mask 0x%08lx.", g_dwSubnetMaskV4); }
//
// If we successfully read a new interval, save it.
//
if (RegObject.ReadDWORD(REGKEY_VALUE_NOACTIVENOTIFYPOLLINTERVAL, &dwNewValue)) { g_dwNoActiveNotifyPollInterval = dwNewValue; DPFX(DPFPREP, 1, "Using no-active-notify recommended poll interval %u ms.", g_dwNoActiveNotifyPollInterval); }
//
// If we successfully read a new interval, save it.
//
if (RegObject.ReadDWORD(REGKEY_VALUE_MINUPDATESERVERSTATUSINTERVAL, &dwNewValue)) { g_dwMinUpdateServerStatusInterval = dwNewValue; DPFX(DPFPREP, 1, "Using minimum update-server-status interval %u ms.", g_dwMinUpdateServerStatusInterval); }
//
// If we successfully read a new value, save it.
//
if (RegObject.ReadDWORD(REGKEY_VALUE_POLLINTERVALBACKOFF, &dwNewValue)) { if (dwNewValue != 0) { g_dwPollIntervalBackoff = dwNewValue; DPFX(DPFPREP, 1, "Using poll interval backoff between 0 and %u ms.", g_dwPollIntervalBackoff); } else { DPFX(DPFPREP, 0, "Ignoring invalid poll interval backoff setting, using default between 0 and %u ms!", g_dwPollIntervalBackoff); } }
//
// If we successfully read a new interval, save it.
//
if (RegObject.ReadDWORD(REGKEY_VALUE_MAXPOLLINTERVAL, &dwNewValue)) { //
// Make sure the value is greater than the starting value.
//
if (dwNewValue >= g_dwNoActiveNotifyPollInterval) { g_dwMaxPollInterval = dwNewValue; DPFX(DPFPREP, 1, "Using max poll interval of %u ms.", g_dwMaxPollInterval); } else { g_dwMaxPollInterval = g_dwNoActiveNotifyPollInterval; DPFX(DPFPREP, 0, "Ignoring max poll interval of %u ms, the starting value is %u ms.", g_dwMaxPollInterval); } } else { //
// Make sure the max poll interval default value is greater than
// the starting value because we may have read in a new
// g_dwNoActiveNotifyPollInterval that makes the default
// g_dwMaxPollInterval invalid.
//
if (g_dwMaxPollInterval < g_dwNoActiveNotifyPollInterval) { g_dwMaxPollInterval = g_dwNoActiveNotifyPollInterval; DPFX(DPFPREP, 0, "Resetting max poll interval to %u ms so as to meet starting value.", g_dwMaxPollInterval); } }
//
// If we successfully read a new boolean, save it.
//
if (RegObject.ReadBOOL(REGKEY_VALUE_KEEPPOLLINGFORREMOTEGATEWAY, &fNewValue)) { g_fKeepPollingForRemoteGateway = fNewValue; if (g_fKeepPollingForRemoteGateway) { DPFX(DPFPREP, 1, "Will continue to poll for remote gateways."); } else { //
// This is actually default behavior, but print out a statement
// anyway.
//
DPFX(DPFPREP, 1, "Continually polling for remote gateways is disallowed by registry key."); } }
//
// If we successfully read a new value, save it.
//
if (RegObject.ReadDWORD(REGKEY_VALUE_REUSEPORTTIME, &dwNewValue)) { g_dwReusePortTime = dwNewValue; DPFX(DPFPREP, 1, "Reusing remote gateway discovery ports for %u ms.", g_dwReusePortTime); }
//
// If we successfully read a new value, save it.
//
if (RegObject.ReadDWORD(REGKEY_VALUE_CACHELIFEFOUND, &dwNewValue)) { g_dwCacheLifeFound = dwNewValue; DPFX(DPFPREP, 1, "Caching found addresses for %u ms.", g_dwCacheLifeFound); }
//
// If we successfully read a new value, save it.
//
if (RegObject.ReadDWORD(REGKEY_VALUE_CACHELIFENOTFOUND, &dwNewValue)) { g_dwCacheLifeNotFound = dwNewValue; DPFX(DPFPREP, 1, "Caching not-found addresses for %u ms.", g_dwCacheLifeNotFound); }
//
// Okay, we're done. Drop the lock.
//
DNLeaveCriticalSection(&g_csGlobalsLock);
//
// Done reading registry.
//
RegObject.Close(); } } // ReadRegistrySettings
#undef DPF_MODNAME
#define DPF_MODNAME "InitializeGlobalRand"
//=============================================================================
// InitializeGlobalRand
//-----------------------------------------------------------------------------
//
// Description: Initializes the fallback global psuedo-random number
// generator, using the given seed value.
//
// Arguments:
// DWORD dwSeed - Seed to use.
//
// Returns: None.
//=============================================================================
void InitializeGlobalRand(const DWORD dwSeed) { //
// We don't need to hold a lock, since this should only be done once,
// during initialization time.
//
g_dwHoldRand = dwSeed; } // InitializeGlobalRand
#undef DPF_MODNAME
#define DPF_MODNAME "GetGlobalRand"
//=============================================================================
// GetGlobalRand
//-----------------------------------------------------------------------------
//
// Description: Generates a pseudo-random DWORD.
//
// Arguments: None.
//
// Returns: Pseudo-random number.
//=============================================================================
DWORD GetGlobalRand(void) { HCRYPTPROV hCryptProv; DWORD dwResult; WORD wResult1; WORD wResult2; #ifdef DBG
DWORD dwError; #endif // DBG
if (CryptAcquireContext(&hCryptProv, NULL, NULL, PROV_RSA_FULL, CRYPT_VERIFYCONTEXT)) { if (CryptGenRandom(hCryptProv, sizeof(dwResult), (BYTE*) (&dwResult))) { CryptReleaseContext(hCryptProv, 0); return dwResult; } #ifdef DBG
else { dwError = GetLastError(); DPFX(DPFPREP, 0, "Crypto couldn't generate random number (err = %u)!", dwError); } #endif // DBG
CryptReleaseContext(hCryptProv, 0); } #ifdef DBG
else { dwError = GetLastError(); DPFX(DPFPREP, 0, "Couldn't acquire crypto provider context (err = %u)!", dwError); } #endif // DBG
//
// We couldn't use the crypto API to generate a random number, so make
// our own based off the C run time source.
//
DNEnterCriticalSection(&g_csGlobalsLock);
g_dwHoldRand = ((g_dwHoldRand * 214013L + 2531011L) >> 16) & 0x7fff; wResult1 = (WORD) g_dwHoldRand; g_dwHoldRand = ((g_dwHoldRand * 214013L + 2531011L) >> 16) & 0x7fff; wResult2 = (WORD) g_dwHoldRand;
DNLeaveCriticalSection(&g_csGlobalsLock);
return MAKELONG(wResult1, wResult2); } // GetGlobalRand
#undef DPF_MODNAME
#define DPF_MODNAME "DoCreateInstance"
//=============================================================================
// DoCreateInstance
//-----------------------------------------------------------------------------
//
// Description: Creates an instance of an interface. Required by the general
// purpose class factory functions.
//
// Arguments:
// LPCLASSFACTORY This - Pointer to class factory.
// LPUNKNOWN pUnkOuter - Pointer to unknown interface.
// REFCLSID rclsid - Reference of GUID of desired interface.
// REFIID riid - Reference to another GUID?
// LPVOID * ppvObj - Pointer to pointer to interface.
//
// Returns: HRESULT
//=============================================================================
HRESULT DoCreateInstance(LPCLASSFACTORY This, LPUNKNOWN pUnkOuter, REFCLSID rclsid, REFIID riid, LPVOID * ppvObj) { HRESULT hr; BOOL fNotCreatedWithCOM; CNATHelpPAST * pNATHelpPAST = NULL;
DNASSERT(ppvObj != NULL);
if (! IsEqualCLSID(rclsid, CLSID_DirectPlayNATHelpPAST)) { //
// This shouldn't happen if they called IClassFactory::CreateObject
// correctly.
//
DNASSERT(FALSE);
//
// Return an error.
//
hr = E_UNEXPECTED; goto Failure; }
//
// If the class factory pointer is NULL, then we were called by the
// DirectPlayNATHelpCreate function.
//
if (This == NULL) { fNotCreatedWithCOM = TRUE; } else { fNotCreatedWithCOM = FALSE; }
//
// Create the object instance.
//
pNATHelpPAST = new CNATHelpPAST(fNotCreatedWithCOM); if (pNATHelpPAST == NULL) { hr = E_OUTOFMEMORY; goto Failure; }
//
// Initialize the base object (which might fail).
//
hr = pNATHelpPAST->InitializeObject(); if (hr != S_OK) { DPFX(DPFPREP, 0, "Couldn't initialize object!"); delete pNATHelpPAST; pNATHelpPAST = NULL; goto Failure; }
//
// Add it to the global list.
//
DNEnterCriticalSection(&g_csGlobalsLock);
pNATHelpPAST->m_blList.InsertBefore(&g_blNATHelpPASTObjs); g_lOutstandingInterfaceCount++; // update count so DllCanUnloadNow works correctly
DNLeaveCriticalSection(&g_csGlobalsLock);
//
// Get the right interface for the caller and bump the refcount.
//
hr = pNATHelpPAST->QueryInterface(riid, ppvObj); if (hr != S_OK) { goto Failure; }
Exit:
//
// Release the local reference to the object. If this function was
// successful, there's still a reference in ppvObj.
//
if (pNATHelpPAST != NULL) { pNATHelpPAST->Release(); pNATHelpPAST = NULL; }
return hr;
Failure:
//
// Make sure we don't hand back a pointer.
//
(*ppvObj) = NULL;
goto Exit; } // DoCreateInstance
#undef DPF_MODNAME
#define DPF_MODNAME "IsClassImplemented"
//=============================================================================
// IsClassImplemented
//-----------------------------------------------------------------------------
//
// Description: Determine if a class is implemented in this DLL. Required by
// the general purpose class factory functions.
//
// Arguments:
// REFCLSID rclsid - Reference to class GUID.
//
// Returns: BOOL
// TRUE - This DLL implements the class.
// FALSE - This DLL doesn't implement the class.
//=============================================================================
BOOL IsClassImplemented(REFCLSID rclsid) { return (IsEqualCLSID(rclsid, CLSID_DirectPlayNATHelpPAST)); } // IsClassImplemented
|