mirror of
https://github.com/reactos/reactos.git
synced 2024-10-31 20:02:55 +00:00
879b1f5754
- replaced DWORD with ULONG in a couple of places - replaced some ULONGs with LONGs in the KD GDB stub - replaced INITIAL_TEB with USER_STACK, as per Nebbet's book, to support both fixed size and expandable stacks - added InterlockedExchangePointer - added the ASM_BREAKPOINT macro as the architecture-dependent assembler code to raise a breakpoint exception - corrected definitions of INT, LONG, DWORD, UINT, ULONG and ULONG32 - corrected IoSetCancelRoutine to use InterlockedExchangePointer - corrected definition of NtCurrentTeb and NtCurrentPeb - corrected DbgBreakPoint and DbgUserBreakPoint not to set up a stack frame (temporary fix with inline assembler - why doesn't GCC understand __declspec(naked)?) - corrected various calls to Interlocked* functions to cast OUT operands to LONG * - corrected various printf format strings - corrected DbgUiIssueRemoteBreakin to use the smallest possible stack (this is what started everything) - removed a DPRINT that accessed pageable memory at non-PASSIVE_LEVEL IRQL - beautified CreateProcessA (another temporary fix - all the new functions will be isolated in the upcoming stand-alone RTL) - prefixed LdrInitializeThunk with a nop that can be overwritten with a breakpoint for debugging purposes (temporary debugging aid until we have user-mode debugger support). Will add support for this to the breakin utility soon - thread creation code rewritten from scratch (some glitches documented inline, but works fine) - thread creation code now duplicated just twice, as opposed to five times (temporary fix - three new, non standard functions have been exported from NTDLL.DLL, will fix later) svn path=/trunk/; revision=4595
256 lines
5.8 KiB
C
256 lines
5.8 KiB
C
/* $Id: pageop.c,v 1.16 2003/04/26 23:13:32 hyperion Exp $
|
|
*
|
|
* COPYRIGHT: See COPYING in the top level directory
|
|
* PROJECT: ReactOS kernel
|
|
* FILE: ntoskrnl/mm/pageop.c
|
|
* PROGRAMMER: David Welch (welch@cwcom.net)
|
|
* UPDATE HISTORY:
|
|
* 27/05/98: Created
|
|
*/
|
|
|
|
/* INCLUDES ****************************************************************/
|
|
|
|
#include <ddk/ntddk.h>
|
|
#include <internal/ps.h>
|
|
#include <internal/mm.h>
|
|
#include <internal/pool.h>
|
|
|
|
#define NDEBUG
|
|
#include <internal/debug.h>
|
|
|
|
/* GLOBALS *******************************************************************/
|
|
|
|
#define PAGEOP_HASH_TABLE_SIZE (32)
|
|
|
|
static KSPIN_LOCK MmPageOpHashTableLock;
|
|
static PMM_PAGEOP MmPageOpHashTable[PAGEOP_HASH_TABLE_SIZE];
|
|
static NPAGED_LOOKASIDE_LIST MmPageOpLookasideList;
|
|
|
|
#define TAG_MM_PAGEOP TAG('M', 'P', 'O', 'P')
|
|
|
|
/* FUNCTIONS *****************************************************************/
|
|
|
|
VOID
|
|
MmReleasePageOp(PMM_PAGEOP PageOp)
|
|
/*
|
|
* FUNCTION: Release a reference to a page operation descriptor
|
|
*/
|
|
{
|
|
KIRQL oldIrql;
|
|
PMM_PAGEOP PrevPageOp;
|
|
|
|
KeAcquireSpinLock(&MmPageOpHashTableLock, &oldIrql);
|
|
PageOp->ReferenceCount--;
|
|
if (PageOp->ReferenceCount > 0)
|
|
{
|
|
KeReleaseSpinLock(&MmPageOpHashTableLock, oldIrql);
|
|
return;
|
|
}
|
|
InterlockedDecrement((LONG *)&PageOp->MArea->PageOpCount);
|
|
PrevPageOp = MmPageOpHashTable[PageOp->Hash];
|
|
if (PrevPageOp == PageOp)
|
|
{
|
|
MmPageOpHashTable[PageOp->Hash] = PageOp->Next;
|
|
KeReleaseSpinLock(&MmPageOpHashTableLock, oldIrql);
|
|
ExFreeToNPagedLookasideList(&MmPageOpLookasideList, PageOp);
|
|
return;
|
|
}
|
|
while (PrevPageOp->Next != NULL)
|
|
{
|
|
if (PrevPageOp->Next == PageOp)
|
|
{
|
|
PrevPageOp->Next = PageOp->Next;
|
|
KeReleaseSpinLock(&MmPageOpHashTableLock, oldIrql);
|
|
ExFreeToNPagedLookasideList(&MmPageOpLookasideList, PageOp);
|
|
return;
|
|
}
|
|
PrevPageOp = PrevPageOp->Next;
|
|
}
|
|
KeReleaseSpinLock(&MmPageOpHashTableLock, oldIrql);
|
|
KeBugCheck(0);
|
|
}
|
|
|
|
PMM_PAGEOP
|
|
MmCheckForPageOp(PMEMORY_AREA MArea, ULONG Pid, PVOID Address,
|
|
PMM_SECTION_SEGMENT Segment, ULONG Offset)
|
|
{
|
|
ULONG Hash;
|
|
KIRQL oldIrql;
|
|
PMM_PAGEOP PageOp;
|
|
|
|
/*
|
|
* Calcuate the hash value for pageop structure
|
|
*/
|
|
if (MArea->Type == MEMORY_AREA_SECTION_VIEW)
|
|
{
|
|
Hash = (((ULONG)Segment) | (((ULONG)Offset) / PAGE_SIZE));
|
|
}
|
|
else
|
|
{
|
|
Hash = (((ULONG)Pid) | (((ULONG)Address) / PAGE_SIZE));
|
|
}
|
|
Hash = Hash % PAGEOP_HASH_TABLE_SIZE;
|
|
|
|
KeAcquireSpinLock(&MmPageOpHashTableLock, &oldIrql);
|
|
|
|
/*
|
|
* Check for an existing pageop structure
|
|
*/
|
|
PageOp = MmPageOpHashTable[Hash];
|
|
while (PageOp != NULL)
|
|
{
|
|
if (MArea->Type == MEMORY_AREA_SECTION_VIEW)
|
|
{
|
|
if (PageOp->Segment == Segment &&
|
|
PageOp->Offset == Offset)
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (PageOp->Pid == Pid &&
|
|
PageOp->Address == Address)
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
PageOp = PageOp->Next;
|
|
}
|
|
|
|
/*
|
|
* If we found an existing pageop then increment the reference count
|
|
* and return it.
|
|
*/
|
|
if (PageOp != NULL)
|
|
{
|
|
PageOp->ReferenceCount++;
|
|
KeReleaseSpinLock(&MmPageOpHashTableLock, oldIrql);
|
|
return(PageOp);
|
|
}
|
|
KeReleaseSpinLock(&MmPageOpHashTableLock, oldIrql);
|
|
return(NULL);
|
|
}
|
|
|
|
PMM_PAGEOP
|
|
MmGetPageOp(PMEMORY_AREA MArea, ULONG Pid, PVOID Address,
|
|
PMM_SECTION_SEGMENT Segment, ULONG Offset, ULONG OpType)
|
|
/*
|
|
* FUNCTION: Get a page operation descriptor corresponding to
|
|
* the memory area and either the segment, offset pair or the
|
|
* pid, address pair.
|
|
*/
|
|
{
|
|
ULONG Hash;
|
|
KIRQL oldIrql;
|
|
PMM_PAGEOP PageOp;
|
|
|
|
/*
|
|
* Calcuate the hash value for pageop structure
|
|
*/
|
|
if (MArea->Type == MEMORY_AREA_SECTION_VIEW)
|
|
{
|
|
Hash = (((ULONG)Segment) | (((ULONG)Offset) / PAGE_SIZE));
|
|
}
|
|
else
|
|
{
|
|
Hash = (((ULONG)Pid) | (((ULONG)Address) / PAGE_SIZE));
|
|
}
|
|
Hash = Hash % PAGEOP_HASH_TABLE_SIZE;
|
|
|
|
KeAcquireSpinLock(&MmPageOpHashTableLock, &oldIrql);
|
|
|
|
/*
|
|
* Check for an existing pageop structure
|
|
*/
|
|
PageOp = MmPageOpHashTable[Hash];
|
|
while (PageOp != NULL)
|
|
{
|
|
if (MArea->Type == MEMORY_AREA_SECTION_VIEW)
|
|
{
|
|
if (PageOp->Segment == Segment &&
|
|
PageOp->Offset == Offset)
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (PageOp->Pid == Pid &&
|
|
PageOp->Address == Address)
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
PageOp = PageOp->Next;
|
|
}
|
|
|
|
/*
|
|
* If we found an existing pageop then increment the reference count
|
|
* and return it.
|
|
*/
|
|
if (PageOp != NULL)
|
|
{
|
|
PageOp->ReferenceCount++;
|
|
KeReleaseSpinLock(&MmPageOpHashTableLock, oldIrql);
|
|
return(PageOp);
|
|
}
|
|
|
|
/*
|
|
* Otherwise add a new pageop.
|
|
*/
|
|
PageOp = ExAllocateFromNPagedLookasideList(&MmPageOpLookasideList);
|
|
if (PageOp == NULL)
|
|
{
|
|
KeReleaseSpinLock(&MmPageOpHashTableLock, oldIrql);
|
|
return(NULL);
|
|
}
|
|
|
|
if (MArea->Type != MEMORY_AREA_SECTION_VIEW)
|
|
{
|
|
PageOp->Pid = Pid;
|
|
PageOp->Address = Address;
|
|
}
|
|
else
|
|
{
|
|
PageOp->Segment = Segment;
|
|
PageOp->Offset = Offset;
|
|
}
|
|
PageOp->ReferenceCount = 1;
|
|
PageOp->Next = MmPageOpHashTable[Hash];
|
|
PageOp->Hash = Hash;
|
|
PageOp->Thread = PsGetCurrentThread();
|
|
PageOp->Abandoned = FALSE;
|
|
PageOp->Status = STATUS_PENDING;
|
|
PageOp->OpType = OpType;
|
|
PageOp->MArea = MArea;
|
|
KeInitializeEvent(&PageOp->CompletionEvent, NotificationEvent, FALSE);
|
|
MmPageOpHashTable[Hash] = PageOp;
|
|
InterlockedIncrement((LONG *)&MArea->PageOpCount);
|
|
|
|
KeReleaseSpinLock(&MmPageOpHashTableLock, oldIrql);
|
|
return(PageOp);
|
|
}
|
|
|
|
VOID
|
|
MmInitializePageOp(VOID)
|
|
{
|
|
memset(MmPageOpHashTable, 0, sizeof(MmPageOpHashTable));
|
|
KeInitializeSpinLock(&MmPageOpHashTableLock);
|
|
|
|
ExInitializeNPagedLookasideList (&MmPageOpLookasideList,
|
|
NULL,
|
|
NULL,
|
|
0,
|
|
sizeof(MM_PAGEOP),
|
|
TAG_MM_PAGEOP,
|
|
50);
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|