mirror of
https://github.com/reactos/reactos.git
synced 2025-01-04 05:20:54 +00:00
694 lines
20 KiB
C
694 lines
20 KiB
C
/*
|
|
* PROJECT: ReactOS Kernel
|
|
* LICENSE: BSD - See COPYING.ARM in the top level directory
|
|
* FILE: ntoskrnl/mm/ARM3/special.c
|
|
* PURPOSE: ARM Memory Manager Special Pool implementation
|
|
* PROGRAMMERS: ReactOS Portable Systems Group
|
|
*/
|
|
|
|
/*
|
|
References:
|
|
http://msdn.microsoft.com/en-us/library/ff551832(v=VS.85).aspx
|
|
*/
|
|
|
|
/* INCLUDES *******************************************************************/
|
|
|
|
#include <ntoskrnl.h>
|
|
#define NDEBUG
|
|
#include <debug.h>
|
|
|
|
#define MODULE_INVOLVED_IN_ARM3
|
|
#include <mm/ARM3/miarm.h>
|
|
|
|
extern ULONG ExpPoolFlags;
|
|
extern PMMPTE MmSystemPteBase;
|
|
|
|
PMMPTE
|
|
NTAPI
|
|
MiReserveAlignedSystemPtes(IN ULONG NumberOfPtes,
|
|
IN MMSYSTEM_PTE_POOL_TYPE SystemPtePoolType,
|
|
IN ULONG Alignment);
|
|
|
|
/* GLOBALS ********************************************************************/
|
|
|
|
#define SPECIAL_POOL_PAGED_PTE 0x2000
|
|
#define SPECIAL_POOL_NONPAGED_PTE 0x4000
|
|
#define SPECIAL_POOL_PAGED 0x8000
|
|
|
|
PVOID MmSpecialPoolStart;
|
|
PVOID MmSpecialPoolEnd;
|
|
PVOID MiSpecialPoolExtra;
|
|
ULONG MiSpecialPoolExtraCount;
|
|
|
|
PMMPTE MiSpecialPoolFirstPte;
|
|
PMMPTE MiSpecialPoolLastPte;
|
|
|
|
PFN_COUNT MmSpecialPagesInUse;
|
|
PFN_COUNT MmSpecialPagesInUsePeak;
|
|
PFN_COUNT MiSpecialPagesPagable;
|
|
PFN_COUNT MiSpecialPagesPagablePeak;
|
|
PFN_COUNT MiSpecialPagesNonPaged;
|
|
PFN_COUNT MiSpecialPagesNonPagedPeak;
|
|
PFN_COUNT MiSpecialPagesNonPagedMaximum;
|
|
|
|
BOOLEAN MmSpecialPoolCatchOverruns = TRUE;
|
|
|
|
typedef struct _MI_FREED_SPECIAL_POOL
|
|
{
|
|
POOL_HEADER OverlaidPoolHeader;
|
|
/* TODO: Add overlaid verifier pool header */
|
|
ULONG Signature;
|
|
ULONG TickCount;
|
|
ULONG NumberOfBytesRequested;
|
|
BOOLEAN Pagable;
|
|
PVOID VirtualAddress;
|
|
PVOID StackPointer;
|
|
ULONG StackBytes;
|
|
PETHREAD Thread;
|
|
UCHAR StackData[0x400];
|
|
} MI_FREED_SPECIAL_POOL, *PMI_FREED_SPECIAL_POOL;
|
|
|
|
/* PRIVATE FUNCTIONS **********************************************************/
|
|
|
|
VOID NTAPI MiTestSpecialPool(VOID);
|
|
|
|
BOOLEAN
|
|
NTAPI
|
|
MmUseSpecialPool(SIZE_T NumberOfBytes, ULONG Tag)
|
|
{
|
|
/* Special pool is not suitable for allocations bigger than 1 page */
|
|
if (NumberOfBytes > (PAGE_SIZE - sizeof(POOL_HEADER)))
|
|
{
|
|
return FALSE;
|
|
}
|
|
|
|
if (MmSpecialPoolTag == '*')
|
|
{
|
|
return TRUE;
|
|
}
|
|
|
|
return Tag == MmSpecialPoolTag;
|
|
}
|
|
|
|
BOOLEAN
|
|
NTAPI
|
|
MmIsSpecialPoolAddress(PVOID P)
|
|
{
|
|
return ((P >= MmSpecialPoolStart) &&
|
|
(P <= MmSpecialPoolEnd));
|
|
}
|
|
|
|
BOOLEAN
|
|
NTAPI
|
|
MmIsSpecialPoolAddressFree(PVOID P)
|
|
{
|
|
PMMPTE PointerPte;
|
|
|
|
ASSERT(MmIsSpecialPoolAddress(P));
|
|
PointerPte = MiAddressToPte(P);
|
|
|
|
if (PointerPte->u.Soft.PageFileHigh == SPECIAL_POOL_PAGED_PTE ||
|
|
PointerPte->u.Soft.PageFileHigh == SPECIAL_POOL_NONPAGED_PTE)
|
|
{
|
|
/* Guard page PTE */
|
|
return FALSE;
|
|
}
|
|
|
|
/* Free PTE */
|
|
return TRUE;
|
|
}
|
|
|
|
VOID
|
|
NTAPI
|
|
MiInitializeSpecialPool(VOID)
|
|
{
|
|
ULONG SpecialPoolPtes, i;
|
|
PMMPTE PointerPte;
|
|
|
|
/* Check if there is a special pool tag */
|
|
if ((MmSpecialPoolTag == 0) ||
|
|
(MmSpecialPoolTag == -1)) return;
|
|
|
|
/* Calculate number of system PTEs for the special pool */
|
|
if (MmNumberOfSystemPtes >= 0x3000)
|
|
SpecialPoolPtes = MmNumberOfSystemPtes / 3;
|
|
else
|
|
SpecialPoolPtes = MmNumberOfSystemPtes / 6;
|
|
|
|
/* Don't let the number go too high */
|
|
if (SpecialPoolPtes > 0x6000) SpecialPoolPtes = 0x6000;
|
|
|
|
/* Round up to the page size */
|
|
SpecialPoolPtes = PAGE_ROUND_UP(SpecialPoolPtes);
|
|
|
|
ASSERT((SpecialPoolPtes & (PTE_PER_PAGE - 1)) == 0);
|
|
|
|
/* Reserve those PTEs */
|
|
do
|
|
{
|
|
PointerPte = MiReserveAlignedSystemPtes(SpecialPoolPtes,
|
|
SystemPteSpace,
|
|
/*0x400000*/0); // FIXME:
|
|
if (PointerPte) break;
|
|
|
|
/* Reserving didn't work, so try to reduce the requested size */
|
|
ASSERT(SpecialPoolPtes >= PTE_PER_PAGE);
|
|
SpecialPoolPtes -= PTE_PER_PAGE;
|
|
} while (SpecialPoolPtes);
|
|
|
|
/* Fail if we couldn't reserve them at all */
|
|
if (!SpecialPoolPtes) return;
|
|
|
|
/* Make sure we got enough */
|
|
ASSERT(SpecialPoolPtes >= PTE_PER_PAGE);
|
|
|
|
/* Save first PTE and its address */
|
|
MiSpecialPoolFirstPte = PointerPte;
|
|
MmSpecialPoolStart = MiPteToAddress(PointerPte);
|
|
|
|
for (i = 0; i < PTE_PER_PAGE / 2; i++)
|
|
{
|
|
/* Point it to the next entry */
|
|
PointerPte->u.List.NextEntry = &PointerPte[2] - MmSystemPteBase;
|
|
|
|
/* Move to the next pair */
|
|
PointerPte += 2;
|
|
}
|
|
|
|
/* Save extra values */
|
|
MiSpecialPoolExtra = PointerPte;
|
|
MiSpecialPoolExtraCount = SpecialPoolPtes - PTE_PER_PAGE;
|
|
|
|
/* Mark the previous PTE as the last one */
|
|
MiSpecialPoolLastPte = PointerPte - 2;
|
|
MiSpecialPoolLastPte->u.List.NextEntry = MM_EMPTY_PTE_LIST;
|
|
|
|
/* Save end address of the special pool */
|
|
MmSpecialPoolEnd = MiPteToAddress(MiSpecialPoolLastPte + 1);
|
|
|
|
/* Calculate maximum non-paged part of the special pool */
|
|
MiSpecialPagesNonPagedMaximum = MmResidentAvailablePages >> 4;
|
|
|
|
/* And limit it if it turned out to be too big */
|
|
if (MmNumberOfPhysicalPages > 0x3FFF)
|
|
MiSpecialPagesNonPagedMaximum = MmResidentAvailablePages >> 3;
|
|
|
|
DPRINT1("Special pool start %p - end %p\n", MmSpecialPoolStart, MmSpecialPoolEnd);
|
|
ExpPoolFlags |= POOL_FLAG_SPECIAL_POOL;
|
|
|
|
//MiTestSpecialPool();
|
|
}
|
|
|
|
NTSTATUS
|
|
NTAPI
|
|
MmExpandSpecialPool(VOID)
|
|
{
|
|
ULONG i;
|
|
PMMPTE PointerPte;
|
|
|
|
MI_ASSERT_PFN_LOCK_HELD();
|
|
|
|
if (MiSpecialPoolExtraCount == 0)
|
|
return STATUS_INSUFFICIENT_RESOURCES;
|
|
|
|
PointerPte = MiSpecialPoolExtra;
|
|
ASSERT(MiSpecialPoolFirstPte == MiSpecialPoolLastPte);
|
|
ASSERT(MiSpecialPoolFirstPte->u.List.NextEntry == MM_EMPTY_PTE_LIST);
|
|
MiSpecialPoolFirstPte->u.List.NextEntry = PointerPte - MmSystemPteBase;
|
|
|
|
ASSERT(MiSpecialPoolExtraCount >= PTE_PER_PAGE);
|
|
for (i = 0; i < PTE_PER_PAGE / 2; i++)
|
|
{
|
|
/* Point it to the next entry */
|
|
PointerPte->u.List.NextEntry = &PointerPte[2] - MmSystemPteBase;
|
|
|
|
/* Move to the next pair */
|
|
PointerPte += 2;
|
|
}
|
|
|
|
/* Save remaining extra values */
|
|
MiSpecialPoolExtra = PointerPte;
|
|
MiSpecialPoolExtraCount -= PTE_PER_PAGE;
|
|
|
|
/* Mark the previous PTE as the last one */
|
|
MiSpecialPoolLastPte = PointerPte - 2;
|
|
MiSpecialPoolLastPte->u.List.NextEntry = MM_EMPTY_PTE_LIST;
|
|
|
|
/* Save new end address of the special pool */
|
|
MmSpecialPoolEnd = MiPteToAddress(MiSpecialPoolLastPte + 1);
|
|
|
|
return STATUS_SUCCESS;
|
|
}
|
|
|
|
PVOID
|
|
NTAPI
|
|
MmAllocateSpecialPool(SIZE_T NumberOfBytes, ULONG Tag, POOL_TYPE PoolType, ULONG SpecialType)
|
|
{
|
|
KIRQL Irql;
|
|
MMPTE TempPte = ValidKernelPte;
|
|
PMMPTE PointerPte;
|
|
PFN_NUMBER PageFrameNumber;
|
|
LARGE_INTEGER TickCount;
|
|
PVOID Entry;
|
|
PPOOL_HEADER Header;
|
|
PFN_COUNT PagesInUse;
|
|
|
|
DPRINT("MmAllocateSpecialPool(%x %x %x %x)\n", NumberOfBytes, Tag, PoolType, SpecialType);
|
|
|
|
/* Check if the pool is initialized and quit if it's not */
|
|
if (!MiSpecialPoolFirstPte) return NULL;
|
|
|
|
/* Get the pool type */
|
|
PoolType &= BASE_POOL_TYPE_MASK;
|
|
|
|
/* Check whether current IRQL matches the pool type */
|
|
Irql = KeGetCurrentIrql();
|
|
|
|
if (((PoolType == PagedPool) && (Irql > APC_LEVEL)) ||
|
|
((PoolType != PagedPool) && (Irql > DISPATCH_LEVEL)))
|
|
{
|
|
/* Bad caller */
|
|
KeBugCheckEx(SPECIAL_POOL_DETECTED_MEMORY_CORRUPTION,
|
|
Irql,
|
|
PoolType,
|
|
NumberOfBytes,
|
|
0x30);
|
|
}
|
|
|
|
/* Some allocations from Mm must never use special pool */
|
|
if (Tag == 'tSmM')
|
|
{
|
|
/* Reject and let normal pool handle it */
|
|
return NULL;
|
|
}
|
|
|
|
/* TODO: Take into account various limitations */
|
|
|
|
/* Heed the maximum limit of nonpaged pages */
|
|
if ((PoolType == NonPagedPool) &&
|
|
(MiSpecialPagesNonPaged > MiSpecialPagesNonPagedMaximum))
|
|
{
|
|
return NULL;
|
|
}
|
|
|
|
/* Lock PFN database */
|
|
Irql = MiAcquirePfnLock();
|
|
|
|
/* Reject allocation in case amount of available pages is too small */
|
|
if (MmAvailablePages < 0x100)
|
|
{
|
|
/* Release the PFN database lock */
|
|
MiReleasePfnLock(Irql);
|
|
DPRINT1("Special pool: MmAvailablePages 0x%x is too small\n", MmAvailablePages);
|
|
return NULL;
|
|
}
|
|
|
|
/* Check if special pool PTE list is exhausted */
|
|
if (MiSpecialPoolFirstPte->u.List.NextEntry == MM_EMPTY_PTE_LIST)
|
|
{
|
|
/* Try to expand it */
|
|
if (!NT_SUCCESS(MmExpandSpecialPool()))
|
|
{
|
|
/* No reserves left, reject this allocation */
|
|
static int once;
|
|
MiReleasePfnLock(Irql);
|
|
if (!once++) DPRINT1("Special pool: No PTEs left!\n");
|
|
return NULL;
|
|
}
|
|
ASSERT(MiSpecialPoolFirstPte->u.List.NextEntry != MM_EMPTY_PTE_LIST);
|
|
}
|
|
|
|
/* Save allocation time */
|
|
KeQueryTickCount(&TickCount);
|
|
|
|
/* Get a pointer to the first PTE */
|
|
PointerPte = MiSpecialPoolFirstPte;
|
|
|
|
/* Set the first PTE pointer to the next one in the list */
|
|
MiSpecialPoolFirstPte = MmSystemPteBase + PointerPte->u.List.NextEntry;
|
|
|
|
/* Allocate a physical page */
|
|
if (PoolType == PagedPool)
|
|
{
|
|
MI_SET_USAGE(MI_USAGE_PAGED_POOL);
|
|
}
|
|
else
|
|
{
|
|
MI_SET_USAGE(MI_USAGE_NONPAGED_POOL);
|
|
}
|
|
MI_SET_PROCESS2("Kernel-Special");
|
|
PageFrameNumber = MiRemoveAnyPage(MI_GET_NEXT_COLOR());
|
|
|
|
/* Initialize PFN and make it valid */
|
|
TempPte.u.Hard.PageFrameNumber = PageFrameNumber;
|
|
MiInitializePfnAndMakePteValid(PageFrameNumber, PointerPte, TempPte);
|
|
|
|
/* Release the PFN database lock */
|
|
MiReleasePfnLock(Irql);
|
|
|
|
/* Increase page counter */
|
|
PagesInUse = InterlockedIncrementUL(&MmSpecialPagesInUse);
|
|
if (PagesInUse > MmSpecialPagesInUsePeak)
|
|
MmSpecialPagesInUsePeak = PagesInUse;
|
|
|
|
/* Put some content into the page. Low value of tick count would do */
|
|
Entry = MiPteToAddress(PointerPte);
|
|
RtlFillMemory(Entry, PAGE_SIZE, TickCount.LowPart);
|
|
|
|
/* Calculate header and entry addresses */
|
|
if ((SpecialType != 0) &&
|
|
((SpecialType == 1) || (!MmSpecialPoolCatchOverruns)))
|
|
{
|
|
/* We catch underruns. Data is at the beginning of the page */
|
|
Header = (PPOOL_HEADER)((PUCHAR)Entry + PAGE_SIZE - sizeof(POOL_HEADER));
|
|
}
|
|
else
|
|
{
|
|
/* We catch overruns. Data is at the end of the page */
|
|
Header = (PPOOL_HEADER)Entry;
|
|
Entry = (PVOID)((ULONG_PTR)((PUCHAR)Entry - NumberOfBytes + PAGE_SIZE) & ~((LONG_PTR)sizeof(POOL_HEADER) - 1));
|
|
}
|
|
|
|
/* Initialize the header */
|
|
RtlZeroMemory(Header, sizeof(POOL_HEADER));
|
|
|
|
/* Save allocation size there */
|
|
Header->Ulong1 = (ULONG)NumberOfBytes;
|
|
|
|
/* Make sure it's all good */
|
|
ASSERT((NumberOfBytes <= PAGE_SIZE - sizeof(POOL_HEADER)) &&
|
|
(PAGE_SIZE <= 32 * 1024));
|
|
|
|
/* Mark it as paged or nonpaged */
|
|
if (PoolType == PagedPool)
|
|
{
|
|
/* Add pagedpool flag into the pool header too */
|
|
Header->Ulong1 |= SPECIAL_POOL_PAGED;
|
|
|
|
/* Also mark the next PTE as special-pool-paged */
|
|
PointerPte[1].u.Soft.PageFileHigh |= SPECIAL_POOL_PAGED_PTE;
|
|
|
|
/* Increase pagable counter */
|
|
PagesInUse = InterlockedIncrementUL(&MiSpecialPagesPagable);
|
|
if (PagesInUse > MiSpecialPagesPagablePeak)
|
|
MiSpecialPagesPagablePeak = PagesInUse;
|
|
}
|
|
else
|
|
{
|
|
/* Mark the next PTE as special-pool-nonpaged */
|
|
PointerPte[1].u.Soft.PageFileHigh |= SPECIAL_POOL_NONPAGED_PTE;
|
|
|
|
/* Increase nonpaged counter */
|
|
PagesInUse = InterlockedIncrementUL(&MiSpecialPagesNonPaged);
|
|
if (PagesInUse > MiSpecialPagesNonPagedPeak)
|
|
MiSpecialPagesNonPagedPeak = PagesInUse;
|
|
}
|
|
|
|
/* Finally save tag and put allocation time into the header's blocksize.
|
|
That time will be used to check memory consistency within the allocated
|
|
page. */
|
|
Header->PoolTag = Tag;
|
|
Header->BlockSize = (UCHAR)TickCount.LowPart;
|
|
DPRINT("%p\n", Entry);
|
|
return Entry;
|
|
}
|
|
|
|
VOID
|
|
NTAPI
|
|
MiSpecialPoolCheckPattern(PUCHAR P, PPOOL_HEADER Header)
|
|
{
|
|
ULONG BytesToCheck, BytesRequested, Index;
|
|
PUCHAR Ptr;
|
|
|
|
/* Get amount of bytes user requested to be allocated by clearing out the paged mask */
|
|
BytesRequested = (Header->Ulong1 & ~SPECIAL_POOL_PAGED) & 0xFFFF;
|
|
ASSERT(BytesRequested <= PAGE_SIZE - sizeof(POOL_HEADER));
|
|
|
|
/* Get a pointer to the end of user's area */
|
|
Ptr = P + BytesRequested;
|
|
|
|
/* Calculate how many bytes to check */
|
|
BytesToCheck = (ULONG)((PUCHAR)PAGE_ALIGN(P) + PAGE_SIZE - Ptr);
|
|
|
|
/* Remove pool header size if we're catching underruns */
|
|
if (((ULONG_PTR)P & (PAGE_SIZE - 1)) == 0)
|
|
{
|
|
/* User buffer is located in the beginning of the page */
|
|
BytesToCheck -= sizeof(POOL_HEADER);
|
|
}
|
|
|
|
/* Check the pattern after user buffer */
|
|
for (Index = 0; Index < BytesToCheck; Index++)
|
|
{
|
|
/* Bugcheck if bytes don't match */
|
|
if (Ptr[Index] != Header->BlockSize)
|
|
{
|
|
KeBugCheckEx(SPECIAL_POOL_DETECTED_MEMORY_CORRUPTION,
|
|
(ULONG_PTR)P,
|
|
(ULONG_PTR)&Ptr[Index],
|
|
Header->BlockSize,
|
|
0x24);
|
|
}
|
|
}
|
|
}
|
|
|
|
VOID
|
|
NTAPI
|
|
MmFreeSpecialPool(PVOID P)
|
|
{
|
|
PMMPTE PointerPte;
|
|
PPOOL_HEADER Header;
|
|
BOOLEAN Overruns = FALSE;
|
|
KIRQL Irql = KeGetCurrentIrql();
|
|
POOL_TYPE PoolType;
|
|
ULONG BytesRequested, BytesReal = 0;
|
|
ULONG PtrOffset;
|
|
PUCHAR b;
|
|
PMI_FREED_SPECIAL_POOL FreedHeader;
|
|
LARGE_INTEGER TickCount;
|
|
PMMPFN Pfn;
|
|
|
|
DPRINT("MmFreeSpecialPool(%p)\n", P);
|
|
|
|
/* Get the PTE */
|
|
PointerPte = MiAddressToPte(P);
|
|
|
|
/* Check if it's valid */
|
|
if (PointerPte->u.Hard.Valid == 0)
|
|
{
|
|
/* Bugcheck if it has NOACCESS or 0 set as protection */
|
|
if (PointerPte->u.Soft.Protection == MM_NOACCESS ||
|
|
!PointerPte->u.Soft.Protection)
|
|
{
|
|
KeBugCheckEx(SPECIAL_POOL_DETECTED_MEMORY_CORRUPTION,
|
|
(ULONG_PTR)P,
|
|
(ULONG_PTR)PointerPte,
|
|
0,
|
|
0x20);
|
|
}
|
|
}
|
|
|
|
/* Determine if it's a underruns or overruns pool pointer */
|
|
PtrOffset = (ULONG)((ULONG_PTR)P & (PAGE_SIZE - 1));
|
|
if (PtrOffset)
|
|
{
|
|
/* Pool catches overruns */
|
|
Header = PAGE_ALIGN(P);
|
|
Overruns = TRUE;
|
|
}
|
|
else
|
|
{
|
|
/* Pool catches underruns */
|
|
Header = (PPOOL_HEADER)((PUCHAR)PAGE_ALIGN(P) + PAGE_SIZE - sizeof(POOL_HEADER));
|
|
}
|
|
|
|
/* Check if it's non paged pool */
|
|
if ((Header->Ulong1 & SPECIAL_POOL_PAGED) == 0)
|
|
{
|
|
/* Non-paged allocation, ensure that IRQ is not higher that DISPATCH */
|
|
PoolType = NonPagedPool;
|
|
ASSERT(PointerPte[1].u.Soft.PageFileHigh == SPECIAL_POOL_NONPAGED_PTE);
|
|
if (Irql > DISPATCH_LEVEL)
|
|
{
|
|
KeBugCheckEx(SPECIAL_POOL_DETECTED_MEMORY_CORRUPTION,
|
|
Irql,
|
|
PoolType,
|
|
(ULONG_PTR)P,
|
|
0x31);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
/* Paged allocation, ensure */
|
|
PoolType = PagedPool;
|
|
ASSERT(PointerPte[1].u.Soft.PageFileHigh == SPECIAL_POOL_PAGED_PTE);
|
|
if (Irql > APC_LEVEL)
|
|
{
|
|
KeBugCheckEx(SPECIAL_POOL_DETECTED_MEMORY_CORRUPTION,
|
|
Irql,
|
|
PoolType,
|
|
(ULONG_PTR)P,
|
|
0x31);
|
|
}
|
|
}
|
|
|
|
/* Get amount of bytes user requested to be allocated by clearing out the paged mask */
|
|
BytesRequested = (Header->Ulong1 & ~SPECIAL_POOL_PAGED) & 0xFFFF;
|
|
ASSERT(BytesRequested <= PAGE_SIZE - sizeof(POOL_HEADER));
|
|
|
|
/* Check memory before the allocated user buffer in case of overruns detection */
|
|
if (Overruns)
|
|
{
|
|
/* Calculate the real placement of the buffer */
|
|
BytesReal = PAGE_SIZE - PtrOffset;
|
|
|
|
/* If they mismatch, it's unrecoverable */
|
|
if (BytesRequested > BytesReal)
|
|
{
|
|
KeBugCheckEx(SPECIAL_POOL_DETECTED_MEMORY_CORRUPTION,
|
|
(ULONG_PTR)P,
|
|
BytesRequested,
|
|
BytesReal,
|
|
0x21);
|
|
}
|
|
|
|
if (BytesRequested + sizeof(POOL_HEADER) < BytesReal)
|
|
{
|
|
KeBugCheckEx(SPECIAL_POOL_DETECTED_MEMORY_CORRUPTION,
|
|
(ULONG_PTR)P,
|
|
BytesRequested,
|
|
BytesReal,
|
|
0x22);
|
|
}
|
|
|
|
/* Actually check the memory pattern */
|
|
for (b = (PUCHAR)(Header + 1); b < (PUCHAR)P; b++)
|
|
{
|
|
if (*b != Header->BlockSize)
|
|
{
|
|
/* Bytes mismatch */
|
|
KeBugCheckEx(SPECIAL_POOL_DETECTED_MEMORY_CORRUPTION,
|
|
(ULONG_PTR)P,
|
|
(ULONG_PTR)b,
|
|
Header->BlockSize,
|
|
0x23);
|
|
}
|
|
}
|
|
}
|
|
|
|
/* Check the memory pattern after the user buffer */
|
|
MiSpecialPoolCheckPattern(P, Header);
|
|
|
|
/* Fill the freed header */
|
|
KeQueryTickCount(&TickCount);
|
|
FreedHeader = (PMI_FREED_SPECIAL_POOL)PAGE_ALIGN(P);
|
|
FreedHeader->Signature = 0x98764321;
|
|
FreedHeader->TickCount = TickCount.LowPart;
|
|
FreedHeader->NumberOfBytesRequested = BytesRequested;
|
|
FreedHeader->Pagable = PoolType;
|
|
FreedHeader->VirtualAddress = P;
|
|
FreedHeader->Thread = PsGetCurrentThread();
|
|
/* TODO: Fill StackPointer and StackBytes */
|
|
FreedHeader->StackPointer = NULL;
|
|
FreedHeader->StackBytes = 0;
|
|
|
|
if (PoolType == NonPagedPool)
|
|
{
|
|
/* Non pagable. Get PFN element corresponding to the PTE */
|
|
Pfn = MI_PFN_ELEMENT(PointerPte->u.Hard.PageFrameNumber);
|
|
|
|
/* Count the page as free */
|
|
InterlockedDecrementUL(&MiSpecialPagesNonPaged);
|
|
|
|
/* Lock PFN database */
|
|
Irql = MiAcquirePfnLock();
|
|
|
|
/* Delete this PFN */
|
|
MI_SET_PFN_DELETED(Pfn);
|
|
|
|
/* Decrement share count of this PFN */
|
|
MiDecrementShareCount(Pfn, PointerPte->u.Hard.PageFrameNumber);
|
|
|
|
MI_ERASE_PTE(PointerPte);
|
|
|
|
/* Flush the TLB */
|
|
//FIXME: Use KeFlushSingleTb() instead
|
|
KeFlushEntireTb(TRUE, TRUE);
|
|
}
|
|
else
|
|
{
|
|
/* Pagable. Delete that virtual address */
|
|
MiDeleteSystemPageableVm(PointerPte, 1, 0, NULL);
|
|
|
|
/* Count the page as free */
|
|
InterlockedDecrementUL(&MiSpecialPagesPagable);
|
|
|
|
/* Lock PFN database */
|
|
Irql = MiAcquirePfnLock();
|
|
}
|
|
|
|
/* Mark next PTE as invalid */
|
|
MI_ERASE_PTE(PointerPte + 1);
|
|
|
|
/* Make sure that the last entry is really the last one */
|
|
ASSERT(MiSpecialPoolLastPte->u.List.NextEntry == MM_EMPTY_PTE_LIST);
|
|
|
|
/* Update the current last PTE next pointer */
|
|
MiSpecialPoolLastPte->u.List.NextEntry = PointerPte - MmSystemPteBase;
|
|
|
|
/* PointerPte becomes the new last PTE */
|
|
PointerPte->u.List.NextEntry = MM_EMPTY_PTE_LIST;
|
|
MiSpecialPoolLastPte = PointerPte;
|
|
|
|
/* Release the PFN database lock */
|
|
MiReleasePfnLock(Irql);
|
|
|
|
/* Update page counter */
|
|
InterlockedDecrementUL(&MmSpecialPagesInUse);
|
|
}
|
|
|
|
VOID
|
|
NTAPI
|
|
MiTestSpecialPool(VOID)
|
|
{
|
|
ULONG i;
|
|
PVOID p1, p2[100];
|
|
//PUCHAR p3;
|
|
ULONG ByteSize;
|
|
POOL_TYPE PoolType = PagedPool;
|
|
|
|
// First allocate/free
|
|
for (i=0; i<100; i++)
|
|
{
|
|
ByteSize = (100 * (i+1)) % (PAGE_SIZE - sizeof(POOL_HEADER));
|
|
p1 = MmAllocateSpecialPool(ByteSize, 'TEST', PoolType, 0);
|
|
DPRINT1("p1 %p size %lu\n", p1, ByteSize);
|
|
MmFreeSpecialPool(p1);
|
|
}
|
|
|
|
// Now allocate all at once, then free at once
|
|
for (i=0; i<100; i++)
|
|
{
|
|
ByteSize = (100 * (i+1)) % (PAGE_SIZE - sizeof(POOL_HEADER));
|
|
p2[i] = MmAllocateSpecialPool(ByteSize, 'TEST', PoolType, 0);
|
|
DPRINT1("p2[%lu] %p size %lu\n", i, p1, ByteSize);
|
|
}
|
|
for (i=0; i<100; i++)
|
|
{
|
|
DPRINT1("Freeing %p\n", p2[i]);
|
|
MmFreeSpecialPool(p2[i]);
|
|
}
|
|
|
|
// Overrun the buffer to test
|
|
//ByteSize = 16;
|
|
//p3 = MmAllocateSpecialPool(ByteSize, 'TEST', NonPagedPool, 0);
|
|
//p3[ByteSize] = 0xF1; // This should cause an exception
|
|
|
|
// Underrun the buffer to test
|
|
//p3 = MmAllocateSpecialPool(ByteSize, 'TEST', NonPagedPool, 1);
|
|
//p3--;
|
|
//*p3 = 0xF1; // This should cause an exception
|
|
|
|
}
|
|
|
|
/* EOF */
|