2020-09-30 16:53:55 +02:00

443 lines
10 KiB
C

/*++
Copyright (c) 1989 Microsoft Corporation
Module Name:
NwData.c
Abstract:
This module declares the global data used by the Nw file system.
Author:
Colin Watson [ColinW] 19-Dec-1992
Anoop Anantha [AnoopA] 24-Jun-1998
Revision History:
--*/
#include "Procs.h"
#include <stdlib.h>
//
// The debug trace level
//
#define Dbg (DEBUG_TRACE_CATCH_EXCEPTIONS)
PEPROCESS FspProcess;
PDEVICE_OBJECT FileSystemDeviceObject = NULL;
//
// The volume control block for the redirector device.
//
RCB NwRcb;
//
// The ScbSpinLock protects the entire ScbQueue and the first part of the
// Scb entries on the queue. The first part of the Scb includes the name
// of the server and a reference count
//
KSPIN_LOCK ScbSpinLock;
LIST_ENTRY ScbQueue;
//
// The NwTimerSpinLock protects the Timer and TimerStop flag.
//
KSPIN_LOCK NwTimerSpinLock;
//
// A permanent SCB to synchronize access to the network.
//
NONPAGED_SCB NwPermanentNpScb;
LARGE_INTEGER NwMaxLarge = {MAXULONG,MAXLONG};
//
// tommye MS 90541 / MCS 277
//
// Set NwAbsoluteTotalWaitTime to 200, which is 100 half-seconds (duh). This gets
// referenced in NwProcessPositiveAck, which comes from the TimerDPC about every
// half second or so. This is the longest about of retries that we will send a
// packet that we've gotten a positive ACK on.
//
ULONG NwAbsoluteTotalWaitTime = 200;
TDI_ADDRESS_IPX OurAddress = {0,0,0,0,0,0,0,0};
UNICODE_STRING IpxTransportName;
HANDLE IpxHandle = NULL;
PDEVICE_OBJECT pIpxDeviceObject = NULL;
PFILE_OBJECT pIpxFileObject = NULL;
LIST_ENTRY LogonList;
LOGON Guest;
LARGE_INTEGER DefaultLuid = ANONYMOUS_LOGON_LUID;
//
// A global list of VCBs, and a monotonic increasing VCB entry, used to
// control connection enumeration.
//
LIST_ENTRY GlobalVcbList;
ULONG CurrentVcbEntry;
#if 0
//
// HACKHACK - List of outstanding find notify request
// Protected by NwRcb resource.
//
LIST_ENTRY FnList;
#endif
//
// Drive mapping table of redirected drives. 26 disk drive mappings +
// 10 LPT mappings.
//
// Netware supports 32 disk redirections, but this funkiness is handled
// by the 16-bit code.
//
PVCB GlobalDriveMapTable[DRIVE_MAP_TABLE_SIZE]; //MultiUser
FAST_IO_DISPATCH NwFastIoDispatch;
//
// Scavenger related data
//
ULONG NwScavengerTickCount; // The current tick count
ULONG NwScavengerTickRunCount; // The count at which to run the scavenger routine
KSPIN_LOCK NwScavengerSpinLock; // Lock to protect access to the above.
//
// Worker thread
//
BOOLEAN WorkerThreadRunning = FALSE;
//
// Message queue data
//
LIST_ENTRY NwGetMessageList; // List of Get Message IRP contexts
KSPIN_LOCK NwMessageSpinLock; // Protects the list above.
//
// Pending lock list
//
LIST_ENTRY NwPendingLockList; // List of pending File lock IRP contexts
KSPIN_LOCK NwPendingLockSpinLock;// Protects the list above.
//
// Lock to synchronize all file opens.
//
ERESOURCE NwOpenResource;
//
// Configuration data
//
LONG PreferNDSBrowsing = 0; // when attempting to connect to UNC paths, attempt NDS connection first
BOOLEAN NwBurstModeEnabled = FALSE;
ULONG NwMaxSendSize = 0;
ULONG NwMaxReceiveSize = 0;
ULONG NwPrintOptions = 0x98;
UNICODE_STRING NwProviderName = { 0, 0, NULL };
LONG MaxSendDelay = 50000;
LONG MaxReceiveDelay = 50000;
LONG MinSendDelay = 0;
LONG MinReceiveDelay = 0;
LONG BurstSuccessCount = 1;
LONG BurstSuccessCount2 = 3;
LONG AllowGrowth = 0;
LONG DontShrink = 0;
LONG SendExtraNcp = 1;
LONG DefaultMaxPacketSize = 0;
LONG PacketThreshold = 1500; // Size to use Large vs Small PacketAdjustment
LONG LargePacketAdjustment = 38;
LONG LipPacketAdjustment = 0;
LONG LipAccuracy = BURST_PACKET_SIZE_TOLERANCE;
LONG Japan = 0; // Controls special DBCS translation
LONG Korean = 0; // Controls special Korean translation
LONG DisableReadCache = 0; // disable file i/o read cache
LONG DisableWriteCache = 0; // disable file i/o write cache
LONG FavourLongNames = 1 ; // use LFN where possible
DWORD LongNameFlags = 0; // flags for handling long names
ULONG DirCacheEntries = 1; // number of directory entries we cache
LARGE_INTEGER TimeOutEventInterval = {0, 0};
LONG MaxWriteTimeout = 50 ; // tick counts (see write.c)
LONG MaxReadTimeout = 50 ; // tick counts (see read.c)
LONG WriteTimeoutMultiplier = 100; // expressed as percentage (see write.c)
LONG ReadTimeoutMultiplier = 100; // expressed as percentage (see read.c)
ULONG EnableMultipleConnects = 0;
ULONG AllowSeedServerRedirection = 0;
ULONG ReadExecOnlyFiles = 0;
ULONG DisableAltFileName = 1;
ULONG NdsObjectCacheSize = 0;
ULONG NdsObjectCacheTimeout = 10;
//
// Static storage area for perfmon statistics
//
NW_REDIR_STATISTICS Stats;
ULONG ContextCount = 0;
//
// Data structure used to track discardable code.
//
SECTION_DESCRIPTOR NwSectionDescriptor;
ERESOURCE NwUnlockableCodeResource;
//
// The lock timeout value.
//
ULONG LockTimeoutThreshold = 1;
//
// The Kernel Queue from where the reconnect work items are picked up.
//
KQUEUE KernelQueue;
#ifndef _PNP_POWER_
//
// The TDI PNP Bind handle.
//
HANDLE TdiBindingHandle = NULL;
UNICODE_STRING TdiIpxDeviceName;
WCHAR IpxDevice[] = L"\\Device\\NwlnkIpx";
#endif
//
// We can't have the scavenger and a line change request running
// at the same time since they both run on worker threads and
// walk across all the SCBs. Therefore, when either is running,
// we set the WorkerRunning value used by the scavenger to TRUE.
// If a scavenger run tries to happen while a line change request
// is running, it gets skipped. If a line change request comes in
// while the scavenger is running, we set DelayedProcessLineChange
// to TRUE and run it when the scavenger finishes.
//
// These values are protected by the existing scavenger spin lock.
//
BOOLEAN DelayedProcessLineChange = FALSE;
PIRP DelayedLineChangeIrp = NULL;
#ifdef NWDBG
ULONG NwDebug = 0;
//ULONG NwDebug = 0xffffffbf;
ULONG NwMemDebug = 0xffffffff;
LONG NwDebugTraceIndent = 0;
ULONG NwFsdEntryCount = 0;
ULONG NwFspEntryCount = 0;
ULONG NwIoCallDriverCount = 0;
LONG NwPerformanceTimerLevel = 0x00000000;
ULONG NwTotalTicks[32] = { 0 };
//
// Debug data for tracking pool usage
//
KSPIN_LOCK NwDebugInterlock;
ERESOURCE NwDebugResource;
LIST_ENTRY NwPagedPoolList;
LIST_ENTRY NwNonpagedPoolList;
ULONG MdlCount;
ULONG IrpCount;
#endif // NWDBG
//
// Configurable parameters.
//
SHORT DefaultRetryCount = DEFAULT_RETRY_COUNT;
#ifdef _PNP_POWER_
BOOLEAN fPoweringDown = FALSE;
#endif
#ifdef ALLOC_PRAGMA
#pragma alloc_text( PAGE, NwInitializeData )
#endif
VOID
NwInitializeData(
VOID
)
{
LARGE_INTEGER Now;
PAGED_CODE();
NwRcb.State = RCB_STATE_STOPPED;
#ifdef NWDBG
// Initialize pool before allocating any memory
InitializeListHead( &NwPagedPoolList );
InitializeListHead( &NwNonpagedPoolList );
ExInitializeResourceLite( &NwDebugResource );
KeInitializeSpinLock( &NwDebugInterlock );
MdlCount = 0;
IrpCount = 0;
#endif
ExInitializeResourceLite( &NwOpenResource );
//
// Initialize the scavenger spin lock and run tick count.
//
KeInitializeSpinLock( &NwScavengerSpinLock );
NwScavengerTickRunCount = DEFAULT_SCAVENGER_TICK_RUN_COUNT;
//
// Initialize the timer spin lock.
//
KeInitializeSpinLock( &NwTimerSpinLock );
RtlInitUnicodeString( &IpxTransportName, NULL );
#ifndef _PNP_POWER_
RtlInitUnicodeString( &TdiIpxDeviceName, IpxDevice );
#endif
//
// Allocate a permanent Non-paged SCB. This SCB is used to
// synchronize access to finding the nearest server.
// This initialization must be done before the first possible call
// to UnloadDriver.
//
RtlZeroMemory( &NwPermanentNpScb, sizeof( NONPAGED_SCB ) );
NwPermanentNpScb.NodeTypeCode = NW_NTC_SCBNP;
NwPermanentNpScb.NodeByteSize = sizeof(NONPAGED_SCB);
NwPermanentNpScb.Reference = 1;
InitializeListHead( &NwPermanentNpScb.Requests );
//
// Initialize the logonlist to have a default entry with server NULL,
// username "GUEST" and null password. This will always be the last
// entry on the logonlist so that the workstation service can supply
// an override.
//
InitializeListHead( &LogonList );
Guest.NodeTypeCode = NW_NTC_LOGON;
Guest.NodeByteSize = sizeof(LOGON);
{
//Multi-user. Initialize the DriveMapTable
int i;
for ( i = 0; i < DRIVE_MAP_TABLE_SIZE; i ++ )
Guest.DriveMapTable[i] = NULL;
}
RtlInitUnicodeString( &Guest.ServerName, NULL );
RtlInitUnicodeString( &Guest.PassWord, NULL );
RtlInitUnicodeString( &Guest.UserName, L"GUEST" );
Guest.UserUid = DefaultLuid;
InitializeListHead( &Guest.NdsCredentialList );
InsertTailList( &LogonList, &Guest.Next );
//
// Initialize the global VCB list.
//
InitializeListHead( &GlobalVcbList );
CurrentVcbEntry = 1;
//
// Initialize the Get message queue.
//
InitializeListHead( &NwGetMessageList );
KeInitializeSpinLock( &NwMessageSpinLock );
//
// Initialize the Pending lock queue.
//
InitializeListHead( &NwPendingLockList );
KeInitializeSpinLock( &NwPendingLockSpinLock );
//
// Insert the Permanent SCB in the global list of SCBs.
//
InsertHeadList( &ScbQueue, &NwPermanentNpScb.ScbLinks );
//
// Initialize the Kernel Queue Object. Only one thread has to
// be concurrently active.
//
KeInitializeQueue( &KernelQueue, 1 );
//
// Spawn off our own worker thread which will service reroute and
// reconnect attempts.
//
SpawnWorkerThread();
#if 0
// HACKHACK
InitializeListHead( &FnList );
#endif
//
// Seed the random number generator.
//
KeQuerySystemTime( &Now );
srand( Now.LowPart );
RtlZeroMemory( &Stats, sizeof( NW_REDIR_STATISTICS ) );
ExInitializeResourceLite( &NwUnlockableCodeResource );
NwSectionDescriptor.Base = BurstReadTimeout;
NwSectionDescriptor.Handle = 0;
NwSectionDescriptor.ReferenceCount = 0;
return;
}