|
|
//+---------------------------------------------------------------------------
//
// Microsoft Windows
// Copyright (C) Microsoft Corporation, 1992 - 1995.
//
// File: libmain.cxx
//
// Contents: LibMain for nds.dll
//
// Functions: LibMain, DllGetClassObject
//
// History: 25-Oct-94 KrishnaG Created.
//
//----------------------------------------------------------------------------
#include "ldap.hxx"
#pragma hdrstop
BOOL fInitializeCritSect = FALSE; HINSTANCE g_hInst = NULL; extern HMODULE g_hActiveDs;
typedef DWORD (*PF_DllGetClassObject) ( REFCLSID clsid, REFIID iid, LPVOID FAR* ppverved );
//---------------------------------------------------------------------------
// ADs debug print, mem leak and object tracking-related stuff
//---------------------------------------------------------------------------
DECLARE_INFOLEVEL(ADs)
//+---------------------------------------------------------------------------
//
// Function: ShutDown
//
// Synopsis: Function to handle printing out heap debugging display
//
//----------------------------------------------------------------------------
inline VOID ShutDown() { #if DBG==1
#ifndef MSVC
DUMP_TRACKING_INFO_DELETE(); DeleteCriticalSection(&g_csOT); #endif // ifndef MSVC
DeleteCriticalSection(&g_csDP); #endif
}
extern "C" DWORD heapInfoLevel; extern "C" DWORD OtInfoLevel; extern "C" DWORD ADsInfoLevel;
extern CRITICAL_SECTION g_RootDSECritSect;
extern CRITICAL_SECTION g_ExtCritSect;
extern CRITICAL_SECTION g_TypeInfoCritSect;
extern CRITICAL_SECTION g_DispTypeInfoCritSect;
extern CRITICAL_SECTION g_SystemAPICritSect;
CRITICAL_SECTION g_LockCritSect;
//+---------------------------------------------------------------------------
//
// Function: GetINIHeapInfoLevel
//
// Synopsis: Gets various infolevel values from win.ini
//
//----------------------------------------------------------------------------
inline VOID GetINIHeapInfoLevel() { #if DBG==1
const INT MAXINFOLEN=11; TCHAR awcs[MAXINFOLEN];
#ifndef MSVC
if (GetProfileString(TEXT("LDAP"),TEXT("heapInfoLevel"), TEXT("00000003"), awcs,MAXINFOLEN)) heapInfoLevel = _tcstoul(awcs, NULL, 16);
if (GetProfileString(TEXT("LDAP"),TEXT("Ot"), TEXT("00000003"), awcs, MAXINFOLEN)) OtInfoLevel = _tcstoul(awcs, NULL, 16);
#endif // MSVC
if (GetProfileString(TEXT("LDAP"),TEXT("ADsInfoLevel"), TEXT("00000003"), awcs,MAXINFOLEN)) ADsInfoLevel = _tcstoul(awcs, NULL, 16); #endif
}
// Globals
ULONG g_ulObjCount = 0; // Number of objects alive in oleds.dll
ULONG g_ulLocks = 0; // Number of Provider objects alive
CLDAPProviderCF g_cfProvider; CLDAPNamespaceCF g_cfNamespace; CADSystemInfoCF g_cfADSystemInfo; CNameTranslateCF g_cfNameTranslate;
//+------------------------------------------------------------------------
//
// oleds class factories
//
//-------------------------------------------------------------------------
struct CLSCACHE { const CLSID * pclsid; IClassFactory * pCF; };
CLSCACHE g_aclscache[] = { &CLSID_LDAPProvider, &g_cfProvider, &CLSID_LDAPNamespace, &g_cfNamespace, &CLSID_NameTranslate, &g_cfNameTranslate, &CLSID_ADSystemInfo, &g_cfADSystemInfo };
extern PCLASS_ENTRY gpClassHead;
//+---------------------------------------------------------------
//
// Function: DllGetClassObject
//
// Synopsis: Standard DLL entrypoint for locating class factories
//
//----------------------------------------------------------------
STDAPI DllGetClassObject(REFCLSID clsid, REFIID iid, LPVOID FAR* ppv) { HRESULT hr = E_NOINTERFACE; size_t i; HKEY hKey = NULL; HINSTANCE hDll = NULL ;
if (ppv) *ppv = NULL;
for (i = 0; i < ARRAY_SIZE(g_aclscache); i++) { if (IsEqualCLSID(clsid, *g_aclscache[i].pclsid)) { hr = g_aclscache[i].pCF->QueryInterface(iid, ppv); RRETURN(hr); } }
//
// This workaround is for the special case where an old version of ADSI
// is installed in the system. Installing that will overwrite the registry
// with the old setting for pathcracker. The pathcracker object used to live
// on adsldp.
// The following code redirects the call to the DllGetClassObject in
// activeds if the Pathname object is being requested. It also fixes the
// registry to point to the correct DLL.
//
if (IsEqualCLSID(clsid, CLSID_Pathname)) { PF_DllGetClassObject pfDllGetClassObject= NULL ; WCHAR szPathDescriptor[] = L"ADs Pathname Object"; WCHAR szDllName[] = L"activeds.dll"; DWORD WinError;
if (!(hDll = LoadLibraryHelper(szDllName))) { BAIL_ON_FAILURE(hr = HRESULT_FROM_WIN32(GetLastError())); }
if (!(pfDllGetClassObject = (PF_DllGetClassObject)GetProcAddress(hDll, "DllGetClassObject"))) { BAIL_ON_FAILURE(hr = HRESULT_FROM_WIN32(GetLastError())); }
hr = (*pfDllGetClassObject)(clsid, iid, ppv); BAIL_ON_FAILURE(hr);
//
// Setting the general description
// Even if any of the operations below fails, we'll just bail with the
// hr from DllGetClassObject.
//
WinError = RegOpenKeyEx(HKEY_CLASSES_ROOT, L"CLSID\\{080d0d78-f421-11d0-a36e-00c04fb950dc}", NULL, KEY_ALL_ACCESS, &hKey); if (WinError != ERROR_SUCCESS) { goto error; }
WinError = RegSetValueEx(hKey, NULL, 0, REG_SZ, (BYTE *)szPathDescriptor, (wcslen(szPathDescriptor)+1) * sizeof(WCHAR)); if (WinError != ERROR_SUCCESS) { goto error; }
RegCloseKey(hKey); hKey = NULL;
//
// Setting the inprocserver
//
WinError = RegOpenKeyEx(HKEY_CLASSES_ROOT, L"CLSID\\{080d0d78-f421-11d0-a36e-00c04fb950dc}\\InprocServer32", NULL, KEY_ALL_ACCESS, &hKey); if (WinError != ERROR_SUCCESS) { goto error; }
WinError = RegSetValueEx(hKey, NULL, 0, REG_SZ, (BYTE *)szDllName, (wcslen(szDllName)+1) * sizeof(WCHAR)); if (WinError != ERROR_SUCCESS) { goto error; } }
//
// Add Debugging Code to indicate that the oleds.DllGetClassObject has been called with an unknown CLSID.
//
error: if (hDll) { FreeLibrary(hDll); } if (hKey) { RegCloseKey(hKey); } return hr; }
//+---------------------------------------------------------------
//
// Function: DllCanUnloadNow
//
// Synopsis: Standard DLL entrypoint to determine if DLL can be unloaded
//
//---------------------------------------------------------------
STDAPI DllCanUnloadNow(void) { HRESULT hr;
hr = S_FALSE;
//
// Both the ldap and utils\cdispmgr count need to be 0
//
if (AggregatorDllCanUnload() && DllReadyToUnload()) {
EnterCriticalSection(&g_LockCritSect); if(0 == g_ulLocks) { hr = S_OK; } LeaveCriticalSection(&g_LockCritSect);
}
return hr; }
//+---------------------------------------------------------------
//
// Function: LibMain
//
// Synopsis: Standard DLL initialization entrypoint
//
//---------------------------------------------------------------
EXTERN_C BOOL __cdecl LibMain(HINSTANCE hInst, ULONG ulReason, LPVOID pvReserved) { HRESULT hr; DWORD dwCritSectIniStage = 0;
switch (ulReason) { case DLL_PROCESS_ATTACH: //
// In try to catch possibily of init crit sects failing.
//
__try {
DisableThreadLibraryCalls(hInst);
g_hInst = hInst;
g_hActiveDs = GetModuleHandle(TEXT("activeds.dll"));
// Maybe we should check the handle.
InitializeCriticalSection(&g_RootDSECritSect); dwCritSectIniStage = 1; InitializeCriticalSection(&g_ExtCritSect); dwCritSectIniStage = 2; InitializeCriticalSection(&g_TypeInfoCritSect); dwCritSectIniStage = 3; InitializeCriticalSection(&g_DispTypeInfoCritSect); dwCritSectIniStage = 4; InitializeCriticalSection(&g_csLoadLibsCritSect); dwCritSectIniStage = 5; InitializeCriticalSection(&g_LockCritSect); dwCritSectIniStage = 6;
InitializeCriticalSection(&g_SystemAPICritSect); dwCritSectIniStage = 7;
#if DBG==1
InitializeCriticalSection(&g_csDP); dwCritSectIniStage = 8; #ifndef MSVC
InitializeCriticalSection(&g_csOT); dwCritSectIniStage = 9; InitializeCriticalSection(&g_csMem); dwCritSectIniStage = 10; #endif
#endif
fInitializeCritSect = TRUE; } __except (EXCEPTION_EXECUTE_HANDLER) {
//
// Something went wrong
//
switch(dwCritSectIniStage) { #if DBG==1
#ifndef MSVC
case 10: DeleteCriticalSection(&g_csMem); case 9: DeleteCriticalSection(&g_csOT); #endif
case 8: DeleteCriticalSection(&g_csDP); #endif
case 7: DeleteCriticalSection(&g_SystemAPICritSect); case 6: DeleteCriticalSection(&g_LockCritSect); case 5: DeleteCriticalSection(&g_csLoadLibsCritSect); case 4: DeleteCriticalSection(&g_DispTypeInfoCritSect); case 3: DeleteCriticalSection(&g_TypeInfoCritSect); case 2: DeleteCriticalSection(&g_ExtCritSect); case 1: DeleteCriticalSection(&g_RootDSECritSect);
} return FALSE; }
break;
case DLL_PROCESS_DETACH:
//
// free global list of class entries for 3rd party ext
//
if (gpClassHead) { FreeClassesList(gpClassHead); }
if (gpszStickyServerName) { FreeADsStr(gpszStickyServerName); gpszStickyServerName = NULL; }
if (gpszStickyDomainName) { FreeADsStr(gpszStickyDomainName); gpszStickyDomainName = NULL; }
//
// Good idea to delete all the critical sections
//
if(fInitializeCritSect) { FreeServerType(); #if DBG==1
#ifndef MSVC
DeleteCriticalSection(&g_csOT); DeleteCriticalSection(&g_csMem); #endif
DeleteCriticalSection(&g_csDP); #endif
DeleteCriticalSection(&g_RootDSECritSect); DeleteCriticalSection(&g_ExtCritSect); DeleteCriticalSection(&g_TypeInfoCritSect); DeleteCriticalSection(&g_DispTypeInfoCritSect); DeleteCriticalSection(&g_csLoadLibsCritSect); DeleteCriticalSection(&g_LockCritSect); DeleteCriticalSection(&g_SystemAPICritSect);
}
//
// Should be ok to free the dynamically loaded libs.
//
if (g_hDllNtdsapi) { FreeLibrary((HMODULE) g_hDllNtdsapi); g_hDllNtdsapi = NULL; }
if (g_hDllSecur32) { FreeLibrary((HMODULE) g_hDllSecur32); }
break;
default: break; }
return TRUE; }
//+---------------------------------------------------------------------------
//
// Function: DllMain
//
// Synopsis: entry point for NT - post .546
//
//----------------------------------------------------------------------------
BOOL DllMain(HANDLE hDll, DWORD dwReason, LPVOID lpReserved) { return LibMain((HINSTANCE)hDll, dwReason, lpReserved); }
|