592 lines
16 KiB
C
Raw Normal View History

2001-01-01 00:00:00 +01:00
/*++
Copyright (c) 1992 Microsoft Corporation
Module Name:
NwInit.c
Abstract:
This module implements the DRIVER_INITIALIZATION routine for NetWare
Author:
Colin Watson [ColinW] 15-Dec-1992
Revision History:
--*/
#include "Procs.h"
#include "wdmsec.h"
#define Dbg (DEBUG_TRACE_LOAD)
//
// Private declaration because ZwQueryDefaultLocale isn't in any header.
//
NTSYSAPI
NTSTATUS
NTAPI
ZwQueryDefaultLocale(
IN BOOLEAN UserProfile,
OUT PLCID DefaultLocaleId
);
NTSTATUS
DriverEntry(
IN PDRIVER_OBJECT DriverObject,
IN PUNICODE_STRING RegistryPath
);
VOID
UnloadDriver(
IN PDRIVER_OBJECT DriverObject
);
VOID
GetConfigurationInformation(
PUNICODE_STRING RegistryPath
);
VOID
ReadValue(
HANDLE ParametersHandle,
PLONG pVar,
PWCHAR Name
);
#ifdef ALLOC_PRAGMA
#pragma alloc_text( PAGE, DriverEntry )
#pragma alloc_text( PAGE, GetConfigurationInformation )
#pragma alloc_text( PAGE, ReadValue )
#endif
#if 0 // Not pageable
UnloadDriver
#endif
#ifdef _PNP_POWER_
extern HANDLE TdiBindingHandle;
#endif
static ULONG IrpStackSize;
NTSTATUS
DriverEntry(
IN PDRIVER_OBJECT DriverObject,
IN PUNICODE_STRING RegistryPath
)
/*++
Routine Description:
This is the initialization routine for the Nw file system
device driver. This routine creates the device object for the FileSystem
device and performs all other driver initialization.
Arguments:
DriverObject - Pointer to driver object created by the system.
Return Value:
NTSTATUS - The function value is the final status from the initialization
operation.
--*/
{
NTSTATUS Status;
UNICODE_STRING UnicodeString;
UNICODE_STRING SddlString;
PAGED_CODE();
//DbgBreakPoint();
InitializeAttach( );
NwInitializeData();
//NwInitializePidTable(); // Terminal Server code merge -
// NwInitalizePidTable in the NwAllocateAndInitScb
// Pid table is per SCB base.
//
// Create the device object.
//
RtlInitUnicodeString( &UnicodeString, DD_NWFS_DEVICE_NAME_U );
RtlInitUnicodeString( &SddlString, L"D:P(A;;GX;;;WD)(A;;GA;;;SY)(A;;GA;;;BA)(A;;GA;;;LS)(A;;GA;;;NS)(A;;GX;;;RC)" );
Status = IoCreateDeviceSecure( DriverObject,
0,
&UnicodeString,
FILE_DEVICE_NETWORK_FILE_SYSTEM,
FILE_REMOTE_DEVICE,
FALSE,
&SddlString,
NULL,
&FileSystemDeviceObject );
if (!NT_SUCCESS( Status )) {
Error(EVENT_NWRDR_CANT_CREATE_DEVICE, Status, NULL, 0, 0);
return Status;
}
//
// Initialize parameters to the defaults.
//
IrpStackSize = NWRDR_IO_STACKSIZE;
//
// Attempt to read config information from the registry
//
GetConfigurationInformation( RegistryPath );
//
// Set the stack size.
//
FileSystemDeviceObject->StackSize = (CCHAR)IrpStackSize;
//
// Initialize the driver object with this driver's entry points.
//
DriverObject->MajorFunction[IRP_MJ_CREATE] = (PDRIVER_DISPATCH)NwFsdCreate;
DriverObject->MajorFunction[IRP_MJ_CLEANUP] = (PDRIVER_DISPATCH)NwFsdCleanup;
DriverObject->MajorFunction[IRP_MJ_CLOSE] = (PDRIVER_DISPATCH)NwFsdClose;
DriverObject->MajorFunction[IRP_MJ_FILE_SYSTEM_CONTROL] = (PDRIVER_DISPATCH)NwFsdFileSystemControl;
DriverObject->MajorFunction[IRP_MJ_DEVICE_CONTROL] = (PDRIVER_DISPATCH)NwFsdDeviceIoControl;
DriverObject->MajorFunction[IRP_MJ_QUERY_INFORMATION] = (PDRIVER_DISPATCH)NwFsdQueryInformation;
DriverObject->MajorFunction[IRP_MJ_QUERY_VOLUME_INFORMATION] = (PDRIVER_DISPATCH)NwFsdQueryVolumeInformation;
DriverObject->MajorFunction[IRP_MJ_SET_VOLUME_INFORMATION] = (PDRIVER_DISPATCH)NwFsdSetVolumeInformation;
DriverObject->MajorFunction[IRP_MJ_DIRECTORY_CONTROL] = (PDRIVER_DISPATCH)NwFsdDirectoryControl;
DriverObject->MajorFunction[IRP_MJ_READ] = (PDRIVER_DISPATCH)NwFsdRead;
DriverObject->MajorFunction[IRP_MJ_WRITE] = (PDRIVER_DISPATCH)NwFsdWrite;
DriverObject->MajorFunction[IRP_MJ_SET_INFORMATION] = (PDRIVER_DISPATCH)NwFsdSetInformation;
DriverObject->MajorFunction[IRP_MJ_LOCK_CONTROL] = (PDRIVER_DISPATCH)NwFsdLockControl;
DriverObject->MajorFunction[IRP_MJ_FLUSH_BUFFERS] = (PDRIVER_DISPATCH)NwFsdFlushBuffers;
#ifdef _PNP_POWER_
DriverObject->MajorFunction[IRP_MJ_PNP] = (PDRIVER_DISPATCH)NwFsdProcessPnpIrp;
#endif
/*
DriverObject->MajorFunction[IRP_MJ_QUERY_EA] = (PDRIVER_DISPATCH)NwFsdQueryEa;
DriverObject->MajorFunction[IRP_MJ_SET_EA] = (PDRIVER_DISPATCH)NwFsdSetEa;
DriverObject->MajorFunction[IRP_MJ_SHUTDOWN] = (PDRIVER_DISPATCH)NwFsdShutdown;
*/
DriverObject->DriverUnload = UnloadDriver;
#if NWFASTIO
DriverObject->FastIoDispatch = &NwFastIoDispatch;
NwFastIoDispatch.SizeOfFastIoDispatch = sizeof(FAST_IO_DISPATCH);
NwFastIoDispatch.FastIoCheckIfPossible = NULL;
NwFastIoDispatch.FastIoRead = NwFastRead;
NwFastIoDispatch.FastIoWrite = NwFastWrite;
NwFastIoDispatch.FastIoQueryBasicInfo = NwFastQueryBasicInfo;
NwFastIoDispatch.FastIoQueryStandardInfo = NwFastQueryStandardInfo;
NwFastIoDispatch.FastIoLock = NULL;
NwFastIoDispatch.FastIoUnlockSingle = NULL;
NwFastIoDispatch.FastIoUnlockAll = NULL;
NwFastIoDispatch.FastIoUnlockAllByKey = NULL;
NwFastIoDispatch.FastIoDeviceControl = NULL;
#endif
NwInitializeRcb( &NwRcb );
InitializeIrpContext( );
NwPermanentNpScb.State = SCB_STATE_DISCONNECTING;
//
// Do a kludge here so that we get to the "real" global variables.
//
//NlsLeadByteInfo = *(PUSHORT *)NlsLeadByteInfo;
//NlsMbCodePageTag = *(*(PBOOLEAN *)&NlsMbCodePageTag);
#ifndef IFS
FsRtlLegalAnsiCharacterArray = *(PUCHAR *)FsRtlLegalAnsiCharacterArray;
#endif
//
// Register as a file system, notifies filters
//
IoRegisterFileSystem(FileSystemDeviceObject);
DebugTrace(0, Dbg, "NetWare redirector loaded\n", 0);
//
// And return to our caller
//
return( STATUS_SUCCESS );
}
VOID
UnloadDriver(
IN PDRIVER_OBJECT DriverObject
)
/*++
Routine Description:
This is the unload routine for the NetWare redirector filesystem.
Arguments:
DriverObject - pointer to the driver object for the redirector
Return Value:
None
--*/
{
KIRQL OldIrql;
NTSTATUS status;
//
// Lock down code
//
DebugTrace(0, Dbg, "UnloadDriver called\n", 0);
//
// Unregister as a file system, notifies filters
//
IoUnregisterFileSystem(FileSystemDeviceObject);
//
// tommye - MS bug 33463
//
// Clean up our cached credentials - this fixes a
// memory leak when we shut down.
//
{
LARGE_INTEGER Unused;
KeQuerySystemTime( &Unused );
CleanupSupplementalCredentials(Unused, TRUE);
}
NwReferenceUnlockableCodeSection ();
TerminateWorkerThread();
#ifdef _PNP_POWER_
//
// Unregister the bind handler with tdi.
//
if ( TdiBindingHandle != NULL ) {
status = TdiDeregisterPnPHandlers( TdiBindingHandle );
TdiBindingHandle = NULL;
DebugTrace(0, Dbg,"TDI binding handle deregistered\n",0);
}
#endif
IpxClose();
IPX_Close_Socket( &NwPermanentNpScb.Server );
KeAcquireSpinLock( &ScbSpinLock, &OldIrql );
RemoveEntryList( &NwPermanentNpScb.ScbLinks );
KeReleaseSpinLock( &ScbSpinLock, OldIrql );
DestroyAllScb();
UninitializeIrpContext();
NwDereferenceUnlockableCodeSection ();
NwUnlockCodeSections(FALSE);
StopTimer();
if (IpxTransportName.Buffer != NULL) {
FREE_POOL(IpxTransportName.Buffer);
}
if ( NwProviderName.Buffer != NULL ) {
FREE_POOL( NwProviderName.Buffer );
}
//NwUninitializePidTable(); //Terminal Server code merge -
//NwUninitializePidTable is called in
//NwDeleteScb. Pid table is per SCB base.
ASSERT( IsListEmpty( &NwPagedPoolList ) );
ASSERT( IsListEmpty( &NwNonpagedPoolList ) );
ASSERT( MdlCount == 0 );
ASSERT( IrpCount == 0 );
NwDeleteRcb( &NwRcb );
#ifdef NWDBG
ExDeleteResourceLite( &NwDebugResource );
#endif
ExDeleteResourceLite( &NwOpenResource );
ExDeleteResourceLite( &NwUnlockableCodeResource );
IoDeleteDevice(FileSystemDeviceObject);
DebugTrace(0, Dbg, "NetWare redirector unloaded\n\n", 0);
}
VOID
GetConfigurationInformation(
PUNICODE_STRING RegistryPath
)
/*++
Routine Description:
This routine read redirector configuration information from the registry.
Arguments:
RegistryPath - A pointer the a path to the
Return Value:
None
--*/
{
UNICODE_STRING UnicodeString;
HANDLE ConfigHandle;
HANDLE ParametersHandle;
NTSTATUS Status;
OBJECT_ATTRIBUTES ObjectAttributes;
ULONG TimeOutEventinMins = 0L;
LCID lcid;
PAGED_CODE();
Japan = FALSE;
Korean = FALSE;
ZwQueryDefaultLocale( FALSE, &lcid );
if (PRIMARYLANGID(lcid) == LANG_JAPANESE ||
PRIMARYLANGID(lcid) == LANG_KOREAN ||
PRIMARYLANGID(lcid) == LANG_CHINESE) {
Japan = TRUE;
if (PRIMARYLANGID(lcid) == LANG_KOREAN){
Korean = TRUE;
}
}
InitializeObjectAttributes(
&ObjectAttributes,
RegistryPath, // name
OBJ_CASE_INSENSITIVE, // attributes
NULL, // root
NULL // security descriptor
);
Status = ZwOpenKey ( &ConfigHandle, KEY_READ, &ObjectAttributes );
if (!NT_SUCCESS(Status)) {
return;
}
RtlInitUnicodeString( &UnicodeString, L"Parameters" );
InitializeObjectAttributes(
&ObjectAttributes,
&UnicodeString,
OBJ_CASE_INSENSITIVE,
ConfigHandle,
NULL
);
Status = ZwOpenKey( &ParametersHandle, KEY_READ, &ObjectAttributes );
if ( !NT_SUCCESS( Status ) ) {
ZwClose( ConfigHandle );
return;
}
ReadValue( ParametersHandle, &IrpStackSize, L"IrpStackSize" );
ReadValue( ParametersHandle, &MaxSendDelay, L"MaxSendDelay" );
ReadValue( ParametersHandle, &MaxReceiveDelay, L"MaxReceiveDelay" );
ReadValue( ParametersHandle, &MinSendDelay, L"MinSendDelay" );
ReadValue( ParametersHandle, &MinReceiveDelay, L"MinReceiveDelay" );
ReadValue( ParametersHandle, &BurstSuccessCount, L"BurstSuccessCount" );
ReadValue( ParametersHandle, &BurstSuccessCount2, L"BurstSuccessCount2" );
ReadValue( ParametersHandle, &MaxReadTimeout, L"MaxReadTimeout" );
ReadValue( ParametersHandle, &MaxWriteTimeout, L"MaxWriteTimeout" );
ReadValue( ParametersHandle, &ReadTimeoutMultiplier, L"ReadTimeoutMultiplier" );
ReadValue( ParametersHandle, &WriteTimeoutMultiplier, L"WriteTimeoutMultiplier" );
ReadValue( ParametersHandle, &AllowGrowth, L"AllowGrowth" );
ReadValue( ParametersHandle, &DontShrink, L"DontShrink" );
ReadValue( ParametersHandle, &SendExtraNcp, L"SendExtraNcp" );
ReadValue( ParametersHandle, &DefaultMaxPacketSize, L"DefaultMaxPacketSize" );
ReadValue( ParametersHandle, &PacketThreshold, L"PacketThreshold" );
ReadValue( ParametersHandle, &LargePacketAdjustment, L"LargePacketAdjustment" );
ReadValue( ParametersHandle, &LipPacketAdjustment, L"LipPacketAdjustment" );
ReadValue( ParametersHandle, &LipAccuracy, L"LipAccuracy" );
ReadValue( ParametersHandle, &DisableReadCache, L"DisableReadCache" );
ReadValue( ParametersHandle, &DisableWriteCache, L"DisableWriteCache" );
ReadValue( ParametersHandle, &FavourLongNames, L"FavourLongNames" );
ReadValue( ParametersHandle, &LongNameFlags, L"LongNameFlags" );
ReadValue( ParametersHandle, &DirCacheEntries, L"DirectoryCacheSize" );
if( DirCacheEntries == 0 ) {
DirCacheEntries = 1;
}
if( DirCacheEntries > MAX_DIR_CACHE_ENTRIES ) {
DirCacheEntries = MAX_DIR_CACHE_ENTRIES;
}
ReadValue( ParametersHandle, &LockTimeoutThreshold, L"LockTimeout" );
ReadValue( ParametersHandle, &TimeOutEventinMins, L"TimeOutEventinMins");
ReadValue( ParametersHandle, &EnableMultipleConnects, L"EnableMultipleConnects");
ReadValue( ParametersHandle, &AllowSeedServerRedirection, L"AllowSeedServerRedirection");
ReadValue( ParametersHandle, &ReadExecOnlyFiles, L"ReadExecOnlyFiles");
ReadValue( ParametersHandle, &DisableAltFileName, L"DisableAltFileName");
ReadValue( ParametersHandle, &NwAbsoluteTotalWaitTime, L"AbsoluteTotalWaitTime");
ReadValue( ParametersHandle, &NdsObjectCacheSize, L"NdsObjectCacheSize" );
ReadValue( ParametersHandle, &NdsObjectCacheTimeout, L"NdsObjectCacheTimeout" );
ReadValue ( ParametersHandle, &PreferNDSBrowsing, L"PreferNDSBrowsing" );
//
// Make sure the object cache values are within bounds.
//
// NOTE: If the timeout is set to zero, then the cache is
// effectively disabled. NdsObjectCacheSize is set
// to zero to accomplish this.
//
if( NdsObjectCacheSize > MAX_NDS_OBJECT_CACHE_SIZE ) {
NdsObjectCacheSize = MAX_NDS_OBJECT_CACHE_SIZE;
}
if( NdsObjectCacheTimeout > MAX_NDS_OBJECT_CACHE_TIMEOUT ) {
NdsObjectCacheTimeout = MAX_NDS_OBJECT_CACHE_TIMEOUT;
} else if( NdsObjectCacheTimeout == 0 ) {
NdsObjectCacheSize = 0;
}
if (!TimeOutEventinMins) {
//
// If for some reason, the registry has set the TimeOutEventInterval
// to zero, reset to the default value to avoid divide-by-zero
//
TimeOutEventinMins = DEFAULT_TIMEOUT_EVENT_INTERVAL;
}
TimeOutEventInterval.QuadPart = TimeOutEventinMins * 60 * SECONDS;
//
// tommye - MS bug 2743 we now get the RetryCount from the registry, providing
// a default of DEFAULT_RETRY_COUNT.
//
{
LONG TempRetryCount;
TempRetryCount = DEFAULT_RETRY_COUNT;
ReadValue( ParametersHandle, &TempRetryCount, L"DefaultRetryCount");
DefaultRetryCount = (SHORT) TempRetryCount & 0xFFFF;
}
ZwClose( ParametersHandle );
ZwClose( ConfigHandle );
}
VOID
ReadValue(
HANDLE ParametersHandle,
PLONG pVar,
PWCHAR Name
)
/*++
Routine Description:
This routine reads a single redirector configuration value from the registry.
Arguments:
Parameters - Supplies where to look for values.
pVar - Address of the variable to receive the new value if the name exists.
Name - Name whose value is to be loaded.
Return Value:
None
--*/
{
WCHAR Storage[256];
UNICODE_STRING UnicodeString;
NTSTATUS Status;
ULONG BytesRead;
PKEY_VALUE_FULL_INFORMATION Value = (PKEY_VALUE_FULL_INFORMATION)Storage;
PAGED_CODE();
UnicodeString.Buffer = Storage;
RtlInitUnicodeString(&UnicodeString, Name );
Status = ZwQueryValueKey(
ParametersHandle,
&UnicodeString,
KeyValueFullInformation,
Value,
sizeof(Storage),
&BytesRead );
if ( NT_SUCCESS( Status ) ) {
if ( Value->DataLength >= sizeof(ULONG) ) {
*pVar = *(LONG UNALIGNED *)( (PCHAR)Value + Value->DataOffset );
}
}
}