378 lines
9.0 KiB
C++
378 lines
9.0 KiB
C++
/*++
|
||
|
||
Copyright (C) Microsoft Corporation, 1991 - 1999
|
||
|
||
Module Name:
|
||
|
||
initnt.cxx
|
||
|
||
Abstract:
|
||
|
||
This module contains the code used to initialize the RPC runtime. One
|
||
routine gets called when a process attaches to the dll. Another routine
|
||
gets called the first time an RPC API is called.
|
||
|
||
Author:
|
||
|
||
Michael Montague (mikemon) 03-May-1991
|
||
|
||
Revision History:
|
||
Kamen Moutafov (KamenM) Dec 99 - Feb 2000 - Support for cell debugging stuff
|
||
|
||
--*/
|
||
|
||
#include <precomp.hxx>
|
||
#include <hndlsvr.hxx>
|
||
#include <thrdctx.hxx>
|
||
#include <rpccfg.h>
|
||
#include <rc4.h>
|
||
#include <randlib.h>
|
||
#include <epmap.h>
|
||
#include <CellHeap.hxx>
|
||
|
||
#include <lpcpack.hxx>
|
||
|
||
int RpcHasBeenInitialized = 0;
|
||
RTL_CRITICAL_SECTION GlobalMutex;
|
||
|
||
RPC_SERVER * GlobalRpcServer;
|
||
BOOL g_fClientSideDebugInfoEnabled = FALSE;
|
||
BOOL g_fServerSideDebugInfoEnabled = FALSE;
|
||
BOOL g_fSendEEInfo = FALSE;
|
||
LRPC_SERVER *GlobalLrpcServer = NULL;
|
||
HINSTANCE hInstanceDLL ;
|
||
|
||
EXTERN_C HINSTANCE g_hRpcrt4;
|
||
|
||
DWORD gPageSize;
|
||
DWORD gThreadTimeout;
|
||
UINT gNumberOfProcessors;
|
||
DWORD gAllocationGranularity;
|
||
BOOL gfServerPlatform;
|
||
ULONGLONG gPhysicalMemorySize; // in megabytes
|
||
|
||
//
|
||
// By default the non pipe arguments cannot be more than 4 Megs
|
||
//
|
||
DWORD gMaxRpcSize = 0x400000;
|
||
DWORD gProrateStart = 0;
|
||
DWORD gProrateMax = 0;
|
||
DWORD gProrateFactor = 0;
|
||
void *g_rc4SafeCtx = 0;
|
||
|
||
extern "C" {
|
||
|
||
BOOLEAN
|
||
InitializeDLL (
|
||
IN HINSTANCE DllHandle,
|
||
IN ULONG Reason,
|
||
IN PCONTEXT Context OPTIONAL
|
||
)
|
||
/*++
|
||
|
||
Routine Description:
|
||
|
||
This routine will get called: when a process attaches to this dll, and
|
||
when a process detaches from this dll.
|
||
|
||
Return Value:
|
||
|
||
TRUE - Initialization successfully occurred.
|
||
|
||
FALSE - Insufficient memory is available for the process to attach to
|
||
this dll.
|
||
|
||
--*/
|
||
{
|
||
NTSTATUS NtStatus;
|
||
|
||
UNUSED(Context);
|
||
|
||
switch (Reason)
|
||
{
|
||
case DLL_PROCESS_ATTACH:
|
||
hInstanceDLL = DllHandle ;
|
||
g_hRpcrt4 = DllHandle;
|
||
|
||
GlobalMutex.DebugInfo = NULL;
|
||
NtStatus = RtlInitializeCriticalSectionAndSpinCount(&GlobalMutex, PREALLOCATE_EVENT_MASK);
|
||
if (NT_SUCCESS(NtStatus) == 0)
|
||
{
|
||
return(FALSE);
|
||
}
|
||
|
||
// initialize safe rc4 operations.
|
||
if(!rc4_safe_startup( &g_rc4SafeCtx ))
|
||
{
|
||
(void) RtlDeleteCriticalSection(&GlobalMutex);
|
||
return FALSE;
|
||
}
|
||
break;
|
||
|
||
case DLL_PROCESS_DETACH:
|
||
//
|
||
// If shutting down because of a FreeLibrary call, cleanup
|
||
//
|
||
|
||
if (Context == NULL)
|
||
{
|
||
ShutdownLrpcClient();
|
||
}
|
||
|
||
if (GlobalMutex.DebugInfo != NULL)
|
||
{
|
||
NtStatus = RtlDeleteCriticalSection(&GlobalMutex);
|
||
ASSERT(NT_SUCCESS(NtStatus));
|
||
}
|
||
|
||
if (g_rc4SafeCtx)
|
||
rc4_safe_shutdown( g_rc4SafeCtx ); // free safe rc4 resources.
|
||
|
||
break;
|
||
|
||
case DLL_THREAD_DETACH:
|
||
THREAD * Thread = RpcpGetThreadPointer();
|
||
|
||
#ifdef RPC_OLD_IO_PROTECTION
|
||
if (Thread)
|
||
{
|
||
Thread->UnprotectThread();
|
||
}
|
||
#else
|
||
delete Thread;
|
||
#endif
|
||
|
||
break;
|
||
}
|
||
|
||
return(TRUE);
|
||
}
|
||
|
||
} //extern "C" end
|
||
|
||
#ifdef NO_RECURSIVE_MUTEXES
|
||
unsigned int RecursionCount = 0;
|
||
#endif // NO_RECURSIVE_MUTEXES
|
||
|
||
extern int InitializeRpcAllocator(void);
|
||
extern RPC_STATUS ReadPolicySettings(void);
|
||
|
||
const ULONG MEGABYTE = 0x100000;
|
||
|
||
typedef struct tagBasicSystemInfo
|
||
{
|
||
DWORD m_dwPageSize;
|
||
ULONGLONG m_dwPhysicalMemorySize;
|
||
DWORD m_dwNumberOfProcessors;
|
||
ULONG AllocationGranularity;
|
||
BOOL m_fServerPlatform;
|
||
} BasicSystemInfo;
|
||
|
||
BOOL
|
||
GetBasicSystemInfo (
|
||
IN OUT BasicSystemInfo *basicSystemInfo
|
||
)
|
||
/*++
|
||
|
||
Routine Description:
|
||
|
||
Gets basic system information. We don't use the Win32 GetSystemInfo, because
|
||
under NT it accesses the image header, which may not be available if the image
|
||
was loaded from the network, and the network failed. Therefore, we need a function
|
||
that accesses just what we need, and nothing else.
|
||
|
||
Arguments:
|
||
|
||
The basic system info structure.
|
||
|
||
Return Value:
|
||
|
||
0 - failure
|
||
|
||
non-0 - success.
|
||
|
||
--*/
|
||
{
|
||
//
|
||
// Query system info (for # of processors) and product type
|
||
//
|
||
|
||
SYSTEM_BASIC_INFORMATION BasicInfo;
|
||
NTSTATUS Status;
|
||
BOOL b;
|
||
|
||
Status = NtQuerySystemInformation(
|
||
SystemBasicInformation,
|
||
&BasicInfo,
|
||
sizeof(BasicInfo),
|
||
NULL
|
||
);
|
||
if ( !NT_SUCCESS(Status) )
|
||
{
|
||
DbgPrintEx(DPFLTR_RPCPROXY_ID,
|
||
DPFLTR_ERROR_LEVEL,
|
||
"RPCTRANS: NtQuerySystemInformation failed: %x\n",
|
||
Status);
|
||
|
||
return 0;
|
||
}
|
||
|
||
basicSystemInfo->m_dwPageSize = BasicInfo.PageSize;
|
||
basicSystemInfo->m_dwNumberOfProcessors = BasicInfo.NumberOfProcessors;
|
||
basicSystemInfo->m_dwPhysicalMemorySize = ((BasicInfo.NumberOfPhysicalPages * (ULONGLONG) basicSystemInfo->m_dwPageSize) / MEGABYTE);
|
||
basicSystemInfo->AllocationGranularity = BasicInfo.AllocationGranularity;
|
||
|
||
NT_PRODUCT_TYPE type;
|
||
b = RtlGetNtProductType(&type);
|
||
if (b)
|
||
{
|
||
basicSystemInfo->m_fServerPlatform = (type != NtProductWinNt);
|
||
return 1;
|
||
}
|
||
else
|
||
{
|
||
DbgPrintEx(DPFLTR_RPCPROXY_ID,
|
||
DPFLTR_ERROR_LEVEL,
|
||
"RpcGetNtProductType failed, usign default\n");
|
||
return 0;
|
||
}
|
||
}
|
||
|
||
|
||
RPC_STATUS
|
||
PerformRpcInitialization (
|
||
void
|
||
)
|
||
/*++
|
||
|
||
Routine Description:
|
||
|
||
This routine will get called the first time that an RPC runtime API is
|
||
called. There is actually a race condition, which we prevent by grabbing
|
||
a mutex and then performing the initialization. We only want to
|
||
initialize once.
|
||
|
||
Return Value:
|
||
|
||
RPC_S_OK - This status code indicates that the runtime has been correctly
|
||
initialized and is ready to go.
|
||
|
||
RPC_S_OUT_OF_MEMORY - If initialization failed, it is most likely due to
|
||
insufficient memory being available.
|
||
|
||
--*/
|
||
{
|
||
if ( RpcHasBeenInitialized == 0 )
|
||
{
|
||
RequestGlobalMutex();
|
||
if ( RpcHasBeenInitialized == 0 )
|
||
{
|
||
RPC_STATUS Status;
|
||
BasicSystemInfo SystemInfo;
|
||
BOOL b;
|
||
|
||
b = GetBasicSystemInfo(&SystemInfo);
|
||
if (!b)
|
||
{
|
||
ClearGlobalMutex();
|
||
return RPC_S_OUT_OF_MEMORY;
|
||
}
|
||
|
||
gNumberOfProcessors = SystemInfo.m_dwNumberOfProcessors;
|
||
gPageSize = SystemInfo.m_dwPageSize;
|
||
gAllocationGranularity = SystemInfo.AllocationGranularity;
|
||
gfServerPlatform = SystemInfo.m_fServerPlatform;
|
||
gPhysicalMemorySize = SystemInfo.m_dwPhysicalMemorySize;
|
||
|
||
// Should be something like 64kb / 4kb.
|
||
ASSERT(gAllocationGranularity % gPageSize == 0);
|
||
|
||
if (( InitializeRpcAllocator() != 0)
|
||
|| ( InitializeServerDLL() != 0 ))
|
||
{
|
||
ClearGlobalMutex();
|
||
return(RPC_S_OUT_OF_MEMORY);
|
||
}
|
||
|
||
Status = InitializeEPMapperClient();
|
||
if (Status != RPC_S_OK)
|
||
{
|
||
ClearGlobalMutex();
|
||
return Status;
|
||
}
|
||
|
||
Status = ReadPolicySettings();
|
||
if (Status != RPC_S_OK)
|
||
{
|
||
ClearGlobalMutex();
|
||
return Status;
|
||
}
|
||
|
||
if (gfServerPlatform)
|
||
{
|
||
gThreadTimeout = 90*1000;
|
||
}
|
||
else
|
||
{
|
||
gThreadTimeout = 30*1000;
|
||
}
|
||
|
||
Status = InitializeCellHeap();
|
||
if (Status != RPC_S_OK)
|
||
{
|
||
ClearGlobalMutex();
|
||
return Status;
|
||
}
|
||
|
||
RpcHasBeenInitialized = 1;
|
||
ClearGlobalMutex();
|
||
|
||
if (LoadLibrary(RPC_CONST_SSTRING("rpcrt4.dll")) == 0)
|
||
{
|
||
return RPC_S_OUT_OF_MEMORY;
|
||
}
|
||
}
|
||
else
|
||
{
|
||
ClearGlobalMutex();
|
||
}
|
||
}
|
||
return(RPC_S_OK);
|
||
}
|
||
|
||
#ifdef DBG
|
||
long lGlobalMutexCount = 0;
|
||
#endif
|
||
|
||
|
||
void
|
||
GlobalMutexRequestExternal (
|
||
void
|
||
)
|
||
/*++
|
||
|
||
Routine Description:
|
||
|
||
Request the global mutex.
|
||
|
||
--*/
|
||
{
|
||
GlobalMutexRequest();
|
||
}
|
||
|
||
|
||
void
|
||
GlobalMutexClearExternal (
|
||
void
|
||
)
|
||
/*++
|
||
|
||
Routine Description:
|
||
|
||
Clear the global mutex.
|
||
|
||
--*/
|
||
{
|
||
GlobalMutexClear();
|
||
}
|