1459 lines
45 KiB
C
1459 lines
45 KiB
C
|
/*++
|
|||
|
|
|||
|
Copyright (c) 1992 Microsoft Corporation
|
|||
|
|
|||
|
Module Name:
|
|||
|
|
|||
|
heappriv.h
|
|||
|
|
|||
|
Abstract:
|
|||
|
|
|||
|
Private include file used by heap allocator (heap.c, heapdll.c and
|
|||
|
heapdbg.c)
|
|||
|
|
|||
|
Author:
|
|||
|
|
|||
|
Steve Wood (stevewo) 25-Oct-1994
|
|||
|
|
|||
|
Revision History:
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
#ifndef _RTL_HEAP_PRIVATE_
|
|||
|
#define _RTL_HEAP_PRIVATE_
|
|||
|
|
|||
|
#include "heappage.h"
|
|||
|
|
|||
|
//
|
|||
|
// In private builds (PRERELEASE = 1) we allow using the new low fragmentation heap
|
|||
|
// for processes that set the DisableLookaside registry key. The main purpose is to
|
|||
|
// allow testing the new heap API.
|
|||
|
//
|
|||
|
|
|||
|
#ifndef PRERELEASE
|
|||
|
|
|||
|
#define DISABLE_REGISTRY_TEST_HOOKS
|
|||
|
|
|||
|
#endif
|
|||
|
|
|||
|
//
|
|||
|
// Disable FPO optimization so even retail builds get somewhat reasonable
|
|||
|
// stack backtraces
|
|||
|
//
|
|||
|
|
|||
|
#if i386
|
|||
|
// #pragma optimize("y",off)
|
|||
|
#endif
|
|||
|
|
|||
|
#if DBG
|
|||
|
#define HEAPASSERT(exp) if (!(exp)) RtlAssert( #exp, __FILE__, __LINE__, NULL )
|
|||
|
#else
|
|||
|
#define HEAPASSERT(exp)
|
|||
|
#endif
|
|||
|
|
|||
|
//
|
|||
|
// Define Minimum lookaside list depth.
|
|||
|
//
|
|||
|
|
|||
|
#define MINIMUM_LOOKASIDE_DEPTH 4
|
|||
|
|
|||
|
//
|
|||
|
// This variable contains the fill pattern used for heap tail checking
|
|||
|
//
|
|||
|
|
|||
|
extern const UCHAR CheckHeapFillPattern[ CHECK_HEAP_TAIL_SIZE ];
|
|||
|
|
|||
|
|
|||
|
//
|
|||
|
// Here are the locking routines for the heap (kernel and user)
|
|||
|
//
|
|||
|
|
|||
|
#ifdef NTOS_KERNEL_RUNTIME
|
|||
|
|
|||
|
//
|
|||
|
// Kernel mode heap uses the kernel resource package for locking
|
|||
|
//
|
|||
|
|
|||
|
#define RtlInitializeLockRoutine(L) ExInitializeResourceLite((PERESOURCE)(L))
|
|||
|
#define RtlAcquireLockRoutine(L) ExAcquireResourceExclusiveLite((PERESOURCE)(L),TRUE)
|
|||
|
#define RtlReleaseLockRoutine(L) ExReleaseResourceLite((PERESOURCE)(L))
|
|||
|
#define RtlDeleteLockRoutine(L) ExDeleteResourceLite((PERESOURCE)(L))
|
|||
|
#define RtlOkayToLockRoutine(L) ExOkayToLockRoutineLite((PERESOURCE)(L))
|
|||
|
|
|||
|
#else // #ifdef NTOS_KERNEL_ROUTINE
|
|||
|
|
|||
|
//
|
|||
|
// User mode heap uses the critical section package for locking
|
|||
|
//
|
|||
|
|
|||
|
#ifndef PREALLOCATE_EVENT_MASK
|
|||
|
|
|||
|
#define PREALLOCATE_EVENT_MASK 0x80000000 // Defined only in dll\resource.c
|
|||
|
|
|||
|
#endif // PREALLOCATE_EVENT_MASK
|
|||
|
|
|||
|
#define RtlInitializeLockRoutine(L) RtlInitializeCriticalSectionAndSpinCount((PRTL_CRITICAL_SECTION)(L),(PREALLOCATE_EVENT_MASK | 4000))
|
|||
|
#define RtlAcquireLockRoutine(L) RtlEnterCriticalSection((PRTL_CRITICAL_SECTION)(L))
|
|||
|
#define RtlReleaseLockRoutine(L) RtlLeaveCriticalSection((PRTL_CRITICAL_SECTION)(L))
|
|||
|
#define RtlDeleteLockRoutine(L) RtlDeleteCriticalSection((PRTL_CRITICAL_SECTION)(L))
|
|||
|
#define RtlOkayToLockRoutine(L) NtdllOkayToLockRoutine((PVOID)(L))
|
|||
|
|
|||
|
#endif // #ifdef NTOS_KERNEL_RUNTIME
|
|||
|
|
|||
|
|
|||
|
//
|
|||
|
// Here are some debugging macros for the heap
|
|||
|
//
|
|||
|
|
|||
|
#ifdef NTOS_KERNEL_RUNTIME
|
|||
|
|
|||
|
#define HEAP_DEBUG_FLAGS 0
|
|||
|
#define DEBUG_HEAP(F) FALSE
|
|||
|
#define SET_LAST_STATUS(S) NOTHING;
|
|||
|
|
|||
|
#else // #ifdef NTOS_KERNEL_ROUTINE
|
|||
|
|
|||
|
#define HEAP_DEBUG_FLAGS (HEAP_VALIDATE_PARAMETERS_ENABLED | \
|
|||
|
HEAP_VALIDATE_ALL_ENABLED | \
|
|||
|
HEAP_CAPTURE_STACK_BACKTRACES | \
|
|||
|
HEAP_CREATE_ENABLE_TRACING | \
|
|||
|
HEAP_FLAG_PAGE_ALLOCS)
|
|||
|
#define DEBUG_HEAP(F) ((F & HEAP_DEBUG_FLAGS) && !(F & HEAP_SKIP_VALIDATION_CHECKS))
|
|||
|
#define SET_LAST_STATUS(S) {NtCurrentTeb()->LastErrorValue = RtlNtStatusToDosError( NtCurrentTeb()->LastStatusValue = (ULONG)(S) );}
|
|||
|
|
|||
|
#endif // #ifdef NTOS_KERNEL_RUNTIME
|
|||
|
|
|||
|
|
|||
|
//
|
|||
|
// Here are the macros used for debug printing and breakpoints
|
|||
|
//
|
|||
|
|
|||
|
#ifdef NTOS_KERNEL_RUNTIME
|
|||
|
|
|||
|
#define HeapDebugPrint( _x_ ) {DbgPrint _x_;}
|
|||
|
|
|||
|
#define HeapDebugBreak( _x_ ) {if (KdDebuggerEnabled) DbgBreakPoint();}
|
|||
|
|
|||
|
#else // #ifdef NTOS_KERNEL_ROUTINE
|
|||
|
|
|||
|
#define HeapDebugPrint( _x_ ) \
|
|||
|
{ \
|
|||
|
PLIST_ENTRY _Module; \
|
|||
|
PLDR_DATA_TABLE_ENTRY _Entry; \
|
|||
|
\
|
|||
|
_Module = NtCurrentPeb()->Ldr->InLoadOrderModuleList.Flink; \
|
|||
|
_Entry = CONTAINING_RECORD( _Module, \
|
|||
|
LDR_DATA_TABLE_ENTRY, \
|
|||
|
InLoadOrderLinks); \
|
|||
|
DbgPrint("HEAP[%wZ]: ", &_Entry->BaseDllName); \
|
|||
|
DbgPrint _x_; \
|
|||
|
}
|
|||
|
|
|||
|
#define HeapDebugBreak( _x_ ) \
|
|||
|
{ \
|
|||
|
VOID RtlpBreakPointHeap( PVOID BadAddress ); \
|
|||
|
\
|
|||
|
RtlpBreakPointHeap( (_x_) ); \
|
|||
|
}
|
|||
|
|
|||
|
#endif // #ifdef NTOS_KERNEL_RUNTIME
|
|||
|
|
|||
|
//
|
|||
|
// Virtual memory hook for virtual alloc functions
|
|||
|
//
|
|||
|
|
|||
|
#ifdef NTOS_KERNEL_RUNTIME
|
|||
|
|
|||
|
#define RtlpHeapFreeVirtualMemory(P,A,S,F) \
|
|||
|
ZwFreeVirtualMemory(P,A,S,F)
|
|||
|
|
|||
|
#else // NTOS_KERNEL_RUNTIME
|
|||
|
|
|||
|
//
|
|||
|
// The user mode call needs to call the secmem virtual free
|
|||
|
// as well to update the memory counters per heap
|
|||
|
//
|
|||
|
|
|||
|
#define RtlpHeapFreeVirtualMemory(P,A,S,F) \
|
|||
|
RtlpSecMemFreeVirtualMemory(P,A,S,F)
|
|||
|
|
|||
|
#endif // NTOS_KERNEL_RUNTIME
|
|||
|
|
|||
|
|
|||
|
ULONG
|
|||
|
RtlpHeapExceptionFilter (
|
|||
|
NTSTATUS ExceptionCode
|
|||
|
);
|
|||
|
|
|||
|
|
|||
|
//
|
|||
|
// Implemented in heap.c
|
|||
|
//
|
|||
|
|
|||
|
BOOLEAN
|
|||
|
RtlpInitializeHeapSegment (
|
|||
|
IN PHEAP Heap,
|
|||
|
IN PHEAP_SEGMENT Segment,
|
|||
|
IN UCHAR SegmentIndex,
|
|||
|
IN ULONG Flags,
|
|||
|
IN PVOID BaseAddress,
|
|||
|
IN PVOID UnCommittedAddress,
|
|||
|
IN PVOID CommitLimitAddress
|
|||
|
);
|
|||
|
|
|||
|
PHEAP_FREE_ENTRY
|
|||
|
RtlpCoalesceFreeBlocks (
|
|||
|
IN PHEAP Heap,
|
|||
|
IN PHEAP_FREE_ENTRY FreeBlock,
|
|||
|
IN OUT PSIZE_T FreeSize,
|
|||
|
IN BOOLEAN RemoveFromFreeList
|
|||
|
);
|
|||
|
|
|||
|
VOID
|
|||
|
RtlpDeCommitFreeBlock (
|
|||
|
IN PHEAP Heap,
|
|||
|
IN PHEAP_FREE_ENTRY FreeBlock,
|
|||
|
IN SIZE_T FreeSize
|
|||
|
);
|
|||
|
|
|||
|
VOID
|
|||
|
RtlpInsertFreeBlock (
|
|||
|
IN PHEAP Heap,
|
|||
|
IN PHEAP_FREE_ENTRY FreeBlock,
|
|||
|
IN SIZE_T FreeSize
|
|||
|
);
|
|||
|
|
|||
|
PHEAP_FREE_ENTRY
|
|||
|
RtlpFindAndCommitPages (
|
|||
|
IN PHEAP Heap,
|
|||
|
IN PHEAP_SEGMENT Segment,
|
|||
|
IN OUT PSIZE_T Size,
|
|||
|
IN PVOID AddressWanted OPTIONAL
|
|||
|
);
|
|||
|
|
|||
|
PVOID
|
|||
|
RtlAllocateHeapSlowly (
|
|||
|
IN PVOID HeapHandle,
|
|||
|
IN ULONG Flags,
|
|||
|
IN SIZE_T Size
|
|||
|
);
|
|||
|
|
|||
|
BOOLEAN
|
|||
|
RtlFreeHeapSlowly (
|
|||
|
IN PVOID HeapHandle,
|
|||
|
IN ULONG Flags,
|
|||
|
IN PVOID BaseAddress
|
|||
|
);
|
|||
|
|
|||
|
SIZE_T
|
|||
|
RtlpGetSizeOfBigBlock (
|
|||
|
IN PHEAP_ENTRY BusyBlock
|
|||
|
);
|
|||
|
|
|||
|
PHEAP_ENTRY_EXTRA
|
|||
|
RtlpGetExtraStuffPointer (
|
|||
|
PHEAP_ENTRY BusyBlock
|
|||
|
);
|
|||
|
|
|||
|
BOOLEAN
|
|||
|
RtlpCheckBusyBlockTail (
|
|||
|
IN PHEAP_ENTRY BusyBlock
|
|||
|
);
|
|||
|
|
|||
|
|
|||
|
//
|
|||
|
// Implemented in heapdll.c
|
|||
|
//
|
|||
|
|
|||
|
VOID
|
|||
|
RtlpAddHeapToProcessList (
|
|||
|
IN PHEAP Heap
|
|||
|
);
|
|||
|
|
|||
|
VOID
|
|||
|
RtlpRemoveHeapFromProcessList (
|
|||
|
IN PHEAP Heap
|
|||
|
);
|
|||
|
|
|||
|
PHEAP_FREE_ENTRY
|
|||
|
RtlpCoalesceHeap (
|
|||
|
IN PHEAP Heap
|
|||
|
);
|
|||
|
|
|||
|
BOOLEAN
|
|||
|
RtlpCheckHeapSignature (
|
|||
|
IN PHEAP Heap,
|
|||
|
IN PCHAR Caller
|
|||
|
);
|
|||
|
|
|||
|
VOID
|
|||
|
RtlDetectHeapLeaks();
|
|||
|
|
|||
|
|
|||
|
//
|
|||
|
// Implemented in heapdbg.c
|
|||
|
//
|
|||
|
|
|||
|
BOOLEAN
|
|||
|
RtlpValidateHeapEntry (
|
|||
|
IN PHEAP Heap,
|
|||
|
IN PHEAP_ENTRY BusyBlock,
|
|||
|
IN PCHAR Reason
|
|||
|
);
|
|||
|
|
|||
|
BOOLEAN
|
|||
|
RtlpValidateHeap (
|
|||
|
IN PHEAP Heap,
|
|||
|
IN BOOLEAN AlwaysValidate
|
|||
|
);
|
|||
|
|
|||
|
VOID
|
|||
|
RtlpUpdateHeapListIndex (
|
|||
|
USHORT OldIndex,
|
|||
|
USHORT NewIndex
|
|||
|
);
|
|||
|
|
|||
|
BOOLEAN
|
|||
|
RtlpValidateHeapHeaders(
|
|||
|
IN PHEAP Heap,
|
|||
|
IN BOOLEAN Recompute
|
|||
|
);
|
|||
|
|
|||
|
|
|||
|
#ifndef NTOS_KERNEL_RUNTIME
|
|||
|
|
|||
|
//
|
|||
|
// Nondedicated free list optimization
|
|||
|
//
|
|||
|
|
|||
|
#if DBG
|
|||
|
|
|||
|
//
|
|||
|
// Define HEAP_VALIDATE_INDEX to activate a validation of the index
|
|||
|
// after each operation with non-dedicated list
|
|||
|
// This is only for debug-test, to make sure the list and index is consistent
|
|||
|
//
|
|||
|
|
|||
|
//#define HEAP_VALIDATE_INDEX
|
|||
|
|
|||
|
#endif // DBG
|
|||
|
|
|||
|
|
|||
|
#define HEAP_FRONT_LOOKASIDE 1
|
|||
|
#define HEAP_FRONT_LOWFRAGHEAP 2
|
|||
|
|
|||
|
#define RtlpGetLookasideHeap(H) \
|
|||
|
(((H)->FrontEndHeapType == HEAP_FRONT_LOOKASIDE) ? (H)->FrontEndHeap : NULL)
|
|||
|
|
|||
|
#define RtlpGetLowFragHeap(H) \
|
|||
|
(((H)->FrontEndHeapType == HEAP_FRONT_LOWFRAGHEAP) ? (H)->FrontEndHeap : NULL)
|
|||
|
|
|||
|
#define RtlpIsFrontHeapUnlocked(H) \
|
|||
|
((H)->FrontHeapLockCount == 0)
|
|||
|
|
|||
|
#define RtlpLockFrontHeap(H) \
|
|||
|
{ \
|
|||
|
(H)->FrontHeapLockCount += 1; \
|
|||
|
}
|
|||
|
|
|||
|
#define RtlpUnlockFrontHeap(H) \
|
|||
|
{ \
|
|||
|
(H)->FrontHeapLockCount -= 1; \
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
#define HEAP_INDEX_THRESHOLD 32
|
|||
|
|
|||
|
//
|
|||
|
// Heap performance counter support
|
|||
|
//
|
|||
|
|
|||
|
#define HEAP_OP_COUNT 2
|
|||
|
|
|||
|
#define HEAP_OP_ALLOC 0
|
|||
|
#define HEAP_OP_FREE 1
|
|||
|
|
|||
|
//
|
|||
|
// The time / per operation is measured ones at 16 operations
|
|||
|
//
|
|||
|
|
|||
|
#define HEAP_SAMPLING_MASK 0x000001FF
|
|||
|
|
|||
|
#define HEAP_SAMPLING_COUNT 100
|
|||
|
|
|||
|
typedef struct _HEAP_PERF_DATA {
|
|||
|
|
|||
|
UINT64 CountFrequence;
|
|||
|
UINT64 OperationTime[HEAP_OP_COUNT];
|
|||
|
|
|||
|
//
|
|||
|
// The data bellow are only for sampling
|
|||
|
//
|
|||
|
|
|||
|
ULONG Sequence;
|
|||
|
|
|||
|
UINT64 TempTime[HEAP_OP_COUNT];
|
|||
|
ULONG TempCount[HEAP_OP_COUNT];
|
|||
|
|
|||
|
} HEAP_PERF_DATA, *PHEAP_PERF_DATA;
|
|||
|
|
|||
|
#define HEAP_PERF_DECLARE_TIMER() \
|
|||
|
UINT64 _HeapPerfStartTimer, _HeapPerfEndTimer;
|
|||
|
|
|||
|
#define HEAP_PERF_START_TIMER(H) \
|
|||
|
{ \
|
|||
|
PHEAP_INDEX HeapIndex = (PHEAP_INDEX)(H)->LargeBlocksIndex; \
|
|||
|
if ( (HeapIndex != NULL) && \
|
|||
|
(!((HeapIndex->PerfData.Sequence++) & HEAP_SAMPLING_MASK)) ) { \
|
|||
|
\
|
|||
|
NtQueryPerformanceCounter( (PLARGE_INTEGER)&_HeapPerfStartTimer , NULL); \
|
|||
|
} else { \
|
|||
|
_HeapPerfStartTimer = 0; \
|
|||
|
} \
|
|||
|
}
|
|||
|
|
|||
|
#define HEAP_PERF_STOP_TIMER(H,OP) \
|
|||
|
{ \
|
|||
|
if (_HeapPerfStartTimer) { \
|
|||
|
PHEAP_INDEX HeapIndex = (PHEAP_INDEX)(H)->LargeBlocksIndex; \
|
|||
|
\
|
|||
|
NtQueryPerformanceCounter( (PLARGE_INTEGER)&_HeapPerfEndTimer , NULL); \
|
|||
|
HeapIndex->PerfData.TempTime[OP] += (_HeapPerfEndTimer - _HeapPerfStartTimer); \
|
|||
|
\
|
|||
|
if ((HeapIndex->PerfData.TempCount[OP]++) >= HEAP_SAMPLING_COUNT) { \
|
|||
|
HeapIndex->PerfData.OperationTime[OP] = HeapIndex->PerfData.TempTime[OP] / (HeapIndex->PerfData.TempCount[OP] - 1); \
|
|||
|
\
|
|||
|
HeapIndex->PerfData.TempCount[OP] = 0; \
|
|||
|
HeapIndex->PerfData.TempTime[OP] = 0; \
|
|||
|
} \
|
|||
|
} \
|
|||
|
}
|
|||
|
|
|||
|
#define RtlpRegisterOperation(H,S,Op) \
|
|||
|
{ \
|
|||
|
PHEAP_LOOKASIDE Lookaside; \
|
|||
|
\
|
|||
|
if ( (Lookaside = (PHEAP_LOOKASIDE)RtlpGetLookasideHeap(H)) ) { \
|
|||
|
\
|
|||
|
SIZE_T Index = (S) >> 10; \
|
|||
|
\
|
|||
|
if (Index >= HEAP_MAXIMUM_FREELISTS) { \
|
|||
|
\
|
|||
|
Index = HEAP_MAXIMUM_FREELISTS - 1; \
|
|||
|
} \
|
|||
|
\
|
|||
|
Lookaside[Index].Counters[(Op)] += 1; \
|
|||
|
} \
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// The heap index structure
|
|||
|
//
|
|||
|
|
|||
|
typedef struct _HEAP_INDEX {
|
|||
|
|
|||
|
ULONG ArraySize;
|
|||
|
ULONG VirtualMemorySize;
|
|||
|
|
|||
|
//
|
|||
|
// The timing counters are available only on heaps
|
|||
|
// with an index created
|
|||
|
//
|
|||
|
|
|||
|
HEAP_PERF_DATA PerfData;
|
|||
|
|
|||
|
LONG LargeBlocksCacheDepth;
|
|||
|
LONG LargeBlocksCacheMaxDepth;
|
|||
|
LONG LargeBlocksCacheMinDepth;
|
|||
|
LONG LargeBlocksCacheSequence;
|
|||
|
|
|||
|
struct {
|
|||
|
|
|||
|
ULONG Committs;
|
|||
|
ULONG Decommitts;
|
|||
|
LONG LargestDepth;
|
|||
|
LONG LargestRequiredDepth;
|
|||
|
|
|||
|
} CacheStats;
|
|||
|
|
|||
|
union {
|
|||
|
|
|||
|
PULONG FreeListsInUseUlong;
|
|||
|
PUCHAR FreeListsInUseBytes;
|
|||
|
} u;
|
|||
|
|
|||
|
PHEAP_FREE_ENTRY * FreeListHints;
|
|||
|
|
|||
|
} HEAP_INDEX, *PHEAP_INDEX;
|
|||
|
|
|||
|
//
|
|||
|
// Macro for setting a bit in the freelist vector to indicate entries are
|
|||
|
// present.
|
|||
|
//
|
|||
|
|
|||
|
#define SET_INDEX_BIT( HeapIndex, AllocIndex ) \
|
|||
|
{ \
|
|||
|
ULONG _Index_; \
|
|||
|
ULONG _Bit_; \
|
|||
|
\
|
|||
|
_Index_ = (AllocIndex) >> 3; \
|
|||
|
_Bit_ = (1 << ((AllocIndex) & 7)); \
|
|||
|
\
|
|||
|
(HeapIndex)->u.FreeListsInUseBytes[ _Index_ ] |= _Bit_; \
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// Macro for clearing a bit in the freelist vector to indicate entries are
|
|||
|
// not present.
|
|||
|
//
|
|||
|
|
|||
|
#define CLEAR_INDEX_BIT( HeapIndex, AllocIndex ) \
|
|||
|
{ \
|
|||
|
ULONG _Index_; \
|
|||
|
ULONG _Bit_; \
|
|||
|
\
|
|||
|
_Index_ = (AllocIndex) >> 3; \
|
|||
|
_Bit_ = (1 << ((AllocIndex) & 7)); \
|
|||
|
\
|
|||
|
(HeapIndex)->u.FreeListsInUseBytes[ _Index_ ] ^= _Bit_; \
|
|||
|
}
|
|||
|
|
|||
|
VOID
|
|||
|
RtlpInitializeListIndex (
|
|||
|
IN PHEAP Heap
|
|||
|
);
|
|||
|
|
|||
|
PLIST_ENTRY
|
|||
|
RtlpFindEntry (
|
|||
|
IN PHEAP Heap,
|
|||
|
IN ULONG Size
|
|||
|
);
|
|||
|
|
|||
|
VOID
|
|||
|
RtlpUpdateIndexRemoveBlock (
|
|||
|
IN PHEAP Heap,
|
|||
|
IN PHEAP_FREE_ENTRY FreeEntry
|
|||
|
);
|
|||
|
|
|||
|
VOID
|
|||
|
RtlpUpdateIndexInsertBlock (
|
|||
|
IN PHEAP Heap,
|
|||
|
IN PHEAP_FREE_ENTRY FreeEntry
|
|||
|
);
|
|||
|
|
|||
|
VOID
|
|||
|
RtlpFlushCacheContents (
|
|||
|
IN PHEAP Heap
|
|||
|
);
|
|||
|
|
|||
|
extern LONG RtlpSequenceNumberTest;
|
|||
|
|
|||
|
#define RtlpCheckLargeCache(H) \
|
|||
|
{ \
|
|||
|
PHEAP_INDEX HeapIndex = (PHEAP_INDEX)(H)->LargeBlocksIndex; \
|
|||
|
if ((HeapIndex != NULL) && \
|
|||
|
(HeapIndex->LargeBlocksCacheSequence >= RtlpSequenceNumberTest)) { \
|
|||
|
\
|
|||
|
RtlpFlushCacheContents(Heap); \
|
|||
|
} \
|
|||
|
}
|
|||
|
|
|||
|
VOID
|
|||
|
RtlpFlushLargestCacheBlock (
|
|||
|
IN PHEAP Heap
|
|||
|
);
|
|||
|
|
|||
|
#ifdef HEAP_VALIDATE_INDEX
|
|||
|
|
|||
|
//
|
|||
|
// The validation code for index
|
|||
|
//
|
|||
|
|
|||
|
BOOLEAN
|
|||
|
RtlpValidateNonDedicatedList (
|
|||
|
IN PHEAP Heap
|
|||
|
);
|
|||
|
|
|||
|
|
|||
|
#else // HEAP_VALIDATE_INDEX
|
|||
|
|
|||
|
#define RtlpValidateNonDedicatedList(H)
|
|||
|
|
|||
|
#endif // HEAP_VALIDATE_INDEX
|
|||
|
|
|||
|
|
|||
|
#else // NTOS_KERNEL_RUNTIME
|
|||
|
|
|||
|
#define HEAP_PERF_DECLARE_TIMER()
|
|||
|
|
|||
|
#define HEAP_PERF_START_TIMER(H)
|
|||
|
|
|||
|
#define HEAP_PERF_STOP_TIMER(H,Op)
|
|||
|
|
|||
|
#define RtlpRegisterOperation(H,S,Op)
|
|||
|
|
|||
|
|
|||
|
#define RtlpInitializeListIndex(H)
|
|||
|
|
|||
|
#define RtlpFindEntry(H,S) (NULL)
|
|||
|
|
|||
|
#define RtlpUpdateIndexRemoveBlock(H,F)
|
|||
|
|
|||
|
#define RtlpUpdateIndexInsertBlock(H,F)
|
|||
|
|
|||
|
#define RtlpCheckLargeCache(H)
|
|||
|
|
|||
|
#define RtlpValidateNonDedicatedList(H)
|
|||
|
|
|||
|
#endif // NTOS_KERNEL_RUNTIME
|
|||
|
|
|||
|
|
|||
|
//
|
|||
|
// An extra bitmap manipulation routine
|
|||
|
//
|
|||
|
|
|||
|
#define RtlFindFirstSetRightMember(Set) \
|
|||
|
(((Set) & 0xFFFF) ? \
|
|||
|
(((Set) & 0xFF) ? \
|
|||
|
RtlpBitsClearLow[(Set) & 0xFF] : \
|
|||
|
RtlpBitsClearLow[((Set) >> 8) & 0xFF] + 8) : \
|
|||
|
((((Set) >> 16) & 0xFF) ? \
|
|||
|
RtlpBitsClearLow[ ((Set) >> 16) & 0xFF] + 16 : \
|
|||
|
RtlpBitsClearLow[ (Set) >> 24] + 24) \
|
|||
|
)
|
|||
|
|
|||
|
|
|||
|
//
|
|||
|
// Macro for setting a bit in the freelist vector to indicate entries are
|
|||
|
// present.
|
|||
|
//
|
|||
|
|
|||
|
#define SET_FREELIST_BIT( H, FB ) \
|
|||
|
{ \
|
|||
|
ULONG _Index_; \
|
|||
|
ULONG _Bit_; \
|
|||
|
\
|
|||
|
HEAPASSERT((FB)->Size < HEAP_MAXIMUM_FREELISTS); \
|
|||
|
\
|
|||
|
_Index_ = (FB)->Size >> 3; \
|
|||
|
_Bit_ = (1 << ((FB)->Size & 7)); \
|
|||
|
\
|
|||
|
HEAPASSERT(((H)->u.FreeListsInUseBytes[ _Index_ ] & _Bit_) == 0); \
|
|||
|
\
|
|||
|
(H)->u.FreeListsInUseBytes[ _Index_ ] |= _Bit_; \
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// Macro for clearing a bit in the freelist vector to indicate entries are
|
|||
|
// not present.
|
|||
|
//
|
|||
|
|
|||
|
#define CLEAR_FREELIST_BIT( H, FB ) \
|
|||
|
{ \
|
|||
|
ULONG _Index_; \
|
|||
|
ULONG _Bit_; \
|
|||
|
\
|
|||
|
HEAPASSERT((FB)->Size < HEAP_MAXIMUM_FREELISTS); \
|
|||
|
\
|
|||
|
_Index_ = (FB)->Size >> 3; \
|
|||
|
_Bit_ = (1 << ((FB)->Size & 7)); \
|
|||
|
\
|
|||
|
HEAPASSERT((H)->u.FreeListsInUseBytes[ _Index_ ] & _Bit_); \
|
|||
|
HEAPASSERT(IsListEmpty(&(H)->FreeLists[ (FB)->Size ])); \
|
|||
|
\
|
|||
|
(H)->u.FreeListsInUseBytes[ _Index_ ] ^= _Bit_; \
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
//
|
|||
|
// This macro inserts a free block into the appropriate free list including
|
|||
|
// the [0] index list with entry filling if necessary
|
|||
|
//
|
|||
|
|
|||
|
#define RtlpInsertFreeBlockDirect( H, FB, SIZE ) \
|
|||
|
{ \
|
|||
|
PLIST_ENTRY _HEAD, _NEXT; \
|
|||
|
PHEAP_FREE_ENTRY _FB1; \
|
|||
|
\
|
|||
|
HEAPASSERT((FB)->Size == (SIZE)); \
|
|||
|
(FB)->Flags &= ~(HEAP_ENTRY_FILL_PATTERN | \
|
|||
|
HEAP_ENTRY_EXTRA_PRESENT | \
|
|||
|
HEAP_ENTRY_BUSY); \
|
|||
|
\
|
|||
|
if ((H)->Flags & HEAP_FREE_CHECKING_ENABLED) { \
|
|||
|
\
|
|||
|
RtlFillMemoryUlong( (PCHAR)((FB) + 1), \
|
|||
|
((SIZE) << HEAP_GRANULARITY_SHIFT) - \
|
|||
|
sizeof( *(FB) ), \
|
|||
|
FREE_HEAP_FILL ); \
|
|||
|
\
|
|||
|
(FB)->Flags |= HEAP_ENTRY_FILL_PATTERN; \
|
|||
|
} \
|
|||
|
\
|
|||
|
if ((SIZE) < HEAP_MAXIMUM_FREELISTS) { \
|
|||
|
\
|
|||
|
_HEAD = &(H)->FreeLists[ (SIZE) ]; \
|
|||
|
\
|
|||
|
if (IsListEmpty(_HEAD)) { \
|
|||
|
\
|
|||
|
SET_FREELIST_BIT( H, FB ); \
|
|||
|
} \
|
|||
|
\
|
|||
|
} else { \
|
|||
|
\
|
|||
|
_HEAD = &(H)->FreeLists[ 0 ]; \
|
|||
|
_NEXT = (H)->LargeBlocksIndex ? \
|
|||
|
RtlpFindEntry(H, SIZE) : \
|
|||
|
_HEAD->Flink; \
|
|||
|
\
|
|||
|
while (_HEAD != _NEXT) { \
|
|||
|
\
|
|||
|
_FB1 = CONTAINING_RECORD( _NEXT, HEAP_FREE_ENTRY, FreeList ); \
|
|||
|
\
|
|||
|
if ((SIZE) <= _FB1->Size) { \
|
|||
|
\
|
|||
|
break; \
|
|||
|
\
|
|||
|
} else { \
|
|||
|
\
|
|||
|
_NEXT = _NEXT->Flink; \
|
|||
|
} \
|
|||
|
} \
|
|||
|
\
|
|||
|
_HEAD = _NEXT; \
|
|||
|
} \
|
|||
|
\
|
|||
|
InsertTailList( _HEAD, &(FB)->FreeList ); \
|
|||
|
RtlpUpdateIndexInsertBlock(H, FB); \
|
|||
|
RtlpValidateNonDedicatedList(H); \
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// This version of RtlpInsertFreeBlockDirect does no filling.
|
|||
|
//
|
|||
|
|
|||
|
#define RtlpFastInsertFreeBlockDirect( H, FB, SIZE ) \
|
|||
|
{ \
|
|||
|
if ((SIZE) < HEAP_MAXIMUM_FREELISTS) { \
|
|||
|
\
|
|||
|
RtlpFastInsertDedicatedFreeBlockDirect( H, FB, SIZE ); \
|
|||
|
\
|
|||
|
} else { \
|
|||
|
\
|
|||
|
RtlpFastInsertNonDedicatedFreeBlockDirect( H, FB, SIZE ); \
|
|||
|
} \
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// This version of RtlpInsertFreeBlockDirect only works for dedicated free
|
|||
|
// lists and doesn't do any filling.
|
|||
|
//
|
|||
|
|
|||
|
#define RtlpFastInsertDedicatedFreeBlockDirect( H, FB, SIZE ) \
|
|||
|
{ \
|
|||
|
PLIST_ENTRY _HEAD; \
|
|||
|
\
|
|||
|
HEAPASSERT((FB)->Size == (SIZE)); \
|
|||
|
\
|
|||
|
if (!((FB)->Flags & HEAP_ENTRY_LAST_ENTRY)) { \
|
|||
|
\
|
|||
|
HEAPASSERT(((PHEAP_ENTRY)(FB) + (SIZE))->PreviousSize == (SIZE)); \
|
|||
|
} \
|
|||
|
\
|
|||
|
(FB)->Flags &= HEAP_ENTRY_LAST_ENTRY; \
|
|||
|
\
|
|||
|
_HEAD = &(H)->FreeLists[ (SIZE) ]; \
|
|||
|
\
|
|||
|
if (IsListEmpty(_HEAD)) { \
|
|||
|
\
|
|||
|
SET_FREELIST_BIT( H, FB ); \
|
|||
|
} \
|
|||
|
\
|
|||
|
InsertTailList( _HEAD, &(FB)->FreeList ); \
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// This version of RtlpInsertFreeBlockDirect only works for nondedicated free
|
|||
|
// lists and doesn't do any filling.
|
|||
|
//
|
|||
|
|
|||
|
#define RtlpFastInsertNonDedicatedFreeBlockDirect( H, FB, SIZE ) \
|
|||
|
{ \
|
|||
|
PLIST_ENTRY _HEAD, _NEXT; \
|
|||
|
PHEAP_FREE_ENTRY _FB1; \
|
|||
|
\
|
|||
|
HEAPASSERT((FB)->Size == (SIZE)); \
|
|||
|
\
|
|||
|
if (!((FB)->Flags & HEAP_ENTRY_LAST_ENTRY)) { \
|
|||
|
\
|
|||
|
HEAPASSERT(((PHEAP_ENTRY)(FB) + (SIZE))->PreviousSize == (SIZE)); \
|
|||
|
} \
|
|||
|
\
|
|||
|
(FB)->Flags &= (HEAP_ENTRY_LAST_ENTRY); \
|
|||
|
\
|
|||
|
_HEAD = &(H)->FreeLists[ 0 ]; \
|
|||
|
_NEXT = (H)->LargeBlocksIndex ? \
|
|||
|
RtlpFindEntry(H, SIZE) : \
|
|||
|
_HEAD->Flink; \
|
|||
|
\
|
|||
|
while (_HEAD != _NEXT) { \
|
|||
|
\
|
|||
|
_FB1 = CONTAINING_RECORD( _NEXT, HEAP_FREE_ENTRY, FreeList ); \
|
|||
|
\
|
|||
|
if ((SIZE) <= _FB1->Size) { \
|
|||
|
\
|
|||
|
break; \
|
|||
|
\
|
|||
|
} else { \
|
|||
|
\
|
|||
|
_NEXT = _NEXT->Flink; \
|
|||
|
} \
|
|||
|
} \
|
|||
|
\
|
|||
|
InsertTailList( _NEXT, &(FB)->FreeList ); \
|
|||
|
RtlpUpdateIndexInsertBlock(H, FB); \
|
|||
|
RtlpValidateNonDedicatedList(H); \
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
//
|
|||
|
// This macro removes a block from its free list with fill checking if
|
|||
|
// necessary
|
|||
|
//
|
|||
|
|
|||
|
#define RtlpRemoveFreeBlock( H, FB ) \
|
|||
|
{ \
|
|||
|
RtlpFastRemoveFreeBlock( H, FB ) \
|
|||
|
\
|
|||
|
if ((FB)->Flags & HEAP_ENTRY_FILL_PATTERN) { \
|
|||
|
\
|
|||
|
SIZE_T cb, cbEqual; \
|
|||
|
PVOID p; \
|
|||
|
\
|
|||
|
cb = ((FB)->Size << HEAP_GRANULARITY_SHIFT) - sizeof( *(FB) ); \
|
|||
|
\
|
|||
|
if ((FB)->Flags & HEAP_ENTRY_EXTRA_PRESENT && \
|
|||
|
cb > sizeof( HEAP_FREE_ENTRY_EXTRA )) { \
|
|||
|
\
|
|||
|
cb -= sizeof( HEAP_FREE_ENTRY_EXTRA ); \
|
|||
|
} \
|
|||
|
\
|
|||
|
cbEqual = RtlCompareMemoryUlong( (PCHAR)((FB) + 1), \
|
|||
|
cb, \
|
|||
|
FREE_HEAP_FILL ); \
|
|||
|
\
|
|||
|
if (cbEqual != cb) { \
|
|||
|
\
|
|||
|
HeapDebugPrint(( \
|
|||
|
"HEAP: Free Heap block %lx modified at %lx after it was freed\n", \
|
|||
|
(FB), \
|
|||
|
(PCHAR)((FB) + 1) + cbEqual )); \
|
|||
|
\
|
|||
|
HeapDebugBreak((FB)); \
|
|||
|
} \
|
|||
|
} \
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
#ifndef NTOS_KERNEL_RUNTIME
|
|||
|
|
|||
|
VOID
|
|||
|
RtlpHeapReportCorruption (
|
|||
|
IN PVOID Address );
|
|||
|
|
|||
|
#else // NTOS_KERNEL_RUNTIME
|
|||
|
|
|||
|
#define RtlpHeapReportCorruption(__x__)
|
|||
|
|
|||
|
#endif // NTOS_KERNEL_RUNTIME
|
|||
|
|
|||
|
//
|
|||
|
// This version of RtlpRemoveFreeBlock does no fill checking
|
|||
|
//
|
|||
|
|
|||
|
#define RtlpFastRemoveFreeBlock( H, FB ) \
|
|||
|
{ \
|
|||
|
PLIST_ENTRY _EX_Blink; \
|
|||
|
PLIST_ENTRY _EX_Flink; \
|
|||
|
\
|
|||
|
_EX_Flink = (FB)->FreeList.Flink; \
|
|||
|
_EX_Blink = (FB)->FreeList.Blink; \
|
|||
|
\
|
|||
|
if ( (_EX_Blink->Flink == _EX_Flink->Blink) && \
|
|||
|
(_EX_Blink->Flink == &(FB)->FreeList) ) { \
|
|||
|
\
|
|||
|
RtlpUpdateIndexRemoveBlock(H, FB); \
|
|||
|
\
|
|||
|
_EX_Blink->Flink = _EX_Flink; \
|
|||
|
_EX_Flink->Blink = _EX_Blink; \
|
|||
|
\
|
|||
|
if ((_EX_Flink == _EX_Blink) && \
|
|||
|
((FB)->Size < HEAP_MAXIMUM_FREELISTS)) { \
|
|||
|
\
|
|||
|
CLEAR_FREELIST_BIT( H, FB ); \
|
|||
|
} \
|
|||
|
RtlpValidateNonDedicatedList(H); \
|
|||
|
\
|
|||
|
} else { \
|
|||
|
\
|
|||
|
RtlpHeapReportCorruption(&(FB)->FreeList); \
|
|||
|
} \
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// This version of RtlpRemoveFreeBlock only works for dedicated free lists
|
|||
|
// (where we know that (FB)->Mask != 0) and doesn't do any fill checking
|
|||
|
//
|
|||
|
|
|||
|
#define RtlpFastRemoveDedicatedFreeBlock( H, FB ) \
|
|||
|
{ \
|
|||
|
PLIST_ENTRY _EX_Blink; \
|
|||
|
PLIST_ENTRY _EX_Flink; \
|
|||
|
\
|
|||
|
_EX_Flink = (FB)->FreeList.Flink; \
|
|||
|
_EX_Blink = (FB)->FreeList.Blink; \
|
|||
|
\
|
|||
|
if ( (_EX_Blink->Flink == _EX_Flink->Blink)&& \
|
|||
|
(_EX_Blink->Flink == &(FB)->FreeList) ){ \
|
|||
|
\
|
|||
|
_EX_Blink->Flink = _EX_Flink; \
|
|||
|
_EX_Flink->Blink = _EX_Blink; \
|
|||
|
\
|
|||
|
} else { \
|
|||
|
\
|
|||
|
RtlpHeapReportCorruption(&(FB)->FreeList);\
|
|||
|
} \
|
|||
|
\
|
|||
|
if (_EX_Flink == _EX_Blink) { \
|
|||
|
\
|
|||
|
CLEAR_FREELIST_BIT( H, FB ); \
|
|||
|
} \
|
|||
|
}
|
|||
|
|
|||
|
BOOLEAN
|
|||
|
FORCEINLINE
|
|||
|
RtlpHeapRemoveEntryList(
|
|||
|
IN PLIST_ENTRY Entry
|
|||
|
)
|
|||
|
{
|
|||
|
PLIST_ENTRY Blink;
|
|||
|
PLIST_ENTRY Flink;
|
|||
|
|
|||
|
Flink = Entry->Flink;
|
|||
|
Blink = Entry->Blink;
|
|||
|
|
|||
|
if ( (Blink->Flink == Flink->Blink) &&
|
|||
|
(Blink->Flink == Entry) ) {
|
|||
|
|
|||
|
Blink->Flink = Flink;
|
|||
|
Flink->Blink = Blink;
|
|||
|
|
|||
|
} else {
|
|||
|
|
|||
|
RtlpHeapReportCorruption(Entry);
|
|||
|
}
|
|||
|
|
|||
|
return (BOOLEAN)(Flink == Blink);
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
//
|
|||
|
// This version of RtlpRemoveFreeBlock only works for dedicated free lists
|
|||
|
// (where we know that (FB)->Mask == 0) and doesn't do any fill checking
|
|||
|
//
|
|||
|
|
|||
|
#define RtlpFastRemoveNonDedicatedFreeBlock( H, FB ) \
|
|||
|
{ \
|
|||
|
RtlpUpdateIndexRemoveBlock(H, FB); \
|
|||
|
RtlpHeapRemoveEntryList(&(FB)->FreeList); \
|
|||
|
RtlpValidateNonDedicatedList(H); \
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
|
|||
|
//
|
|||
|
// Heap tagging routines implemented in heapdll.c
|
|||
|
//
|
|||
|
|
|||
|
#if DBG
|
|||
|
|
|||
|
#define IS_HEAP_TAGGING_ENABLED() (TRUE)
|
|||
|
|
|||
|
#else
|
|||
|
|
|||
|
#define IS_HEAP_TAGGING_ENABLED() (RtlGetNtGlobalFlags() & FLG_HEAP_ENABLE_TAGGING)
|
|||
|
|
|||
|
#endif // DBG
|
|||
|
|
|||
|
//
|
|||
|
// ORDER IS IMPORTANT HERE...SEE RtlpUpdateTagEntry sources
|
|||
|
//
|
|||
|
|
|||
|
typedef enum _HEAP_TAG_ACTION {
|
|||
|
|
|||
|
AllocationAction,
|
|||
|
VirtualAllocationAction,
|
|||
|
FreeAction,
|
|||
|
VirtualFreeAction,
|
|||
|
ReAllocationAction,
|
|||
|
VirtualReAllocationAction
|
|||
|
|
|||
|
} HEAP_TAG_ACTION;
|
|||
|
|
|||
|
PWSTR
|
|||
|
RtlpGetTagName (
|
|||
|
PHEAP Heap,
|
|||
|
USHORT TagIndex
|
|||
|
);
|
|||
|
|
|||
|
USHORT
|
|||
|
RtlpUpdateTagEntry (
|
|||
|
PHEAP Heap,
|
|||
|
USHORT TagIndex,
|
|||
|
SIZE_T OldSize, // Only valid for ReAllocation and Free actions
|
|||
|
SIZE_T NewSize, // Only valid for ReAllocation and Allocation actions
|
|||
|
HEAP_TAG_ACTION Action
|
|||
|
);
|
|||
|
|
|||
|
VOID
|
|||
|
RtlpResetTags (
|
|||
|
PHEAP Heap
|
|||
|
);
|
|||
|
|
|||
|
VOID
|
|||
|
RtlpDestroyTags (
|
|||
|
PHEAP Heap
|
|||
|
);
|
|||
|
|
|||
|
|
|||
|
//
|
|||
|
// Define heap lookaside list allocation functions.
|
|||
|
//
|
|||
|
|
|||
|
typedef struct _HEAP_LOOKASIDE {
|
|||
|
SLIST_HEADER ListHead;
|
|||
|
|
|||
|
USHORT Depth;
|
|||
|
USHORT MaximumDepth;
|
|||
|
|
|||
|
ULONG TotalAllocates;
|
|||
|
ULONG AllocateMisses;
|
|||
|
ULONG TotalFrees;
|
|||
|
ULONG FreeMisses;
|
|||
|
|
|||
|
ULONG LastTotalAllocates;
|
|||
|
ULONG LastAllocateMisses;
|
|||
|
|
|||
|
ULONG Counters[2];
|
|||
|
|
|||
|
} HEAP_LOOKASIDE, *PHEAP_LOOKASIDE;
|
|||
|
|
|||
|
NTKERNELAPI
|
|||
|
VOID
|
|||
|
RtlpInitializeHeapLookaside (
|
|||
|
IN PHEAP_LOOKASIDE Lookaside,
|
|||
|
IN USHORT Depth
|
|||
|
);
|
|||
|
|
|||
|
NTKERNELAPI
|
|||
|
VOID
|
|||
|
RtlpDeleteHeapLookaside (
|
|||
|
IN PHEAP_LOOKASIDE Lookaside
|
|||
|
);
|
|||
|
|
|||
|
VOID
|
|||
|
RtlpAdjustHeapLookasideDepth (
|
|||
|
IN PHEAP_LOOKASIDE Lookaside
|
|||
|
);
|
|||
|
|
|||
|
NTKERNELAPI
|
|||
|
PVOID
|
|||
|
RtlpAllocateFromHeapLookaside (
|
|||
|
IN PHEAP_LOOKASIDE Lookaside
|
|||
|
);
|
|||
|
|
|||
|
NTKERNELAPI
|
|||
|
BOOLEAN
|
|||
|
RtlpFreeToHeapLookaside (
|
|||
|
IN PHEAP_LOOKASIDE Lookaside,
|
|||
|
IN PVOID Entry
|
|||
|
);
|
|||
|
|
|||
|
#ifndef NTOS_KERNEL_RUNTIME
|
|||
|
|
|||
|
#define HEAP_LFH_INDEX ((UCHAR)0xFF)
|
|||
|
|
|||
|
UCHAR
|
|||
|
FORCEINLINE
|
|||
|
RtlpGetSmallTagIndex(
|
|||
|
IN PHEAP Heap,
|
|||
|
IN PVOID HeapEntry )
|
|||
|
{
|
|||
|
return ((PHEAP_ENTRY)HeapEntry)->SmallTagIndex ^
|
|||
|
((UCHAR)((ULONG_PTR)HeapEntry >> HEAP_GRANULARITY_SHIFT) ^ Heap->Entry.SmallTagIndex);
|
|||
|
}
|
|||
|
|
|||
|
VOID
|
|||
|
FORCEINLINE
|
|||
|
RtlpSetSmallTagIndex(
|
|||
|
IN PHEAP Heap,
|
|||
|
IN PVOID HeapEntry,
|
|||
|
IN UCHAR SmallTagIndex
|
|||
|
)
|
|||
|
{
|
|||
|
((PHEAP_ENTRY)HeapEntry)->SmallTagIndex = SmallTagIndex ^
|
|||
|
((UCHAR)((ULONG_PTR)HeapEntry >> HEAP_GRANULARITY_SHIFT) ^ Heap->Entry.SmallTagIndex);
|
|||
|
}
|
|||
|
|
|||
|
LOGICAL
|
|||
|
FORCEINLINE
|
|||
|
RtlpQuickValidateBlock(
|
|||
|
IN PHEAP Heap,
|
|||
|
IN PVOID HeapEntry )
|
|||
|
{
|
|||
|
UCHAR SegmentIndex = ((PHEAP_ENTRY)HeapEntry)->SegmentIndex;
|
|||
|
if ( SegmentIndex < HEAP_LFH_INDEX ) {
|
|||
|
|
|||
|
#if DBG
|
|||
|
|
|||
|
// The following test is usefull to detect cross heap free and
|
|||
|
// segment index corruption. However it requires fetching the some segment fields
|
|||
|
// and the perf can degrade with the heap size
|
|||
|
|
|||
|
if ( (SegmentIndex > HEAP_MAXIMUM_SEGMENTS)
|
|||
|
||
|
|||
|
(Heap->Segments[SegmentIndex] == NULL)
|
|||
|
||
|
|||
|
(HeapEntry < (PVOID)Heap->Segments[SegmentIndex])
|
|||
|
||
|
|||
|
(HeapEntry >= (PVOID)Heap->Segments[SegmentIndex]->LastValidEntry)) {
|
|||
|
|
|||
|
RtlpHeapReportCorruption(HeapEntry);
|
|||
|
return FALSE;
|
|||
|
}
|
|||
|
|
|||
|
#endif // DBG
|
|||
|
|
|||
|
if (!IS_HEAP_TAGGING_ENABLED()) {
|
|||
|
|
|||
|
if (RtlpGetSmallTagIndex(Heap, HeapEntry) != 0) {
|
|||
|
|
|||
|
RtlpHeapReportCorruption(HeapEntry);
|
|||
|
|
|||
|
return FALSE;
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
return TRUE;
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// Low Fragmentation Heap data structures and internal APIs
|
|||
|
//
|
|||
|
|
|||
|
//
|
|||
|
// The memory barrier exists on IA64 only
|
|||
|
//
|
|||
|
|
|||
|
#if defined(_IA64_)
|
|||
|
|
|||
|
#define RtlMemoryBarrier() __mf ()
|
|||
|
|
|||
|
#else // #if defined(_IA64_)
|
|||
|
|
|||
|
//
|
|||
|
// On x86 and AMD64 ignore the memory barrier
|
|||
|
//
|
|||
|
|
|||
|
#define RtlMemoryBarrier()
|
|||
|
|
|||
|
#endif // #if defined(_IA64_)
|
|||
|
|
|||
|
|
|||
|
extern ULONG RtlpDisableHeapLookaside;
|
|||
|
extern ULONG_PTR RtlpLFHKey;
|
|||
|
|
|||
|
#define HEAP_ENABLE_LOW_FRAG_HEAP 8
|
|||
|
|
|||
|
typedef struct _BLOCK_ENTRY {
|
|||
|
|
|||
|
HEAP_ENTRY;
|
|||
|
|
|||
|
USHORT LinkOffset;
|
|||
|
USHORT Reserved2;
|
|||
|
|
|||
|
} BLOCK_ENTRY, *PBLOCK_ENTRY;
|
|||
|
|
|||
|
|
|||
|
typedef struct _INTERLOCK_SEQ {
|
|||
|
|
|||
|
union {
|
|||
|
|
|||
|
struct {
|
|||
|
|
|||
|
union {
|
|||
|
|
|||
|
struct {
|
|||
|
|
|||
|
USHORT Depth;
|
|||
|
USHORT FreeEntryOffset;
|
|||
|
};
|
|||
|
volatile ULONG OffsetAndDepth;
|
|||
|
};
|
|||
|
volatile ULONG Sequence;
|
|||
|
};
|
|||
|
|
|||
|
volatile LONGLONG Exchg;
|
|||
|
};
|
|||
|
|
|||
|
} INTERLOCK_SEQ, *PINTERLOCK_SEQ;
|
|||
|
|
|||
|
struct _HEAP_USERDATA_HEADER;
|
|||
|
|
|||
|
typedef struct _HEAP_SUBSEGMENT {
|
|||
|
|
|||
|
PVOID Bucket;
|
|||
|
|
|||
|
volatile struct _HEAP_USERDATA_HEADER * UserBlocks;
|
|||
|
|
|||
|
INTERLOCK_SEQ AggregateExchg;
|
|||
|
|
|||
|
union {
|
|||
|
|
|||
|
struct {
|
|||
|
USHORT BlockSize;
|
|||
|
USHORT FreeThreshold;
|
|||
|
USHORT BlockCount;
|
|||
|
UCHAR SizeIndex;
|
|||
|
UCHAR AffinityIndex;
|
|||
|
};
|
|||
|
|
|||
|
ULONG Alignment[2];
|
|||
|
};
|
|||
|
|
|||
|
SINGLE_LIST_ENTRY SFreeListEntry;
|
|||
|
volatile ULONG Lock;
|
|||
|
|
|||
|
} HEAP_SUBSEGMENT, *PHEAP_SUBSEGMENT;
|
|||
|
|
|||
|
typedef struct _HEAP_USERDATA_HEADER {
|
|||
|
|
|||
|
union {
|
|||
|
|
|||
|
SINGLE_LIST_ENTRY SFreeListEntry;
|
|||
|
PHEAP_SUBSEGMENT SubSegment;
|
|||
|
};
|
|||
|
|
|||
|
PVOID HeapHandle;
|
|||
|
|
|||
|
ULONG_PTR SizeIndex;
|
|||
|
ULONG_PTR Signature;
|
|||
|
|
|||
|
} HEAP_USERDATA_HEADER, *PHEAP_USERDATA_HEADER;
|
|||
|
|
|||
|
#define HEAP_NO_CACHE_BLOCK 0x800000
|
|||
|
#define HEAP_LARGEST_LFH_BLOCK 0x4000
|
|||
|
#define HEAP_LFH_USER_SIGNATURE 0xF0E0D0C0
|
|||
|
|
|||
|
#ifdef DISABLE_REGISTRY_TEST_HOOKS
|
|||
|
|
|||
|
#define RtlpIsLowFragHeapEnabled() FALSE
|
|||
|
|
|||
|
#else //DISABLE_REGISTRY_TEST_HOOKS
|
|||
|
|
|||
|
#define RtlpIsLowFragHeapEnabled() \
|
|||
|
((RtlpDisableHeapLookaside & HEAP_ENABLE_LOW_FRAG_HEAP) != 0)
|
|||
|
|
|||
|
#endif //DISABLE_REGISTRY_TEST_HOOKS
|
|||
|
|
|||
|
PHEAP_SUBSEGMENT
|
|||
|
FORCEINLINE
|
|||
|
RtlpGetSubSegment(
|
|||
|
PHEAP_ENTRY Block,
|
|||
|
ULONG_PTR Key
|
|||
|
)
|
|||
|
{
|
|||
|
return (PHEAP_SUBSEGMENT)((ULONG_PTR)Block->SubSegmentCode ^
|
|||
|
(((ULONG_PTR)Block >> HEAP_GRANULARITY_SHIFT) ^ Key ^ RtlpLFHKey));
|
|||
|
}
|
|||
|
|
|||
|
VOID
|
|||
|
FORCEINLINE
|
|||
|
RtlpSetSubSegment(
|
|||
|
PHEAP_ENTRY Block,
|
|||
|
PHEAP_SUBSEGMENT SubSegment,
|
|||
|
ULONG_PTR Key
|
|||
|
)
|
|||
|
{
|
|||
|
Block->SubSegmentCode = (PVOID) (((ULONG_PTR)SubSegment)^
|
|||
|
(((ULONG_PTR)Block >> HEAP_GRANULARITY_SHIFT) ^ Key ^ RtlpLFHKey));
|
|||
|
}
|
|||
|
|
|||
|
ULONG
|
|||
|
FORCEINLINE
|
|||
|
RtlpGetAllocationUnits(
|
|||
|
PHEAP Heap,
|
|||
|
PHEAP_ENTRY Block
|
|||
|
)
|
|||
|
{
|
|||
|
|
|||
|
PHEAP_SUBSEGMENT SubSegment = RtlpGetSubSegment(Block, (ULONG_PTR)Heap);
|
|||
|
|
|||
|
if (Block->SegmentIndex == HEAP_LFH_INDEX) {
|
|||
|
|
|||
|
ULONG ReturnSize = *((volatile USHORT *)&SubSegment->BlockSize);
|
|||
|
|
|||
|
return ReturnSize;
|
|||
|
}
|
|||
|
|
|||
|
return Block->Size;
|
|||
|
}
|
|||
|
|
|||
|
VOID
|
|||
|
FORCEINLINE
|
|||
|
RtlpSetUnusedBytes(PHEAP Heap, PHEAP_ENTRY Block, SIZE_T UnusedBytes)
|
|||
|
{
|
|||
|
if (UnusedBytes < 0xff) {
|
|||
|
|
|||
|
Block->UnusedBytes = (UCHAR)(UnusedBytes);
|
|||
|
|
|||
|
} else {
|
|||
|
|
|||
|
PSIZE_T UnusedBytesULong = (PSIZE_T)(Block + RtlpGetAllocationUnits(Heap, Block));
|
|||
|
|
|||
|
UnusedBytesULong -= 1;
|
|||
|
Block->UnusedBytes = 0xff;
|
|||
|
*UnusedBytesULong = UnusedBytes;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
SIZE_T
|
|||
|
FORCEINLINE
|
|||
|
RtlpGetUnusedBytes(PHEAP Heap, PHEAP_ENTRY Block)
|
|||
|
{
|
|||
|
if (Block->UnusedBytes < 0xff) {
|
|||
|
|
|||
|
return Block->UnusedBytes;
|
|||
|
|
|||
|
} else {
|
|||
|
|
|||
|
PSIZE_T UnusedBytesULong = (PSIZE_T)(Block + RtlpGetAllocationUnits(Heap, Block));
|
|||
|
UnusedBytesULong -= 1;
|
|||
|
|
|||
|
return (*UnusedBytesULong);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
VOID
|
|||
|
RtlpInitializeLowFragHeapManager();
|
|||
|
|
|||
|
HANDLE
|
|||
|
FASTCALL
|
|||
|
RtlpCreateLowFragHeap(
|
|||
|
HANDLE Heap
|
|||
|
);
|
|||
|
|
|||
|
VOID
|
|||
|
FASTCALL
|
|||
|
RtlpDestroyLowFragHeap(
|
|||
|
HANDLE LowFragHeapHandle
|
|||
|
);
|
|||
|
|
|||
|
PVOID
|
|||
|
FASTCALL
|
|||
|
RtlpLowFragHeapAlloc(
|
|||
|
HANDLE LowFragHeapHandle,
|
|||
|
SIZE_T BlockSize
|
|||
|
);
|
|||
|
|
|||
|
BOOLEAN
|
|||
|
FASTCALL
|
|||
|
RtlpLowFragHeapFree(
|
|||
|
HANDLE LowFragHeapHandle,
|
|||
|
PVOID p
|
|||
|
);
|
|||
|
|
|||
|
NTSTATUS
|
|||
|
RtlpActivateLowFragmentationHeap(
|
|||
|
IN PVOID HeapHandle
|
|||
|
);
|
|||
|
|
|||
|
ULONG
|
|||
|
FASTCALL
|
|||
|
RtlpLowFragHeapMultipleAlloc(
|
|||
|
HANDLE LowFragHeapHandle,
|
|||
|
ULONG Flags,
|
|||
|
SIZE_T BlockSize,
|
|||
|
ULONG BlockCount,
|
|||
|
PVOID * Pointers
|
|||
|
);
|
|||
|
|
|||
|
ULONG
|
|||
|
FASTCALL
|
|||
|
RtlpLowFragHeapMultipleFree(
|
|||
|
HANDLE LowFragHeapHandle,
|
|||
|
ULONG Flags,
|
|||
|
ULONG BlockCount,
|
|||
|
PVOID * Pointers
|
|||
|
);
|
|||
|
|
|||
|
#else // NTOS_KERNEL_RUNTIME
|
|||
|
|
|||
|
//
|
|||
|
// The kernel mode heap does not ajdust the heap granularity
|
|||
|
// therefore the unused bytes always fit the UCHAR.
|
|||
|
// No need to check for overflow here
|
|||
|
//
|
|||
|
|
|||
|
ULONG
|
|||
|
FORCEINLINE
|
|||
|
RtlpGetAllocationUnits(
|
|||
|
PHEAP Heap,
|
|||
|
PHEAP_ENTRY Block
|
|||
|
)
|
|||
|
{
|
|||
|
return Block->Size;
|
|||
|
}
|
|||
|
|
|||
|
VOID
|
|||
|
FORCEINLINE
|
|||
|
RtlpSetUnusedBytes(PHEAP Heap, PHEAP_ENTRY Block, SIZE_T UnusedBytes)
|
|||
|
{
|
|||
|
Block->UnusedBytes = (UCHAR)(UnusedBytes);
|
|||
|
}
|
|||
|
|
|||
|
SIZE_T
|
|||
|
FORCEINLINE
|
|||
|
RtlpGetUnusedBytes(PHEAP Heap, PHEAP_ENTRY Block)
|
|||
|
{
|
|||
|
return Block->UnusedBytes;
|
|||
|
}
|
|||
|
|
|||
|
UCHAR
|
|||
|
FORCEINLINE
|
|||
|
RtlpGetSmallTagIndex(
|
|||
|
IN PHEAP Heap,
|
|||
|
IN PVOID HeapEntry )
|
|||
|
{
|
|||
|
return ((PHEAP_ENTRY)HeapEntry)->SmallTagIndex;
|
|||
|
}
|
|||
|
|
|||
|
VOID
|
|||
|
FORCEINLINE
|
|||
|
RtlpSetSmallTagIndex(
|
|||
|
IN PHEAP Heap,
|
|||
|
IN PVOID HeapEntry,
|
|||
|
IN UCHAR SmallTagIndex
|
|||
|
)
|
|||
|
{
|
|||
|
((PHEAP_ENTRY)HeapEntry)->SmallTagIndex = SmallTagIndex;
|
|||
|
}
|
|||
|
|
|||
|
#define RtlpQuickValidateBlock(_x_, _y_) (TRUE)
|
|||
|
|
|||
|
#endif // NTOS_KERNEL_RUNTIME
|
|||
|
|
|||
|
#endif // _RTL_HEAP_PRIVATE_
|