1459 lines
45 KiB
C
Raw Normal View History

2001-01-01 00:00:00 +01:00
/*++
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_