windows-nt/Source/XPSP1/NT/net/winnet/mprinit.cxx

1756 lines
48 KiB
C++
Raw Normal View History

2020-09-26 03:20:57 -05:00
/*++
Copyright (c) 1991 Microsoft Corporation
Module Name:
MPRINIT.CXX
Abstract:
Contains the DLL initialization routine for the Mutiple-Provider
Router. The following are names of functions in this file:
MprDllInit
MprCheckProviders
MprLevel1Init
MprLevel2Init
MprProcessDetach
GetProviderCapabilities
MprFreeGlobalProviderInfo
MprMakeServiceKeyName
MprGetOrderedList
MprGetNumProviders
MprExtractProviderInfo
The router distinguishes between Providers and ActiveProviders. It
maintains global information about GlobalNumProviders and
GlobalNumActiveProviders. The two counts can be distinguished in the
following ways:
GlobalNumProviders - This determines the size of the provider database
that mpr maintains. This count reflects the number of providers for
which we were able to obtain registry information. A provider is
added to the list if we are able to obtain its name and the
name of its dll routine.
GlobalNumActiveProviders - A provider becomes active when we are able
to successfully load the dll, and obtain at least one entry point
for it.
Author:
Dan Lafferty (danl) 6-Oct-1991
Environment:
User Mode - Win32
Notes:
Revision History:
05-May-1999 jschwart
Make provider addition/removal dynamic
09-Aug-1996 AnirudhS
Disabled thread library calls, since they cause extra paging and
are 100% useless. Fixed memory leak of error records on DLL unload.
02-Mar-1995 AnirudhS
Add test for WNNC_ENUM_CONTEXT capability.
09-Aug-1994 Danl
Only do PROCESS_DETACH cleanup work if the PROCESS_DETACH occurs
because of a FreeLibrary.
01-Mar-1994 Danl
Created a seperate location for the Credential Managers GetCaps()
function. This way if a provider has both a credential manager
dll and a network dll, we will be able to direct calls to the
correct GetCaps() function.
08-Apr-1993 Danl
MprLevel1Init: Created a tempOrderedString which will be used
for incrementing through the buffer. This way we can Free
orderedString because it still points to the beginning of the buffer.
24-Nov-1992 Danl
Modified many things to allow for initialization to occur on
first call rather than at DLL initialization time. Also coded
for the future possibility of Authentication Providers.
02-Nov-1992 Danl
Allow MPR to continue operation even if there are no providers, or
if the network is not installed and provider info is not available
in the registry. Removed MprHack.
31-Aug-1992 Danl
Removed use of the list of active providers. Now all providers
in the Provider Order list are loaded at init time.
06-Oct-1991 Danl
Created
--*/
//
// Includes
//
#include "precomp.hxx"
#include <tstring.h> // MEMCPY
#include <debugfmt.h> // FORMAT_LPTSTR
#include "mprres.h" // IDS_NETROOT
#include "connify.h" // MprConnectNotifyInit()
//
// Local Function Prototypes
//
BOOL
MprProcessAttach(
HANDLE DllHandle
);
BOOL
MprProcessDetach(
DWORD dwInitLevel
);
BOOL
GetProviderCapabilities(
LPPROVIDER Provider
);
VOID
MprFreeGlobalProviderInfo(VOID);
BOOL
MprGetOrderedList(
HKEY ControlRootKey,
LPTSTR *OrderString
);
DWORD
MprMakeServiceKeyName(
IN OUT LPTSTR *OrderedNamePtr,
OUT LPTSTR *NameBufferPtr,
OUT LPTSTR *NamePtr
);
DWORD
MprGetNumProviders(
LPTSTR OrderedString
);
BOOL
MprExtractProviderInfo(
HKEY ProviderInfoKey,
LPPROVIDER Provider
);
BOOL
MprInitializeProviderGlobals(
VOID
);
BOOL
MprGetLUIDDeviceMapsEnabled(
PBOOL pResult
);
//
// Constants
//
#define TEMP_SIZE 256
//
// Some registry path names
//
#define PROVIDER_KEY_NAME TEXT("control\\NetworkProvider\\active")
#define PROVIDER_ORDER_KEY TEXT("control\\NetworkProvider\\HwOrder")
#define SERVICES_KEY_NAME TEXT("services\\")
#define SYSTEM_CONTROL_ROOT TEXT("system\\CurrentControlSet")
#define PROVIDER_PATH TEXT("\\NetworkProvider")
#define VALUE_PATHNAME TEXT("ProviderPath")
#define AUTHENT_PATHNAME TEXT("AuthentProviderPath")
#define VALUE_CLASS TEXT("Class")
//
// Defines for the DLL's init state
//
#define MPR_PROVIDERLOCK_CREATED 0x00000001
#define MPR_INITCRITSEC_CREATED 0x00000002
#define MPR_ERRORCRITSEC_CREATED 0x00000004
#define MPR_CACHE_CONSTRUCTED 0x00000008
//
// Global Data Structures
//
LPPROVIDER GlobalProviderInfo=NULL; // A pointer to array of PROVIDER Structures
DWORD GlobalNumProviders=0;
DWORD GlobalNumActiveProviders=0;
CRITICAL_SECTION MprInitCritSec;
HMODULE hDLL;
HANDLE g_hProvidersChanged; // Event set on provider addition/deletion
HKEY g_hProviderKey; // Registry key of the provider list
WCHAR g_wszEntireNetwork[40]; // "Entire Network" string for top-level enum
BOOL g_LUIDDeviceMapsEnabled=FALSE; // LUID device map enabled/disabled
//
// GlobalInitLevel is a set of bit flags used to indicate how much
// initialization work has been done.
//
volatile DWORD GlobalInitLevel = 0;
DWORD MprDebugLevel;
extern "C"
BOOL
MprDllInit(
IN HINSTANCE DllHandle,
IN DWORD Reason,
IN PCONTEXT pContext OPTIONAL
)
/*++
Routine Description:
This routine reads the registry to determine what network providers are
available. It then loads and initializes the Network Providers. After
this, a call is made to each of the providers to get a list of
Capabilities (via WNetGetCaps).
Arguments:
DllHandle - A handle for the DLL?
Reason - The reason for which this routine is being called. This might
be one of the following:
DLL_PROCESS_ATTACH
DLL_THREAD_ATTACH
DLL_THREAD_DETACH
DLL_PROCESS_DETACH
pContext - Pointer to a context structure
Return Value:
Note:
--*/
{
//
// We need to keep track of how far initialization went since
// we may get called with DLL_PROCESS_DETACH if init fails and
// don't want to clean up uninitialized data
//
static DWORD s_dwInitLevel;
if (Reason == DLL_PROCESS_ATTACH) {
#if DBG == 1
//
// Read the debug message level from the registry
//
HKEY MprKey;
if (MprOpenKey(HKEY_LOCAL_MACHINE,
L"Software\\Microsoft\\Windows NT\\CurrentVersion\\Mpr",
&MprKey,
0))
{
MprDebugLevel = (DWORD) MprGetKeyNumberValue(MprKey, L"DebugLevel", 0);
RegCloseKey(MprKey);
}
#endif
//
// create semaphore used to protect global data (DLL handles and
// function pointers.
//
if ((MprLoadLibSemaphore = CreateSemaphore( NULL,1,1,NULL )) == NULL)
{
MPR_LOG1(ERROR,
"MprDllInit: CreateSemaphore FAILED %d\n",
GetLastError());
return FALSE ;
}
if (!MprInitializeProviderGlobals())
{
return FALSE;
}
s_dwInitLevel |= MPR_PROVIDERLOCK_CREATED;
MPR_LOG2(PS,"Process %#lx (%ws) attached to the DLL\n",
GetCurrentProcessId(), GetCommandLineW());
//
// Save the DLL handle, to be used in LoadString
//
hDLL = DllHandle;
//
// Create the critical section to synchronize access to global
// data when doing init on first call.
//
InitializeCriticalSection(&MprInitCritSec);
s_dwInitLevel |= MPR_INITCRITSEC_CREATED;
//
// Initialize critical section that protects the Linked list of
// Error Records.
//
InitializeCriticalSection(&MprErrorRecCritSec);
s_dwInitLevel |= MPR_ERRORCRITSEC_CREATED;
CRoutedOperation::ConstructCache();
s_dwInitLevel |= MPR_CACHE_CONSTRUCTED;
DisableThreadLibraryCalls(DllHandle);
}
else if (Reason == DLL_PROCESS_DETACH) {
//
// Only do cleanup if detach was due to a FreeLibrary() call.
// In this case pContext will be NULL. Otherwise, this gets called
// because the process is terminating. We will let the process
// cleanup code clean up everything.
//
if (pContext == NULL) {
//
// close handle for semaphore
//
CloseHandle(MprLoadLibSemaphore) ;
MprLoadLibSemaphore = NULL ;
MPR_LOG(PS,"Process %#lx detached from the DLL\n",GetCurrentProcessId());
MPR_LOG(TRACE,"******************* CLEAN-UP ********************\n",0);
MprProcessDetach(s_dwInitLevel);
if (s_dwInitLevel & MPR_CACHE_CONSTRUCTED)
{
CRoutedOperation::DestroyCache();
}
MPR_LOG(TRACE,"***************** CLEAN_UP END ******************\n",0);
}
}
else if (Reason == DLL_THREAD_ATTACH) {
// This should never happen
MPR_LOG2(PS,"Thread %#lx.%#lx attached to the DLL\n",
GetCurrentProcessId(), GetCurrentThreadId());
}
else if (Reason == DLL_THREAD_DETACH) {
// This should never happen
MPR_LOG2(PS,"Thread %#lx.%#lx detached from the DLL\n",
GetCurrentProcessId(), GetCurrentThreadId());
}
return(TRUE);
}
BOOL
MprInitializeProviderGlobals(
VOID
)
{
g_hProvidersChanged = CreateEvent(NULL, // Default security
TRUE, // Manual reset
FALSE, // Starts out unsignaled
NULL); // No name
if (g_hProvidersChanged == NULL)
{
MPR_LOG1(ERROR,
"MprInitializeProviderGlobals: CreateEvent FAILED %d\n",
GetLastError());
return FALSE;
}
//
// NOTE: Key handle must be global since RegNotifyChangeKeyValue
// fires when the handle is closed (done in MprProcessDetach)
//
if (!MprOpenKey(HKEY_LOCAL_MACHINE,
SYSTEM_CONTROL_ROOT L"\\" PROVIDER_ORDER_KEY,
&g_hProviderKey,
0))
{
MPR_LOG1(ERROR,
"MprInitializeProviderGlobals: MprOpenKey FAILED %d\n",
GetLastError());
return FALSE;
}
if (RegNotifyChangeKeyValue(g_hProviderKey, // Key
FALSE, // Don't watch subtree
REG_NOTIFY_CHANGE_LAST_SET, // Watch for value changes
g_hProvidersChanged, // Event to signal
TRUE) // Asynchronous
!= ERROR_SUCCESS)
{
//
// If this call fails, we won't notice provider additions/deletions
//
MPR_LOG1(ERROR,
"MprInitializeProviderGlobals: RegNotifyChangeKeyValue FAILED %d\n",
GetLastError());
}
//
// Check if LUID device maps are Enabled
//
MprGetLUIDDeviceMapsEnabled( &g_LUIDDeviceMapsEnabled );
//
// Create the lock that restricts access to the provider array
//
MPRProviderLock.Initialize("P", "Provider");
return TRUE;
}
VOID
MprCheckProviders(
VOID
)
{
ASSERT(!MPRProviderLock.Have());
if (WaitForSingleObject(g_hProvidersChanged, 0) == WAIT_OBJECT_0)
{
//
// The providers have changed -- grab the exclusive lock and
// check again to avoid a race condition. Note that there
// is no need to acquire the MprInitCritsec since an API must
// always acquire the critsec (via INIT_IF_NECESSARY) AFTER it
// acquires the shared provider lock.
//
CProviderExclusiveLock PLock;
if (WaitForSingleObject(g_hProvidersChanged, 0) == WAIT_OBJECT_0)
{
//
// Still set -- this is the first thread here. Free up the provider
// info and reset the init level so the next INIT_IF_NECESSARY call
// will reinitialize the providers as necessary.
//
MPR_LOG0(TRACE,
"MprCheckProviders: Provider list changed -- reinitializing\n");
MprDeleteIndexArray();
MprFreeGlobalProviderInfo();
GlobalInitLevel = 0;
//
// Flush the provider cache
//
CRoutedOperation::DestroyCache();
CRoutedOperation::ConstructCache();
ResetEvent(g_hProvidersChanged);
if (RegNotifyChangeKeyValue(g_hProviderKey, // Key
FALSE, // Don't watch subtree
REG_NOTIFY_CHANGE_LAST_SET, // Watch for value changes
g_hProvidersChanged, // Event to signal
TRUE) // Asynchronous
!= ERROR_SUCCESS)
{
//
// If this call fails we won't notice provider additions/deletions
//
MPR_LOG1(ERROR,
"MprCheckProviders: RegNotifyChangeKeyValue FAILED %d\n",
GetLastError());
}
}
}
}
DWORD
MprLevel1Init(
VOID
)
/*++
Routine Description:
This function looks into the registry to find the names of the providers
and their associated DLLs. A Global table of provider information is
then allocated (GlobalProviderInfo). This table is an array of provider
information structures. The number of elements in the array is stored
in GlobalNumProviders.
If the provider is in the registry ORDERED list, then it is assumed that
complete provider information is stored in the "services" section of
the registry for that provider. This provider information includes such
things as the pathname for the provider dll, and the provider type.
The information is stored in the provider structure for use by the
MprLevel2Init routine.
Arguments:
Return Value:
Note:
--*/
{
DWORD status;
LPPROVIDER provider;
DWORD i;
HKEY controlRootKey;
HKEY providerInfoKey;
LPTSTR orderedString = NULL;
LPTSTR tempOrderedString;
LPTSTR nameBuffer = NULL; // system\LanmanRedirector\NetworkProvider
LPTSTR providerIdString = NULL; // points to provider id in nameBuffer
BOOL oneInitialized = FALSE;
EnterCriticalSection(&MprInitCritSec);
MPR_LOG(TRACE,"******************* LEVEL 1 INIT ********************\n",0);
//
// If this level of initialization is already complete, then return.
//
if (GlobalInitLevel & FIRST_LEVEL) {
LeaveCriticalSection(&MprInitCritSec);
return(WN_SUCCESS);
}
//
// This function had better be called first and only once
//
ASSERT(GlobalInitLevel == 0);
//
// Get a handle to the "current" services part of the registry
//
if(!MprOpenKey(
HKEY_LOCAL_MACHINE, // hKey
SYSTEM_CONTROL_ROOT, // lpSubKey
&controlRootKey, // Newly Opened Key Handle
DA_READ)) { // Desired Access
MPR_LOG(ERROR,"MprLevel1Init: MprOpenKey (System) Error\n",0);
LeaveCriticalSection(&MprInitCritSec);
return(WN_NO_NETWORK);
}
MPR_LOG2(TRACE,"OpenKey %ws,0x%lx\n ",SYSTEM_CONTROL_ROOT,controlRootKey);
//
// Obtain the ordered list information.
// (the orderedString buffer is allocated here)
//
// If this fails, we assume that there are no providers.
//
if (!MprGetOrderedList(controlRootKey, &orderedString)) {
MPR_LOG(ERROR,"Could not get the ordered list of providers\n",0);
RegCloseKey(controlRootKey);
LeaveCriticalSection(&MprInitCritSec);
return(WN_NO_NETWORK);
}
MPR_LOG1(TRACE,"ProviderOrderString = %ws\n",orderedString);
GlobalNumProviders = MprGetNumProviders(orderedString);
//
// Allocate the database in which to store Provider Information.
// This is to be an array of PROVIDER structures. Memory is set
// to zero so that all the provider entry points will be initialized to
// zero.
//
GlobalProviderInfo = (LPPROVIDER) LocalAlloc(
LPTR,
sizeof(PROVIDER) * GlobalNumProviders);
if (GlobalProviderInfo == NULL) {
status = GetLastError();
MPR_LOG(ERROR,"MprLevel1Init: provider array LocalAlloc Failed %d\n",
status);
LocalFree(orderedString);
GlobalNumProviders = 0;
RegCloseKey(controlRootKey);
LeaveCriticalSection(&MprInitCritSec);
return(status);
}
provider = GlobalProviderInfo;
tempOrderedString = orderedString;
for(i=0; i<GlobalNumProviders; i++,provider++) {
//
// Build the Service key name for the next Service/Driver.
// NOTE: This function allocates a nameBuffer.
//
status = MprMakeServiceKeyName(
&tempOrderedString,
&nameBuffer,
&providerIdString );
if (status != WN_SUCCESS) {
goto SkipProvider;
}
//
// Create the path name to the provider information in the
// service's tree.
//
STRCAT(nameBuffer, PROVIDER_PATH);
//
// Open the provider portion of that service/driver's
// node in the service tree.
//
if (!MprOpenKey( controlRootKey, nameBuffer, &providerInfoKey, DA_READ)){
LocalFree(nameBuffer);
goto SkipProvider;
}
MPR_LOG1(TRACE,"\n\t----%ws----\n",providerIdString);
MPR_LOG3(TRACE,"\tOpenKey %ws,\n\t%ws\n,\t0x%lx\n",
SYSTEM_CONTROL_ROOT,
nameBuffer,
providerInfoKey);
//
// Free the memory that was allocated for the name buffer.
//
LocalFree (nameBuffer);
//
// Get the data for the provider from the registry at the
// location referenced by providerInfoKey.
//
if (!MprExtractProviderInfo(providerInfoKey, provider)) {
MPR_LOG(TRACE,"CloseKey 0x%lx\n", providerInfoKey);
RegCloseKey(providerInfoKey);
goto SkipProvider;
}
//
// Close the ProviderInfoKey.
//
MPR_LOG(TRACE,"CloseKey 0x%lx\n", providerInfoKey);
RegCloseKey(providerInfoKey);
//
// We have information for at least one provider.
//
oneInitialized = TRUE;
SkipProvider:
; // The complier needs to find something to go to.
} // End For NumProviders
MPR_LOG(TRACE,"CloseKey 0x%lx\n", controlRootKey);
RegCloseKey(controlRootKey);
if (orderedString != NULL) {
LocalFree(orderedString);
}
GlobalInitLevel = FIRST_LEVEL;
LeaveCriticalSection(&MprInitCritSec);
if (oneInitialized == FALSE) {
return(WN_NO_NETWORK);
}
return(WN_SUCCESS);
}
DWORD
MprLevel2Init(
DWORD InitLevel
)
/*++
Routine Description:
This routine initializes all providers of the class described by
the InitLevel parameter. We loop though all providers that meet
this description, and load the dll and find the dll's function entry
points. If we were successful in getting this information for the
provider, it will be added to our "active" list by having its provider
index placed in the GlobalIndexArray (via MprInitIndexArray).
NOTE:
This routine can potentially be called once for each type of
initialization that is supported. For instance, it may be called for
authentication initialization, and then called later for network
initialization.
Each time this routine is called, a new indexArray is created.
After this array is filled in, it is merged with any existing
indexArray left over from previous calls. The merge is performed
by MprInitIndexArray. Locks on this array must be held in order to
update it.
Arguments:
InitLevel - This indicates the level of initialization. This can be
a NETWORK_LEVEL or CREDENTIAL_LEVEL initialization, or both.
Return Value:
WN_SUCCESS - This is returned if we are able to obtain some entry point
information for at least one provider.
Otherwise, an appropriate error is returned.
--*/
{
DWORD status = WN_SUCCESS;
LPPROVIDER provider;
DWORD i;
LPDWORD indexArray = NULL;
DWORD numActive = 0;
DWORD InitClass = 0;
//
// Before we can do a level2 initialization, we must first check to
// see if level 1 has been completed.
//
if (!(GlobalInitLevel & FIRST_LEVEL)) {
status = MprLevel1Init();
if (status != WN_SUCCESS) {
return(status);
}
}
EnterCriticalSection(&MprInitCritSec);
MPR_LOG(TRACE,"******************* LEVEL 2 INIT ********************\n",0);
//
// If this level of initialization is already complete, then return.
//
if (GlobalInitLevel & InitLevel) {
LeaveCriticalSection(&MprInitCritSec);
return(WN_SUCCESS);
}
//
// Translate InitLevel into an InitClass so it can be compared with the
// provider class.
//
if (InitLevel & NETWORK_LEVEL)
{
InitClass |= NETWORK_TYPE;
//
// Load the "Entire Network" string (used by WNetEnumResource
// for RESOURCE_CONNECTED). Note that WNetEnumResource requires
// only NETWORK_LEVEL initialization, so it's OK to load it
// here and only here.
//
if (g_wszEntireNetwork[0] == L'\0')
{
int cch = LoadString(hDLL,
IDS_NETROOT,
g_wszEntireNetwork,
LENGTH(g_wszEntireNetwork));
ASSERT(cch > 0 && cch < LENGTH(g_wszEntireNetwork));
}
}
if (InitLevel & CREDENTIAL_LEVEL)
{
InitClass |= CREDENTIAL_TYPE;
}
if (InitLevel & NOTIFIEE_LEVEL)
{
status = MprConnectNotifyInit();
GlobalInitLevel |= InitLevel;
LeaveCriticalSection(&MprInitCritSec);
return(status);
}
provider = GlobalProviderInfo;
//
// Allocate storage for the ordered list of indices. This storage is
// freed by MprInitIndexArray.
//
indexArray = (LPDWORD) LocalAlloc(LPTR, sizeof(DWORD) * GlobalNumProviders);
if (indexArray == NULL) {
MPR_LOG(ERROR,"MprProcessAttach: indexArray LocalAlloc Failed %d\n",
GetLastError());
LeaveCriticalSection(&MprInitCritSec);
return(GetLastError());
}
for(i=0; i<GlobalNumProviders; i++,provider++) {
//
// If this provider matches the init type for which we are
// initializing, then load the library and get the entry points.
// Then add the provider array index to the index array.
//
if (provider->InitClass & InitClass) {
//
// Load the DLL and free the memory for its name.
//
if (provider->AuthentDllName != NULL) {
MPR_LOG1(TRACE,"MprLevel2Init: Loading %ws\n",provider->AuthentDllName);
provider->AuthentHandle = LoadLibraryEx(
provider->AuthentDllName,
NULL,
LOAD_WITH_ALTERED_SEARCH_PATH);
if (provider->AuthentHandle == NULL) {
MPR_LOG(ERROR,"MprLevel2Init: LoadLibraryEx Failed %d\n",
GetLastError());
}
LocalFree(provider->AuthentDllName);
provider->AuthentDllName = NULL;
}
if (provider->DllName != NULL) {
MPR_LOG1(TRACE,"MprLevel2Init: Loading %ws\n",provider->DllName);
provider->Handle = LoadLibraryEx(
provider->DllName,
NULL,
LOAD_WITH_ALTERED_SEARCH_PATH);
if (provider->Handle == NULL) {
MPR_LOG(ERROR,"MprLevel2Init: LoadLibraryEx Failed %d\n",
GetLastError());
}
//
// Keep around the DLL name since we need to do another
// LoadLibraryEx to refcount the DLL in WNetOpenEnum
//
}
if ((provider->Handle != NULL) || (provider->AuthentHandle != NULL)) {
MPR_LOG0(TRACE,"MprLevel2Init: LoadLibraryEx success\n");
//
// Obtain the various entry points for the library.
// This is done based on the capabilities that are listed for the
// provider.
//
if (GetProviderCapabilities(provider)) {
//
// Only providers for which we are able to successfully
// get the entry points are added to the active array.
//
// Since this provider information is now initialized, the
// provider is considered ACTIVE. Put the index to the
// provider in the index array.
//
MPR_LOG1(TRACE,"MprLevel2Init: Successfully got "
"capabilities for %ws\n",provider->Resource.lpProvider);
indexArray[numActive] = i;
numActive++;
}
}
} // End If InitLevel match
} // End For NumProviders
//
// Store the information in this indexArray in the GlobalIndexArray.
//
MprInitIndexArray(indexArray, numActive);
GlobalInitLevel |= InitLevel;
LeaveCriticalSection(&MprInitCritSec);
MPR_LOG(TRACE,"******************* END LEVEL 2 INIT ********************\n",0);
return(WN_SUCCESS);
}
BOOL
MprProcessDetach(
DWORD dwInitLevel
)
/*++
Routine Description:
This function cleans up resources for a process when it detaches
from the dll.
Arguments:
none
Return Value:
none
Note:
--*/
{
MprDeleteIndexArray();
MprFreeGlobalProviderInfo();
GlobalNumProviders = 0;
GlobalProviderInfo = NULL;
if (dwInitLevel & MPR_INITCRITSEC_CREATED)
{
DeleteCriticalSection(&MprInitCritSec);
}
RegCloseKey(g_hProviderKey);
CloseHandle(g_hProvidersChanged);
if (dwInitLevel & MPR_PROVIDERLOCK_CREATED)
{
MPRProviderLock.Delete();
}
if (dwInitLevel & MPR_ERRORCRITSEC_CREATED)
{
MprFreeAllErrorRecords();
DeleteCriticalSection(&MprErrorRecCritSec);
}
return(TRUE);
}
BOOL
GetProviderCapabilities(
LPPROVIDER Provider
)
/*++
Routine Description:
This function obtains the provider's capabilities and then gets the
procedure entry point for all supported API.
Arguments:
Provider - A pointer to a PROVIDER structure which will contain all
information that must be maintained for a given provider.
Return Value:
TRUE - If at least one capability is found for this provider.
FALSE - If no capabilities are found for the provider.
History:
Johnl 17-Jan-1992 Added Property dialog support
--*/
//
// The following macro is designed to work within the function
// GetProviderCapabilities().
//
#define GET_ADDRESS(apiName) Provider-> ## apiName = (PF_NP ## apiName) GetProcAddress( \
Provider->Handle, \
"NP"#apiName); \
if (Provider-> ## apiName == NULL) { \
MPR_LOG(ERROR, \
"GetProviderCapabilities: Can't get NP"#apiName " Address %d\n",\
GetLastError()); \
} \
else { \
status = TRUE; \
}
#define GET_AUTH_ADDRESS(apiName) Provider-> ## apiName = (PF_NP ## apiName) GetProcAddress( \
Provider->AuthentHandle, \
"NP"#apiName); \
if (Provider-> ## apiName == NULL) { \
MPR_LOG(ERROR, \
"GetProviderCapabilities: Can't get NP"#apiName " Address %d\n",\
GetLastError()); \
} \
else { \
status = TRUE; \
}
{
DWORD bitMask;
BOOLEAN status=FALSE;
GET_ADDRESS(GetCaps);
if (status) {
//
// Get the Network Type
//
// CODEWORK: Read the type from the registry when possible -- this
// will allow us to avoid doing a Level 2 init in all the cases where
// only the provider type is needed (e.g., MprFindProviderByType).
//
Provider->Type = Provider->GetCaps(WNNC_NET_TYPE);
//
// Reject providers that don't supply their type
//
if (Provider->Type == 0)
{
MPR_LOG(ERROR, "%ws provider reported a net type of 0\n",
Provider->Resource.lpProvider);
ASSERT(!"Network provider didn't report its network type");
return FALSE;
}
//
// Does it support WNetGetUser?
//
bitMask = Provider->GetCaps(WNNC_USER);
if (bitMask & WNNC_USR_GETUSER) {
GET_ADDRESS(GetUser);
}
//
// Connection Api Supported
//
bitMask = Provider->GetCaps(WNNC_CONNECTION);
Provider->ConnectCaps = bitMask;
if (bitMask & WNNC_CON_ADDCONNECTION) {
GET_ADDRESS(AddConnection);
}
if (bitMask & WNNC_CON_ADDCONNECTION3) {
DWORD bitMask2;
GET_ADDRESS(AddConnection3);
GET_ADDRESS(GetReconnectFlags); // optional entry point
//
// Connection Api flags Supported
//
bitMask2 = Provider->GetCaps(WNNC_CONNECTION_FLAGS);
if (bitMask2 == 0) {
bitMask2 = WNNC_CF_DEFAULT;
}
bitMask2 &= WNNC_CF_MAXIMUM;
Provider->ConnectFlagCaps = bitMask2;
}
if (bitMask & WNNC_CON_CANCELCONNECTION) {
GET_ADDRESS(CancelConnection);
}
if (bitMask & WNNC_CON_GETCONNECTIONS) {
GET_ADDRESS(GetConnection);
GET_ADDRESS(GetConnection3);
GET_ADDRESS(GetUniversalName);
}
if (bitMask & WNNC_CON_GETPERFORMANCE) {
GET_ADDRESS(GetConnectionPerformance);
}
//
// Enumeration Api Supported
//
bitMask = Provider->GetCaps(WNNC_ENUMERATION);
if ((bitMask & WNNC_ENUM_GLOBAL) ||
(bitMask & WNNC_ENUM_LOCAL) ||
(bitMask & WNNC_ENUM_CONTEXT)) {
GET_ADDRESS(OpenEnum);
GET_ADDRESS(EnumResource);
GET_ADDRESS(CloseEnum);
}
//
// Admin Api Supported
//
bitMask = Provider->GetCaps(WNNC_ADMIN);
if (bitMask & WNNC_ADM_GETDIRECTORYTYPE) {
GET_ADDRESS(GetDirectoryType);
}
if (bitMask & WNNC_ADM_DIRECTORYNOTIFY) {
GET_ADDRESS(DirectoryNotify);
}
//
// Dialog API Support
//
bitMask = Provider->GetCaps(WNNC_DIALOG);
if (bitMask & WNNC_DLG_PROPERTYDIALOG) {
GET_ADDRESS(GetPropertyText);
GET_ADDRESS(PropertyDialog);
}
if (bitMask & WNNC_DLG_SEARCHDIALOG) {
GET_ADDRESS(SearchDialog);
}
if (bitMask & WNNC_DLG_FORMATNETWORKNAME) {
GET_ADDRESS(FormatNetworkName);
}
if (bitMask & WNNC_DLG_PERMISSIONEDITOR) {
GET_ADDRESS(FMXGetPermCaps);
GET_ADDRESS(FMXEditPerm);
GET_ADDRESS(FMXGetPermHelp);
}
if (bitMask & WNNC_DLG_GETRESOURCEPARENT) {
GET_ADDRESS(GetResourceParent);
}
if (bitMask & WNNC_DLG_GETRESOURCEINFORMATION) {
GET_ADDRESS(GetResourceInformation);
}
}
else {
if (Provider->GetAuthentCaps == NULL) {
Provider->GetAuthentCaps = (PF_NPGetCaps)GetProcAddress(
Provider->AuthentHandle,
"NPGetCaps");
if (Provider->GetAuthentCaps == NULL) {
MPR_LOG(ERROR,
"GetProviderCapabilities: Can't get NPGetCaps %d\n",
GetLastError());
}
else {
status = TRUE;
}
}
//
// If we couldn't get an address for GetCaps from either the
// network provider dll or the authentication provider dll, then
// we should return an error. The rule is, this must be supported
// by one of the providers.
//
if (status == FALSE) {
return(FALSE);
}
}
//
// Get Authentication Provider entry points
//
if (Provider->InitClass & CREDENTIAL_TYPE) {
if (Provider->AuthentHandle == NULL) {
MPR_LOG0(TRACE,"GetProvCaps: CM provider in same DLL\n");
GET_ADDRESS(LogonNotify);
GET_ADDRESS(PasswordChangeNotify);
}
else {
MPR_LOG0(TRACE,"GetProvCaps: CM provider in seperate DLL\n");
GET_AUTH_ADDRESS(LogonNotify);
GET_AUTH_ADDRESS(PasswordChangeNotify);
if (Provider->GetAuthentCaps == NULL) {
Provider->GetAuthentCaps = (PF_NPGetCaps)GetProcAddress(
Provider->AuthentHandle,
"NPGetCaps");
if (Provider->GetAuthentCaps == NULL) {
MPR_LOG(ERROR,
"GetProviderCapabilities: Can't get NPGetCaps %d\n",
GetLastError());
}
else {
status = TRUE;
}
}
}
}
return(status);
}
VOID
MprFreeGlobalProviderInfo(
VOID
)
/*++
Routine Description:
This function walks through the array of provider structures, and
frees up all the valid pointers to the provider name. Then if frees
the GlobalProviderInfo array.
Arguments:
none.
Return Value:
none.
--*/
{
LPPROVIDER provider;
DWORD i;
if (GlobalProviderInfo == NULL)
{
return;
}
provider = GlobalProviderInfo;
//
// Free all the valid pointers to the provider's name.
//
for(i = 0; i < GlobalNumProviders; i++, provider++)
{
MPR_LOG(TRACE, "Freeing Name for provider %d\n",i);
if (provider->Handle != NULL)
{
MPR_LOG1(TRACE,
"Freeing Library for "FORMAT_LPTSTR" \n",
provider->Resource.lpProvider);
FreeLibrary(provider->Handle);
}
LocalFree(provider->DllName);
if (provider->AuthentHandle != NULL)
{
MPR_LOG1(TRACE,
"Freeing authentication library for "FORMAT_LPTSTR" \n",
provider->Resource.lpProvider);
FreeLibrary(provider->AuthentHandle);
}
//
// Should have been freed when the DLL was loaded
//
ASSERT(provider->AuthentDllName == NULL);
LocalFree(provider->Resource.lpRemoteName);
}
//
// Free the top level data structure and reinit the globals
//
MPR_LOG(TRACE, "Freeing GlobalProviderInfo\n",0);
LocalFree(GlobalProviderInfo);
GlobalProviderInfo = NULL;
GlobalNumProviders = 0;
GlobalNumActiveProviders = 0;
}
DWORD
MprMakeServiceKeyName(
IN OUT LPTSTR *OrderedNamePtr,
OUT LPTSTR *NameBufferPtr,
OUT LPTSTR *NamePtr
)
/*++
Routine Description:
This function gets the name of the provider device driver key within
the registry tree. A buffer is allocated that is large enough for
that name as well as the name of the key where the provider information
is stored.
For example, the following buffer is allocated and filled with
only the path prefix followed by the LanmanRedirector name.
NameBuffer = [services\LanmanRedirector ]
The buffer is sized so that it can be filled in later with the
following information.
NameBuffer = [services\LanmanRedirector\NetworkProvider]
The Driver name is needed for comparison with the ordered list names.
Arguments:
OrderedNamePtr - On entry this points to a location where there is a
pointer to the the next name in the ordered list of providers.
These names are seperated by commas. The last entry is followed by
a trailing NUL.
On exit a pointer to the next name in the list is stored here. If
it is the last name, the pointer is NULL.
NameBufferPtr - This is a pointer to a location where the pointer to the
name buffer is to be placed.
NamePtr - This is a a location where the pointer to the provider
name is placed. In the above example, this pointer would point
to the beginnning of the LanmanRedirector portion of the string.
Return Value:
TRUE - We successfully obtained the driver name.
FALSE - We were unsuccessful in obtaining the driver name.
Note:
--*/
{
LPTSTR pNext;
DWORD bufferSize; // number of BYTES in buffer
DWORD nameLength; // number of CHARACTERS in string
//
// The OrderedNamePtr should always be valid. If not there is a software
// error in this code.
//
if (*OrderedNamePtr == NULL) {
MPR_LOG(ERROR,"GetDriverName: The ordered Name Ptr was NULL.\n",0);
return(WN_NO_NETWORK);
}
pNext = *OrderedNamePtr;
//
// Find the next NULL or COMMA.
//
while ( (*pNext != TEXT('\0')) &&
(*pNext != TEXT(',')) ){
pNext++;
}
//
// Allocate a buffer for the name to be stored in
//
bufferSize = (DWORD)((LPBYTE)pNext - (LPBYTE)(*OrderedNamePtr));
nameLength = bufferSize / sizeof(TCHAR);
bufferSize = bufferSize + STRSIZE(PROVIDER_PATH) + STRSIZE(SERVICES_KEY_NAME);
*NameBufferPtr = (LPTSTR) LocalAlloc(LPTR, bufferSize);
if (*NameBufferPtr == NULL) {
MPR_LOG(ERROR,"MprMakeServiceKeyName:LocalAllocFailed %d\n",GetLastError());
return(GetLastError());
}
//
// Copy the path prefix "services\\" followed by the name
// into the buffer and terminate with a NULL character.
//
STRCPY(*NameBufferPtr, SERVICES_KEY_NAME);
*NamePtr = (LPTSTR)(*NameBufferPtr + (STRLEN(*NameBufferPtr)));
STRNCAT(*NameBufferPtr, *OrderedNamePtr, nameLength);
*((*NamePtr) + nameLength) = TEXT('\0');
if (*pNext == TEXT('\0')) {
*OrderedNamePtr = NULL;
}
else {
*OrderedNamePtr = pNext + 1;
}
return (WN_SUCCESS);
}
BOOL
MprGetOrderedList(
HKEY ControlRootKey,
LPTSTR *OrderString
)
/*++
Routine Description:
This function returns a pointer to a string that contains the
ordered list of providers. This ordered list is to be used when
we go through the list of providers with a trial-and-error method
when servicing a WINNET API call.
ALLOCATES STORAGE: This function allocates the buffer for the
OrderString.
Arguments:
OrderString - This is a pointer to a location where the pointer to
the order string is to be placed.
Return Value:
TRUE - The ordered list was found, and the string was returned.
FALSE - A failure occured when attempting to find the ordered list.
--*/
{
HKEY orderKey;
*OrderString = NULL;
//
// Get a handle to the key for the ordered provider information.
//
if(!MprOpenKey(
ControlRootKey, // hKey
PROVIDER_ORDER_KEY, // lpSubKey
&orderKey, // Newly Opened Key Handle
DA_READ)) { // Desired Access
MPR_LOG(ERROR,"MprGetOrderedList: MprOpenKey (ActiveProviders) Error\n",0);
return(FALSE);
}
//
// Get the provider order string from the registry.
//
if (!MprGetKeyValue(orderKey, TEXT("ProviderOrder"), OrderString)) {
MPR_LOG(ERROR,"MprGetOrderedList: MprGetKeyValue Error.\n", 0);
RegCloseKey(orderKey);
return(FALSE);
}
RegCloseKey(orderKey);
return(TRUE);
}
DWORD
MprGetNumProviders(
LPTSTR OrderedString
)
/*++
Routine Description:
This routine finds the number of provider names in the string of
ordered provider names.
Arguments:
OrderedString - This is a pointer to a NUL terminated string of
ordered provider names. Names are seperated by commas.
Return Value:
--*/
{
DWORD count = 0;
LPTSTR pBegin;
pBegin = OrderedString;
while (*OrderedString != TEXT('\0')) {
//
// If a seperator has been found, and if there are characters
// between it and the last seperator (or the beginning), then
// increment the count.
//
if (*OrderedString == TEXT(',')) {
if((OrderedString - pBegin) > 0) {
count++;
}
pBegin = OrderedString + 1;
}
OrderedString++;
}
//
// The last name is expected to be followed by a NUL rather than a
// comma.
//
if ((OrderedString - pBegin) > 0) {
count++;
}
return(count);
}
BOOL
MprExtractProviderInfo(
HKEY ProviderInfoKey,
LPPROVIDER Provider
)
/*++
Routine Description:
This function extracts information from the ProviderInfoKey location
in the registry and stores it in the Provider data structure.
Arguments:
ProviderInfoKey - This is a registry key handle for the location in
the registry where we expect to obtain all the necessary
information about a provider.
Provider - This is a pointer to a provider structure that is to be
filled in with the information from the registry.
Return Value:
TRUE - The operation was completely successful.
FALSE - The operation failed.
Note:
--*/
{
LPTSTR providerName = NULL;
LPTSTR fileName = NULL;
DWORD ValueType;
DWORD Class;
DWORD classSize = sizeof(DWORD);
DWORD status;
MPR_LOG(TRACE,"In MprExtractProviderInfo....\n",0);
//
// Get Provider Name
//
if(!MprGetKeyValue(ProviderInfoKey, TEXT("name"), &providerName) ||
providerName[0] == L'\0') {
MPR_LOG0(ERROR,"MprExtractProviderInfo: Couldn't get provider name\n");
return(FALSE);
}
//
// Initialize all the fields in the resource structure.
// NOTE: The reserved field in dwUsage is set to indicate this is
// a top level structure.
//
Provider->Resource.lpProvider = providerName;
Provider->Resource.lpRemoteName = providerName;
Provider->Resource.dwScope = RESOURCE_GLOBALNET;
Provider->Resource.dwType = 0;
Provider->Resource.dwDisplayType= RESOURCEDISPLAYTYPE_NETWORK;
Provider->Resource.dwUsage = RESOURCEUSAGE_CONTAINER | RESOURCEUSAGE_RESERVED;
Provider->Resource.lpComment = NULL;
//
// Get the Provider Class. If there isn't a Class value in the
// registry, assume it is a NetworkClass only.
//
status = RegQueryValueEx(
ProviderInfoKey, // hKey
VALUE_CLASS, // lpValueName
NULL, // lpTitleIndex
&ValueType, // lpType
(LPBYTE)&Class, // lpData
&classSize); // lpcbData
if (status != NO_ERROR) {
//
// If we get an error we assume it is because there is no key - thus
// indicating that this provider is a network provider only.
//
MPR_LOG0(TRACE,"Couldn't find Authenticator Class value "
"- assume it is Network-type only\n");
Class = WN_NETWORK_CLASS;
}
Provider->InitClass = Class;
MPR_LOG1(TRACE,"MprExtractProviderInfo: Provider InitClass = %d\n",Class);
//
// Get the name of the provider's DLL routine
//
if(!MprGetKeyValue(ProviderInfoKey,VALUE_PATHNAME,&fileName)){
MPR_LOG(ERROR,
"MprExtractProviderInfo: Failed to get the Dll path from registry\n",0);
//
// If this is a network class provider, it MUST have a provider
// dll. Otherwise, we return a failure.
//
if (Class & WN_NETWORK_CLASS) {
return(FALSE);
}
else {
fileName = NULL;
}
}
Provider->DllName = fileName;
//
// If this is a CREDENTIAL_CLASS or PRIMARY_AUTHENT_CLASS provider
// then try to get the dll name for the authentication provider.
//
if (Class & (WN_CREDENTIAL_CLASS | WN_PRIMARY_AUTHENT_CLASS)) {
//
// Get the name of the provider's Credential Management DLL routine
//
if(!MprGetKeyValue(ProviderInfoKey,AUTHENT_PATHNAME,&fileName)){
//
// If we can't get a name for the Authentication Provider's DLL,
// then we will assume the network provider dll exports the
// Credential Management functions.
//
MPR_LOG0(ERROR,
"MprExtractProviderInfo: Failed to get the Authenticator "
"Dll path from registry\n");
//
// If we don't have a provider dll name, or an authent provider
// dll name, then this is an error condition.
//
if (Provider->DllName == NULL) {
return(FALSE);
}
}
else {
Provider->AuthentDllName = fileName;
}
}
return(TRUE);
}
BOOL
MprGetLUIDDeviceMapsEnabled(
PBOOL pResult
)
/*++
Routine Description:
This function calls NtQueryInformationProcess() to determine if
LUID device maps are enabled
Arguments:
pResult - returns the result of "Is LUID device maps are enabled?"
TRUE - LUID device maps are enabled
FALSE - LUID device maps are disabled
Return Value:
TRUE - This is returned if we are able to determine if LUID device
maps are enabled/disabled.
FALSE - Encountered an error
--*/
{
NTSTATUS Status;
ULONG LUIDDeviceMapsEnabled;
BOOL Result;
if( pResult == NULL ) {
return (FALSE);
}
Status = NtQueryInformationProcess( NtCurrentProcess(),
ProcessLUIDDeviceMapsEnabled,
&LUIDDeviceMapsEnabled,
sizeof(LUIDDeviceMapsEnabled),
NULL
);
if( !NT_SUCCESS(Status) ) {
MPR_LOG(ERROR,"MprGetLUIDDeviceMapsEnabled: NtQueryInformationProcess (g_LUIDDeviceMapsEnabled) Error\n",0);
return( FALSE );
}
else{
Result = (LUIDDeviceMapsEnabled != 0);
}
*pResult = Result;
return( TRUE );
}
extern "C" int __cdecl
_purecall(
VOID
)
{
// Make it build... (the compiler will initialize all pure virtual
// function pointers to point to this... If we hit this function,
// there's a really big problem.)
RtlRaiseStatus(STATUS_NOT_IMPLEMENTED);
return 0;
}