NT4/private/ntos/rdr/disccode.c
2020-09-30 17:12:29 +02:00

562 lines
13 KiB
C
Raw Blame History

This file contains invisible Unicode characters

This file contains invisible Unicode characters that are indistinguishable to humans but may be processed differently by a computer. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

/*++
Copyright (c) 1993 Microsoft Corporation
Module Name:
disccode.c
Abstract:
This module contains the code to manage the NT redirectors discardable
code sections.
Author:
Larry Osterman (larryo) 12-Nov-1993
Environment:
Kernel mode.
Revision History:
12-Nov-1993
Created
--*/
//
// Include modules
//
#include "precomp.h"
#pragma hdrstop
BOOLEAN DiscCodeInitialized = FALSE;
VOID
RdrDiscardableCodeRoutine(
IN PVOID Context
);
#ifdef ALLOC_PRAGMA
#pragma alloc_text(PAGE, RdrReferenceDiscardableCode)
#pragma alloc_text(PAGE, RdrDereferenceDiscardableCode)
#pragma alloc_text(PAGE, RdrDiscardableCodeRoutine)
#pragma alloc_text(INIT, RdrInitializeDiscardableCode)
#pragma alloc_text(PAGE, RdrUninitializeDiscardableCode)
#endif
//
// These 7 variables maintain the state needed to manage the redirector
// discardable code section.
//
// The redirector discardable code section is referenced via a call to
// RdrReferenceDiscardableCodeSection, and dereferenced via a call to
// RdrDereferenceDiscardableCodeSection.
//
// If the discardable code section is already mapped into memory, then
// referencing the discardable code section is extremely quick.
//
// When the reference count on the discardable code section drops to 0, a
// timer is set that will actually perform the work needed to uninitalize the
// section. This means that if the reference count goes from 0 to 1 to 0
// frequently, we won't thrash inside MmLockPagableCodeSection.
//
ERESOURCE
RdrDiscardableCodeLock = {0};
ULONG
RdrDiscardableCodeTimeout = 10;
RDR_SECTION
RdrSectionInfo[RdrMaxDiscardableSection] = {0};
extern
PVOID
BowserAllocateViewBuffer(VOID);
extern
VOID
BowserNetlogonTrimMessageQueue(VOID);
VOID
RdrReferenceDiscardableCode(
DISCARDABLE_SECTION_NAME SectionName
)
/*++
Routine Description:
RdrReferenceDiscardableCode is called to reference the redirectors
discardable code section.
If the section is not present in memory, MmLockPagableCodeSection is
called to fault the section into memory.
Arguments:
None.
Return Value:
None.
--*/
{
#if DBG
PVOID caller, callersCaller;
#endif
PRDR_SECTION Section = &RdrSectionInfo[SectionName];
PAGED_CODE();
ExAcquireResourceExclusive(&RdrDiscardableCodeLock, TRUE);
ASSERT( DiscCodeInitialized );
#if DBG
RtlGetCallersAddress(&caller, &callersCaller);
dprintf(DPRT_DISCCODE, (" RdrReferenceDiscardableCode: %ld: Caller: %lx, Callers Caller: %lx\n", SectionName, caller, callersCaller));
#endif
//
// If the reference count is already non zero, just increment it and
// return.
//
if (Section->ReferenceCount) {
Section->ReferenceCount += 1;
dprintf(DPRT_DISCCODE, (" RdrReferenceDiscardableCode: %d: Early out, Refcount now %ld\n", SectionName, Section->ReferenceCount));
//
// Wait for the pages to be faulted in.
//
ExReleaseResource(&RdrDiscardableCodeLock);
return;
}
Section->ReferenceCount += 1;
//
// Cancel the timer, if it is running, we won't be discarding the code
// at this time.
//
// If the cancel timer fails, this is not a problem, since we will be
// bumping a reference count in the MmLockPagableCodeSection, so when
// the timer actually runs and the call to MmUnlockPagableImageSection
// is called, we will simply unlock it.
//
if (Section->Timer != NULL) {
Section->TimerCancelled = TRUE;
if (KeCancelTimer(Section->Timer)) {
//
// Free the timer and DPC, they aren't going to fire anymore.
//
FREE_POOL(Section->Timer);
Section->Timer = NULL;
//
// Set the active event to the signalled state, since we're
// done canceling the timer.
//
KeSetEvent(&Section->TimerDoneEvent, 0, FALSE);
} else {
//
// The timer was active, and we weren't able to cancel it.
// But we marked it for cancellation, and the timer routine
// will recognize this and leave the section locked.
//
}
}
//
// If the discardable code section is still locked, then we're done,
// and we can return right away.
//
if (Section->Locked) {
dprintf(DPRT_DISCCODE, (" RdrReferenceDiscardableCode: %d: Already locked, Refcount now %ld\n", SectionName, Section->ReferenceCount));
ExReleaseResource(&RdrDiscardableCodeLock);
return;
}
ASSERT (Section->CodeHandle == NULL);
ASSERT (Section->DataHandle == NULL);
//
// Lock down the pagable image section.
//
dprintf(DPRT_DISCCODE, (" RdrReferenceDiscardableCode: %d: Lock, Refcount now %ld\n", SectionName, Section->ReferenceCount));
if (Section->CodeBase != NULL) {
Section->CodeHandle = MmLockPagableCodeSection(Section->CodeBase);
ASSERT (Section->CodeHandle != NULL);
}
if (Section->DataBase != NULL) {
Section->DataHandle = MmLockPagableDataSection(Section->DataBase);
ASSERT (Section->DataHandle != NULL);
}
Section->Locked = TRUE;
ExReleaseResource(&RdrDiscardableCodeLock);
}
VOID
RdrDiscardableCodeDpcRoutine(
IN PKDPC Dpc,
IN PVOID Context,
IN PVOID SystemArgument1,
IN PVOID SystemArgument2
)
/*++
Routine Description:
This routine is called when the timeout expires. It is called at Dpc level
to queue a WorkItem to a system worker thread.
Arguments:
IN PKDPC Dpc,
IN PVOID Context,
IN PVOID SystemArgument1,
IN PVOID SystemArgument2
Return Value
None.
--*/
{
PWORK_QUEUE_ITEM discardableWorkItem = Context;
ExQueueWorkItem(discardableWorkItem, CriticalWorkQueue);
UNREFERENCED_PARAMETER(Dpc);
UNREFERENCED_PARAMETER(SystemArgument1);
UNREFERENCED_PARAMETER(SystemArgument2);
}
VOID
RdrDiscardableCodeRoutine(
IN PVOID Context
)
/*++
Routine Description:
RdrDiscardableCodeRoutine is called at task time after the redirector
discardable code timer has fired to actually perform the unlock on the
discardable code section.
Arguments:
Context - Ignored.
Return Value:
None.
--*/
{
PRDR_SECTION Section = Context;
PAGED_CODE();
ExAcquireResourceExclusive(&RdrDiscardableCodeLock, TRUE);
if (Section->TimerCancelled) {
//
// The timer was cancelled after it was scheduled to run.
// Don't unlock the section.
//
} else if (Section->Locked) {
//
// The timer was not cancelled. Unlock the section.
//
Section->Locked = FALSE;
ASSERT (Section->CodeHandle != NULL ||
Section->DataHandle != NULL);
dprintf(DPRT_DISCCODE, ("RDR: Unlock %x\n", Section));
if (Section->CodeHandle != NULL) {
MmUnlockPagableImageSection(Section->CodeHandle);
Section->CodeHandle = NULL;
}
if (Section->DataHandle != NULL) {
MmUnlockPagableImageSection(Section->DataHandle);
Section->DataHandle = NULL;
}
}
//
// Free the timer and DPC, they aren't going to fire anymore.
//
FREE_POOL(Section->Timer);
Section->Timer = NULL;
ExReleaseResource(&RdrDiscardableCodeLock);
KeSetEvent(&Section->TimerDoneEvent, 0, FALSE);
}
VOID
RdrDereferenceDiscardableCode(
DISCARDABLE_SECTION_NAME SectionName
)
/*++
Routine Description:
RdrDereferenceDiscardableCode is called to dereference the redirectors
discardable code section.
When the reference count drops to 0, a timer is set that will fire in <n>
seconds, after which time the section will be unlocked.
Arguments:
None.
Return Value:
None.
--*/
{
#if DBG
PVOID caller, callersCaller;
#endif
PRDR_SECTION Section = &RdrSectionInfo[SectionName];
LARGE_INTEGER discardableCodeTimeout;
PKTIMER Timer;
PKDPC Dpc;
PWORK_QUEUE_ITEM WorkItem;
PAGED_CODE();
ExAcquireResourceExclusive(&RdrDiscardableCodeLock, TRUE);
ASSERT( DiscCodeInitialized );
#if DBG
RtlGetCallersAddress(&caller, &callersCaller);
dprintf(DPRT_DISCCODE, ("RdrDereferenceDiscardableCode: %ld: Caller: %lx, Callers Caller: %lx\n", SectionName, caller, callersCaller));
#endif
ASSERT (Section->ReferenceCount > 0);
//
// If the reference count is above 1, just decrement it and
// return.
//
Section->ReferenceCount -= 1;
if (Section->ReferenceCount) {
dprintf(DPRT_DISCCODE, ("RdrDereferenceDiscardableCode: %d: Early out, Refcount now %ld\n", SectionName, Section->ReferenceCount));
ExReleaseResource(&RdrDiscardableCodeLock);
return;
}
//
// If the discardable code timer is still active (which might happen if
// the RdrReferenceDiscardableCode failed to cancel the timer), we just
// want to bail out and let the timer do the work. It means that we
// discard the code sooner, but that shouldn't be that big a deal.
//
if (Section->Timer != NULL) {
ExReleaseResource(&RdrDiscardableCodeLock);
return;
}
//
// The reference count just went to 0, set a timer to fire in
// RdrDiscardableCodeTimeout seconds. When the timer fires,
// we queue a request to a worker thread and it will lock down
// the pagable code.
//
ASSERT (Section->Timer == NULL);
Timer = ALLOCATE_POOL(NonPagedPool,
sizeof(KTIMER) + sizeof(KDPC) + sizeof(WORK_QUEUE_ITEM),
POOL_DISCTIMER);
if (Timer == NULL) {
ExReleaseResource(&RdrDiscardableCodeLock);
return;
}
Section->Timer = Timer;
KeInitializeTimer(Timer);
Dpc = (PKDPC)(Timer + 1);
WorkItem = (PWORK_QUEUE_ITEM)(Dpc + 1);
KeClearEvent(&Section->TimerDoneEvent);
Section->TimerCancelled = FALSE;
ExInitializeWorkItem(WorkItem, RdrDiscardableCodeRoutine, Section);
KeInitializeDpc(Dpc, RdrDiscardableCodeDpcRoutine, WorkItem);
discardableCodeTimeout.QuadPart = Int32x32To64(RdrDiscardableCodeTimeout, 1000 * -10000);
KeSetTimer(Timer, discardableCodeTimeout, Dpc);
dprintf(DPRT_DISCCODE, ("RdrDereferenceDiscardableCode: %d: Set timer, Refcount now %ld\n", SectionName, Section->ReferenceCount));
ExReleaseResource(&RdrDiscardableCodeLock);
}
VOID
RdrInitializeDiscardableCode(
VOID
)
{
DISCARDABLE_SECTION_NAME SectionName;
PRDR_SECTION Section;
for (SectionName = 0, Section = &RdrSectionInfo[0];
SectionName < RdrMaxDiscardableSection;
SectionName += 1, Section++ ) {
KeInitializeEvent(&Section->TimerDoneEvent,
NotificationEvent,
TRUE);
}
RdrSectionInfo[RdrFileDiscardableSection].CodeBase = RdrBackOff;
RdrSectionInfo[RdrFileDiscardableSection].DataBase = NULL;
RdrSectionInfo[RdrVCDiscardableSection].CodeBase = RdrTdiDisconnectHandler;
RdrSectionInfo[RdrVCDiscardableSection].DataBase = RdrSmbErrorMap;
RdrSectionInfo[RdrConnectionDiscardableSection].CodeBase = RdrReferenceServer;
RdrSectionInfo[RdrConnectionDiscardableSection].DataBase = NULL;
RdrSectionInfo[BowserDiscardableCodeSection].CodeBase = BowserAllocateViewBuffer;
RdrSectionInfo[BowserDiscardableCodeSection].DataBase = NULL;
RdrSectionInfo[BowserNetlogonDiscardableCodeSection].CodeBase = BowserNetlogonTrimMessageQueue;
RdrSectionInfo[BowserNetlogonDiscardableCodeSection].DataBase = NULL;
ExInitializeResource(&RdrDiscardableCodeLock);
DiscCodeInitialized = TRUE;
}
VOID
RdrUninitializeDiscardableCode(
VOID
)
{
DISCARDABLE_SECTION_NAME SectionName;
PRDR_SECTION Section;
PAGED_CODE();
ExAcquireResourceExclusive(&RdrDiscardableCodeLock, TRUE);
DiscCodeInitialized = FALSE;
for (SectionName = 0, Section = &RdrSectionInfo[0];
SectionName < RdrMaxDiscardableSection;
SectionName += 1, Section++ ) {
//
// Cancel the timer if it is running.
//
if (Section->Timer != NULL) {
if (!KeCancelTimer(Section->Timer)) {
//
// The timer was active, and we weren't able to cancel it,
// wait until the timer finishes firing.
//
ExReleaseResource(&RdrDiscardableCodeLock);
KeWaitForSingleObject(&Section->TimerDoneEvent,
Executive, KernelMode, FALSE, NULL);
ExAcquireResourceExclusive(&RdrDiscardableCodeLock, TRUE);
} else {
FREE_POOL(Section->Timer);
Section->Timer = NULL;
}
}
if (Section->Locked) {
//
// Unlock the section.
//
Section->Locked = FALSE;
ASSERT (Section->CodeHandle != NULL ||
Section->DataHandle != NULL);
dprintf(DPRT_DISCCODE, ("RDR: Uninitialize unlock %x\n", Section));
if (Section->CodeHandle != NULL) {
MmUnlockPagableImageSection(Section->CodeHandle);
Section->CodeHandle = NULL;
}
if (Section->DataHandle != NULL) {
MmUnlockPagableImageSection(Section->DataHandle);
Section->DataHandle = NULL;
}
}
}
ExReleaseResource(&RdrDiscardableCodeLock);
ExDeleteResource(&RdrDiscardableCodeLock);
}