4094 lines
108 KiB
C
4094 lines
108 KiB
C
|
/*++
|
|||
|
|
|||
|
Copyright (c) 1989-1995 Microsoft Corporation
|
|||
|
|
|||
|
Module Name:
|
|||
|
|
|||
|
handle.c
|
|||
|
|
|||
|
Abstract:
|
|||
|
|
|||
|
This module implements a set of functions for supporting handles.
|
|||
|
|
|||
|
Author:
|
|||
|
|
|||
|
Steve Wood (stevewo) 25-Apr-1989
|
|||
|
David N. Cutler (davec) 17-May-1995 (rewrite)
|
|||
|
Gary Kimura (GaryKi) 9-Dec-1997 (rerewrite)
|
|||
|
|
|||
|
Adrian Marinescu (adrmarin) 24-May-2000
|
|||
|
Support dynamic changes to the number of levels we use. The code
|
|||
|
performs the best for typical handle table sizes and scales better.
|
|||
|
|
|||
|
Neill Clift (NeillC) 24-Jul-2000
|
|||
|
Make the handle allocate, free and duplicate paths mostly lock free except
|
|||
|
for the lock entry locks, table expansion and locks to solve the A-B-A problem.
|
|||
|
|
|||
|
Revision History:
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
#include "exp.h"
|
|||
|
#pragma hdrstop
|
|||
|
|
|||
|
|
|||
|
//
|
|||
|
// Local constants and support routines
|
|||
|
//
|
|||
|
|
|||
|
//
|
|||
|
// Define global structures that link all handle tables together except the
|
|||
|
// ones where the user has called RemoveHandleTable
|
|||
|
//
|
|||
|
|
|||
|
|
|||
|
#if !DBG // Make this a const varible so its optimized away on free
|
|||
|
const
|
|||
|
#endif
|
|||
|
BOOLEAN ExTraceAllTables = FALSE;
|
|||
|
|
|||
|
EX_PUSH_LOCK HandleTableListLock;
|
|||
|
|
|||
|
ULONG TotalTraceBuffers = 0;
|
|||
|
|
|||
|
#ifdef ALLOC_PRAGMA
|
|||
|
#pragma data_seg("PAGED")
|
|||
|
#endif
|
|||
|
|
|||
|
LIST_ENTRY HandleTableListHead;
|
|||
|
|
|||
|
#ifdef ALLOC_PRAGMA
|
|||
|
#pragma data_seg()
|
|||
|
#endif
|
|||
|
|
|||
|
#if DBG
|
|||
|
#define EXHANDLE_EXTRA_CHECKS 0
|
|||
|
#else
|
|||
|
#define EXHANDLE_EXTRA_CHECKS 0
|
|||
|
#endif
|
|||
|
|
|||
|
#if EXHANDLE_EXTRA_CHECKS
|
|||
|
|
|||
|
#define EXASSERT( exp ) \
|
|||
|
((!(exp)) ? \
|
|||
|
(RtlAssert( #exp, __FILE__, __LINE__, NULL ),FALSE) : \
|
|||
|
TRUE)
|
|||
|
|
|||
|
#else
|
|||
|
|
|||
|
#define EXASSERT ASSERT
|
|||
|
|
|||
|
#endif
|
|||
|
|
|||
|
//
|
|||
|
// This is the sign low bit used to lock handle table entries
|
|||
|
//
|
|||
|
|
|||
|
#define EXHANDLE_TABLE_ENTRY_LOCK_BIT 1
|
|||
|
|
|||
|
#define EX_ADDITIONAL_INFO_SIGNATURE (-2)
|
|||
|
|
|||
|
#define ExpIsValidObjectEntry(Entry) \
|
|||
|
( (Entry != NULL) && (Entry->Object != NULL) && (Entry->NextFreeTableEntry != EX_ADDITIONAL_INFO_SIGNATURE) )
|
|||
|
|
|||
|
|
|||
|
#define TABLE_PAGE_SIZE PAGE_SIZE
|
|||
|
|
|||
|
//
|
|||
|
// Absolute maximum number of handles allowed
|
|||
|
//
|
|||
|
#define MAX_HANDLES (1<<24)
|
|||
|
|
|||
|
#if EXHANDLE_EXTRA_CHECKS
|
|||
|
|
|||
|
//
|
|||
|
// Mask for next free value from the free lists.
|
|||
|
//
|
|||
|
#define FREE_HANDLE_MASK ((MAX_HANDLES<<2) - 1)
|
|||
|
|
|||
|
#else
|
|||
|
|
|||
|
//
|
|||
|
// When no checks compiled in this gets optimized away
|
|||
|
//
|
|||
|
#define FREE_HANDLE_MASK 0xFFFFFFFF
|
|||
|
|
|||
|
#endif
|
|||
|
|
|||
|
//
|
|||
|
// Mask for the free list sequence number
|
|||
|
//
|
|||
|
#define FREE_SEQ_MASK (0xFFFFFFFF & ~FREE_HANDLE_MASK)
|
|||
|
|
|||
|
|
|||
|
#if (FREE_HANDLE_MASK == 0xFFFFFFFF)
|
|||
|
#define FREE_SEQ_INC 0
|
|||
|
#define GetNextSeq() 0
|
|||
|
#else
|
|||
|
//
|
|||
|
// Increment value to progress the sequence number
|
|||
|
//
|
|||
|
#define FREE_SEQ_INC (FREE_HANDLE_MASK + 1)
|
|||
|
ULONG CurrentSeq = 0;
|
|||
|
#define GetNextSeq() (CurrentSeq += FREE_SEQ_INC)
|
|||
|
#endif
|
|||
|
|
|||
|
|
|||
|
|
|||
|
#define LOWLEVEL_COUNT (TABLE_PAGE_SIZE / sizeof(HANDLE_TABLE_ENTRY))
|
|||
|
#define MIDLEVEL_COUNT (PAGE_SIZE / sizeof(PHANDLE_TABLE_ENTRY))
|
|||
|
#define HIGHLEVEL_COUNT MAX_HANDLES / (LOWLEVEL_COUNT * MIDLEVEL_COUNT)
|
|||
|
|
|||
|
#define LOWLEVEL_THRESHOLD LOWLEVEL_COUNT
|
|||
|
#define MIDLEVEL_THRESHOLD (MIDLEVEL_COUNT * LOWLEVEL_COUNT)
|
|||
|
#define HIGHLEVEL_THRESHOLD (MIDLEVEL_COUNT * MIDLEVEL_COUNT * LOWLEVEL_COUNT)
|
|||
|
|
|||
|
#define HIGHLEVEL_SIZE (HIGHLEVEL_COUNT * sizeof (PHANDLE_TABLE_ENTRY))
|
|||
|
|
|||
|
#define LEVEL_CODE_MASK 3
|
|||
|
|
|||
|
//
|
|||
|
// Local support routines
|
|||
|
//
|
|||
|
|
|||
|
PHANDLE_TABLE
|
|||
|
ExpAllocateHandleTable (
|
|||
|
IN PEPROCESS Process OPTIONAL,
|
|||
|
IN BOOLEAN DoInit
|
|||
|
);
|
|||
|
|
|||
|
VOID
|
|||
|
ExpFreeHandleTable (
|
|||
|
IN PHANDLE_TABLE HandleTable
|
|||
|
);
|
|||
|
|
|||
|
BOOLEAN
|
|||
|
ExpAllocateHandleTableEntrySlow (
|
|||
|
IN PHANDLE_TABLE HandleTable,
|
|||
|
IN BOOLEAN DoInit
|
|||
|
);
|
|||
|
|
|||
|
PHANDLE_TABLE_ENTRY
|
|||
|
ExpAllocateHandleTableEntry (
|
|||
|
IN PHANDLE_TABLE HandleTable,
|
|||
|
OUT PEXHANDLE Handle
|
|||
|
);
|
|||
|
|
|||
|
VOID
|
|||
|
ExpFreeHandleTableEntry (
|
|||
|
IN PHANDLE_TABLE HandleTable,
|
|||
|
IN EXHANDLE Handle,
|
|||
|
IN PHANDLE_TABLE_ENTRY HandleTableEntry
|
|||
|
);
|
|||
|
|
|||
|
PHANDLE_TABLE_ENTRY
|
|||
|
ExpLookupHandleTableEntry (
|
|||
|
IN PHANDLE_TABLE HandleTable,
|
|||
|
IN EXHANDLE Handle
|
|||
|
);
|
|||
|
|
|||
|
PHANDLE_TABLE_ENTRY *
|
|||
|
ExpAllocateMidLevelTable (
|
|||
|
IN PHANDLE_TABLE HandleTable,
|
|||
|
IN BOOLEAN DoInit,
|
|||
|
OUT PHANDLE_TABLE_ENTRY *pNewLowLevel
|
|||
|
);
|
|||
|
|
|||
|
PVOID
|
|||
|
ExpAllocateTablePagedPool (
|
|||
|
IN PEPROCESS QuotaProcess OPTIONAL,
|
|||
|
IN SIZE_T NumberOfBytes
|
|||
|
);
|
|||
|
|
|||
|
VOID
|
|||
|
ExpFreeTablePagedPool (
|
|||
|
IN PEPROCESS QuotaProcess OPTIONAL,
|
|||
|
IN PVOID PoolMemory,
|
|||
|
IN SIZE_T NumberOfBytes
|
|||
|
);
|
|||
|
|
|||
|
PHANDLE_TABLE_ENTRY
|
|||
|
ExpAllocateLowLevelTable (
|
|||
|
IN PHANDLE_TABLE HandleTable,
|
|||
|
IN BOOLEAN DoInit
|
|||
|
);
|
|||
|
|
|||
|
VOID
|
|||
|
ExpFreeLowLevelTable (
|
|||
|
IN PEPROCESS QuotaProcess,
|
|||
|
IN PHANDLE_TABLE_ENTRY TableLevel1
|
|||
|
);
|
|||
|
|
|||
|
VOID
|
|||
|
ExpBlockOnLockedHandleEntry (
|
|||
|
PHANDLE_TABLE HandleTable,
|
|||
|
PHANDLE_TABLE_ENTRY HandleTableEntry
|
|||
|
);
|
|||
|
|
|||
|
ULONG
|
|||
|
ExpMoveFreeHandles (
|
|||
|
IN PHANDLE_TABLE HandleTable
|
|||
|
);
|
|||
|
|
|||
|
VOID
|
|||
|
ExpUpdateDebugInfo(
|
|||
|
PHANDLE_TABLE HandleTable,
|
|||
|
PETHREAD CurrentThread,
|
|||
|
HANDLE Handle,
|
|||
|
ULONG Type
|
|||
|
);
|
|||
|
|
|||
|
PVOID
|
|||
|
ExpAllocateTablePagedPoolNoZero (
|
|||
|
IN PEPROCESS QuotaProcess OPTIONAL,
|
|||
|
IN SIZE_T NumberOfBytes
|
|||
|
);
|
|||
|
|
|||
|
|
|||
|
#ifdef ALLOC_PRAGMA
|
|||
|
#pragma alloc_text(INIT, ExInitializeHandleTablePackage)
|
|||
|
#pragma alloc_text(INIT, ExSetHandleTableStrictFIFO)
|
|||
|
#pragma alloc_text(PAGE, ExUnlockHandleTableEntry)
|
|||
|
#pragma alloc_text(PAGE, ExCreateHandleTable)
|
|||
|
#pragma alloc_text(PAGE, ExRemoveHandleTable)
|
|||
|
#pragma alloc_text(PAGE, ExDestroyHandleTable)
|
|||
|
#pragma alloc_text(PAGE, ExEnumHandleTable)
|
|||
|
#pragma alloc_text(PAGE, ExDupHandleTable)
|
|||
|
#pragma alloc_text(PAGE, ExSnapShotHandleTables)
|
|||
|
#pragma alloc_text(PAGE, ExCreateHandle)
|
|||
|
#pragma alloc_text(PAGE, ExDestroyHandle)
|
|||
|
#pragma alloc_text(PAGE, ExChangeHandle)
|
|||
|
#pragma alloc_text(PAGE, ExMapHandleToPointer)
|
|||
|
#pragma alloc_text(PAGE, ExMapHandleToPointerEx)
|
|||
|
#pragma alloc_text(PAGE, ExpAllocateHandleTable)
|
|||
|
#pragma alloc_text(PAGE, ExpFreeHandleTable)
|
|||
|
#pragma alloc_text(PAGE, ExpAllocateHandleTableEntry)
|
|||
|
#pragma alloc_text(PAGE, ExpAllocateHandleTableEntrySlow)
|
|||
|
#pragma alloc_text(PAGE, ExpFreeHandleTableEntry)
|
|||
|
#pragma alloc_text(PAGE, ExpLookupHandleTableEntry)
|
|||
|
#pragma alloc_text(PAGE, ExSweepHandleTable)
|
|||
|
#pragma alloc_text(PAGE, ExpAllocateMidLevelTable)
|
|||
|
#pragma alloc_text(PAGE, ExpAllocateTablePagedPool)
|
|||
|
#pragma alloc_text(PAGE, ExpAllocateTablePagedPoolNoZero)
|
|||
|
#pragma alloc_text(PAGE, ExpFreeTablePagedPool)
|
|||
|
#pragma alloc_text(PAGE, ExpAllocateLowLevelTable)
|
|||
|
#pragma alloc_text(PAGE, ExSetHandleInfo)
|
|||
|
#pragma alloc_text(PAGE, ExpGetHandleInfo)
|
|||
|
#pragma alloc_text(PAGE, ExSnapShotHandleTablesEx)
|
|||
|
#pragma alloc_text(PAGE, ExpFreeLowLevelTable)
|
|||
|
#pragma alloc_text(PAGE, ExpBlockOnLockedHandleEntry)
|
|||
|
#pragma alloc_text(PAGE, ExpMoveFreeHandles)
|
|||
|
#pragma alloc_text(PAGE, ExEnableHandleTracing)
|
|||
|
#pragma alloc_text(PAGE, ExDereferenceHandleDebugInfo)
|
|||
|
#pragma alloc_text(PAGE, ExReferenceHandleDebugInfo)
|
|||
|
#pragma alloc_text(PAGE, ExpUpdateDebugInfo)
|
|||
|
#endif
|
|||
|
|
|||
|
//
|
|||
|
// Define macros to lock and unlock the handle table.
|
|||
|
// We use this lock only for handle table expansion.
|
|||
|
//
|
|||
|
#define ExpLockHandleTableExclusive(xxHandleTable,xxCurrentThread) { \
|
|||
|
KeEnterCriticalRegionThread (xxCurrentThread); \
|
|||
|
ExAcquirePushLockExclusive (&xxHandleTable->HandleTableLock[0]); \
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
#define ExpUnlockHandleTableExclusive(xxHandleTable,xxCurrentThread) { \
|
|||
|
ExReleasePushLockExclusive (&xxHandleTable->HandleTableLock[0]); \
|
|||
|
KeLeaveCriticalRegionThread (xxCurrentThread); \
|
|||
|
}
|
|||
|
|
|||
|
#define ExpLockHandleTableShared(xxHandleTable,xxCurrentThread,xxIdx) { \
|
|||
|
KeEnterCriticalRegionThread (xxCurrentThread); \
|
|||
|
ExAcquirePushLockShared (&xxHandleTable->HandleTableLock[xxIdx]); \
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
#define ExpUnlockHandleTableShared(xxHandleTable,xxCurrentThread,xxIdx) { \
|
|||
|
ExReleasePushLockShared (&xxHandleTable->HandleTableLock[xxIdx]); \
|
|||
|
KeLeaveCriticalRegionThread (xxCurrentThread); \
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
|
|||
|
FORCEINLINE
|
|||
|
ULONG
|
|||
|
ExpInterlockedExchange (
|
|||
|
IN OUT PULONG Index,
|
|||
|
IN ULONG FirstIndex,
|
|||
|
IN PHANDLE_TABLE_ENTRY Entry
|
|||
|
)
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
This performs the following steps:
|
|||
|
1. Set Entry->NextFreeTableEntry = *Index
|
|||
|
2. Loops until *Index == (the value of *Index when we entered the function)
|
|||
|
When they're equal, we set *Index = FirstIndex
|
|||
|
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
Index - Points to the ULONG we want to set.
|
|||
|
|
|||
|
FirstIndex - New value to set Index to.
|
|||
|
|
|||
|
Entry - TableEntry that will get the initial value of *Index before it's
|
|||
|
updated.
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
New value of *Index (i.e. FirstIndex).
|
|||
|
|
|||
|
--*/
|
|||
|
{
|
|||
|
ULONG OldIndex, NewIndex;
|
|||
|
|
|||
|
EXASSERT (Entry->Object == NULL);
|
|||
|
|
|||
|
//
|
|||
|
// Load new value and generate the sequence number on pushes
|
|||
|
//
|
|||
|
|
|||
|
NewIndex = FirstIndex + GetNextSeq();
|
|||
|
|
|||
|
while (1) {
|
|||
|
|
|||
|
//
|
|||
|
// remember original value and
|
|||
|
// archive it in NextFreeTableEntry.
|
|||
|
//
|
|||
|
|
|||
|
OldIndex = *Index;
|
|||
|
Entry->NextFreeTableEntry = OldIndex;
|
|||
|
|
|||
|
|
|||
|
//
|
|||
|
// Swap in the new value, and if the swap occurs
|
|||
|
// successfully, we're done.
|
|||
|
//
|
|||
|
if (OldIndex == (ULONG) InterlockedCompareExchange ((PLONG)Index,
|
|||
|
NewIndex,
|
|||
|
OldIndex)) {
|
|||
|
return OldIndex;
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
ULONG
|
|||
|
ExpMoveFreeHandles (
|
|||
|
IN PHANDLE_TABLE HandleTable
|
|||
|
)
|
|||
|
{
|
|||
|
ULONG OldValue, NewValue;
|
|||
|
ULONG Index, OldIndex, NewIndex, FreeSize;
|
|||
|
PHANDLE_TABLE_ENTRY Entry, FirstEntry;
|
|||
|
EXHANDLE Handle;
|
|||
|
ULONG Idx;
|
|||
|
BOOLEAN StrictFIFO;
|
|||
|
|
|||
|
//
|
|||
|
// First remove all the handles from the free list so we can add them to the
|
|||
|
// list we use for allocates.
|
|||
|
//
|
|||
|
|
|||
|
OldValue = InterlockedExchange ((PLONG)&HandleTable->LastFree,
|
|||
|
0);
|
|||
|
Index = OldValue;
|
|||
|
if (Index == 0) {
|
|||
|
//
|
|||
|
// There are no free handles. Nothing to do.
|
|||
|
//
|
|||
|
return OldValue;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
//
|
|||
|
// We are pushing old entries onto the free list.
|
|||
|
// We have the A-B-A problem here as these items may have been moved here because
|
|||
|
// another thread was using them in the pop code.
|
|||
|
//
|
|||
|
for (Idx = 1; Idx < HANDLE_TABLE_LOCKS; Idx++) {
|
|||
|
ExAcquireReleasePushLockExclusive (&HandleTable->HandleTableLock[Idx]);
|
|||
|
}
|
|||
|
StrictFIFO = HandleTable->StrictFIFO;
|
|||
|
|
|||
|
//
|
|||
|
// If we are strict FIFO then reverse the list to make handle reuse rare.
|
|||
|
//
|
|||
|
if (!StrictFIFO) {
|
|||
|
//
|
|||
|
// We have a complete chain here. If there is no existing chain we
|
|||
|
// can just push this one without any hassles. If we can't then
|
|||
|
// we can just fall into the reversing code anyway as we need
|
|||
|
// to find the end of the chain to continue it.
|
|||
|
//
|
|||
|
|
|||
|
//
|
|||
|
// This is a push so create a new sequence number
|
|||
|
//
|
|||
|
|
|||
|
if (InterlockedCompareExchange ((PLONG)&HandleTable->FirstFree,
|
|||
|
OldValue + GetNextSeq(),
|
|||
|
0) == 0) {
|
|||
|
return OldValue;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// Loop over all the entries and reverse the chain.
|
|||
|
//
|
|||
|
FreeSize = OldIndex = 0;
|
|||
|
FirstEntry = NULL;
|
|||
|
while (1) {
|
|||
|
FreeSize++;
|
|||
|
Handle.Value = Index;
|
|||
|
Entry = ExpLookupHandleTableEntry (HandleTable, Handle);
|
|||
|
|
|||
|
EXASSERT (Entry->Object == NULL);
|
|||
|
|
|||
|
NewIndex = Entry->NextFreeTableEntry;
|
|||
|
Entry->NextFreeTableEntry = OldIndex;
|
|||
|
if (OldIndex == 0) {
|
|||
|
FirstEntry = Entry;
|
|||
|
}
|
|||
|
OldIndex = Index;
|
|||
|
if (NewIndex == 0) {
|
|||
|
break;
|
|||
|
}
|
|||
|
Index = NewIndex;
|
|||
|
}
|
|||
|
|
|||
|
NewValue = ExpInterlockedExchange (&HandleTable->FirstFree,
|
|||
|
OldIndex,
|
|||
|
FirstEntry);
|
|||
|
|
|||
|
//
|
|||
|
// If we haven't got a pool of a few handles then force
|
|||
|
// table expansion to keep the free handle size high
|
|||
|
//
|
|||
|
if (FreeSize < 100 && StrictFIFO) {
|
|||
|
OldValue = 0;
|
|||
|
}
|
|||
|
return OldValue;
|
|||
|
}
|
|||
|
|
|||
|
PHANDLE_TABLE_ENTRY
|
|||
|
ExpAllocateHandleTableEntry (
|
|||
|
IN PHANDLE_TABLE HandleTable,
|
|||
|
OUT PEXHANDLE pHandle
|
|||
|
)
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
This routine does a fast allocate of a free handle. It's lock free if
|
|||
|
possible.
|
|||
|
|
|||
|
Only the rare case of handle table expansion is covered by the handle
|
|||
|
table lock.
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
HandleTable - Supplies the handle table being allocated from.
|
|||
|
|
|||
|
pHandle - Handle returned
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
PHANDLE_TABLE_ENTRY - The allocated handle table entry pointer or NULL
|
|||
|
on failure.
|
|||
|
|
|||
|
--*/
|
|||
|
{
|
|||
|
PKTHREAD CurrentThread;
|
|||
|
ULONG OldValue, NewValue, NewValue1;
|
|||
|
PHANDLE_TABLE_ENTRY Entry;
|
|||
|
EXHANDLE Handle;
|
|||
|
BOOLEAN RetVal;
|
|||
|
ULONG Idx;
|
|||
|
|
|||
|
|
|||
|
CurrentThread = KeGetCurrentThread ();
|
|||
|
while (1) {
|
|||
|
|
|||
|
OldValue = HandleTable->FirstFree;
|
|||
|
|
|||
|
|
|||
|
while (OldValue == 0) {
|
|||
|
//
|
|||
|
// Lock the handle table for exclusive access as we will be
|
|||
|
// allocating a new table level.
|
|||
|
//
|
|||
|
ExpLockHandleTableExclusive (HandleTable, CurrentThread);
|
|||
|
|
|||
|
//
|
|||
|
// If we have multiple threads trying to expand the table at
|
|||
|
// the same time then by just acquiring the table lock we
|
|||
|
// force those threads to complete their allocations and
|
|||
|
// populate the free list. We must check the free list here
|
|||
|
// so we don't expand the list twice without needing to.
|
|||
|
//
|
|||
|
|
|||
|
OldValue = HandleTable->FirstFree;
|
|||
|
if (OldValue != 0) {
|
|||
|
ExpUnlockHandleTableExclusive (HandleTable, CurrentThread);
|
|||
|
break;
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// See if we have any handles on the alternate free list
|
|||
|
// These handles need some locking to move them over.
|
|||
|
//
|
|||
|
OldValue = ExpMoveFreeHandles (HandleTable);
|
|||
|
if (OldValue != 0) {
|
|||
|
ExpUnlockHandleTableExclusive (HandleTable, CurrentThread);
|
|||
|
break;
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// This must be the first thread attempting expansion or all the
|
|||
|
// free handles allocated by another thread got used up in the gap.
|
|||
|
//
|
|||
|
|
|||
|
RetVal = ExpAllocateHandleTableEntrySlow (HandleTable, TRUE);
|
|||
|
|
|||
|
ExpUnlockHandleTableExclusive (HandleTable, CurrentThread);
|
|||
|
|
|||
|
|
|||
|
OldValue = HandleTable->FirstFree;
|
|||
|
|
|||
|
//
|
|||
|
// If ExpAllocateHandleTableEntrySlow had a failed allocation
|
|||
|
// then we want to fail the call. We check for free entries
|
|||
|
// before we exit just in case they got allocated or freed by
|
|||
|
// somebody else in the gap.
|
|||
|
//
|
|||
|
|
|||
|
if (!RetVal) {
|
|||
|
if (OldValue == 0) {
|
|||
|
pHandle->GenericHandleOverlay = NULL;
|
|||
|
return NULL;
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
Handle.Value = (OldValue & FREE_HANDLE_MASK);
|
|||
|
|
|||
|
Entry = ExpLookupHandleTableEntry (HandleTable, Handle);
|
|||
|
|
|||
|
Idx = ((OldValue & FREE_HANDLE_MASK)>>2) % HANDLE_TABLE_LOCKS;
|
|||
|
ExpLockHandleTableShared (HandleTable, CurrentThread, Idx);
|
|||
|
|
|||
|
if (OldValue != *(volatile ULONG *) &HandleTable->FirstFree) {
|
|||
|
ExpUnlockHandleTableShared (HandleTable, CurrentThread, Idx);
|
|||
|
continue;
|
|||
|
}
|
|||
|
|
|||
|
KeMemoryBarrier ();
|
|||
|
|
|||
|
NewValue = *(volatile ULONG *) &Entry->NextFreeTableEntry;
|
|||
|
|
|||
|
NewValue1 = InterlockedCompareExchange ((PLONG)&HandleTable->FirstFree,
|
|||
|
NewValue,
|
|||
|
OldValue);
|
|||
|
|
|||
|
ExpUnlockHandleTableShared (HandleTable, CurrentThread, Idx);
|
|||
|
|
|||
|
if (NewValue1 == OldValue) {
|
|||
|
EXASSERT ((NewValue & FREE_HANDLE_MASK) < HandleTable->NextHandleNeedingPool);
|
|||
|
break;
|
|||
|
} else {
|
|||
|
//
|
|||
|
// We should have eliminated the A-B-A problem so if only the sequence number has
|
|||
|
// changed we are broken.
|
|||
|
//
|
|||
|
EXASSERT ((NewValue1 & FREE_HANDLE_MASK) != (OldValue & FREE_HANDLE_MASK));
|
|||
|
}
|
|||
|
}
|
|||
|
InterlockedIncrement (&HandleTable->HandleCount);
|
|||
|
|
|||
|
*pHandle = Handle;
|
|||
|
|
|||
|
return Entry;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
VOID
|
|||
|
ExpBlockOnLockedHandleEntry (
|
|||
|
PHANDLE_TABLE HandleTable,
|
|||
|
PHANDLE_TABLE_ENTRY HandleTableEntry
|
|||
|
)
|
|||
|
{
|
|||
|
EX_PUSH_LOCK_WAIT_BLOCK WaitBlock;
|
|||
|
LONG_PTR CurrentValue;
|
|||
|
|
|||
|
//
|
|||
|
// Queue our wait block to be signaled by a releasing thread.
|
|||
|
//
|
|||
|
|
|||
|
ExBlockPushLock (&HandleTable->HandleContentionEvent, &WaitBlock);
|
|||
|
|
|||
|
CurrentValue = HandleTableEntry->Value;
|
|||
|
if (CurrentValue == 0 || (CurrentValue&EXHANDLE_TABLE_ENTRY_LOCK_BIT) != 0) {
|
|||
|
ExUnblockPushLock (&HandleTable->HandleContentionEvent, &WaitBlock);
|
|||
|
} else {
|
|||
|
ExWaitForUnblockPushLock (&HandleTable->HandleContentionEvent, &WaitBlock);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
BOOLEAN
|
|||
|
FORCEINLINE
|
|||
|
ExpLockHandleTableEntry (
|
|||
|
PHANDLE_TABLE HandleTable,
|
|||
|
PHANDLE_TABLE_ENTRY HandleTableEntry
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
This routine locks the specified handle table entry. After the entry is
|
|||
|
locked the sign bit will be set.
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
HandleTable - Supplies the handle table containing the entry being locked.
|
|||
|
|
|||
|
HandleTableEntry - Supplies the handle table entry being locked.
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
TRUE if the entry is valid and locked, and FALSE if the entry is
|
|||
|
marked free.
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
LONG_PTR NewValue;
|
|||
|
LONG_PTR CurrentValue;
|
|||
|
|
|||
|
//
|
|||
|
// We are about to take a lock. Make sure we are protected.
|
|||
|
//
|
|||
|
ASSERT ((KeGetCurrentThread()->CombinedApcDisable != 0) || (KeGetCurrentIrql() == APC_LEVEL));
|
|||
|
|
|||
|
//
|
|||
|
// We'll keep on looping reading in the value, making sure it is not null,
|
|||
|
// and if it is not currently locked we'll try for the lock and return
|
|||
|
// true if we get it. Otherwise we'll pause a bit and then try again.
|
|||
|
//
|
|||
|
|
|||
|
|
|||
|
while (TRUE) {
|
|||
|
|
|||
|
CurrentValue = *((volatile LONG_PTR *)&HandleTableEntry->Object);
|
|||
|
|
|||
|
//
|
|||
|
// If the handle value is greater than zero then it is not currently
|
|||
|
// locked and we should try for the lock, by setting the lock bit and
|
|||
|
// doing an interlocked exchange.
|
|||
|
//
|
|||
|
|
|||
|
if (CurrentValue & EXHANDLE_TABLE_ENTRY_LOCK_BIT) {
|
|||
|
|
|||
|
//
|
|||
|
// Remove the
|
|||
|
//
|
|||
|
NewValue = CurrentValue - EXHANDLE_TABLE_ENTRY_LOCK_BIT;
|
|||
|
|
|||
|
if ((LONG_PTR)(InterlockedCompareExchangePointer (&HandleTableEntry->Object,
|
|||
|
(PVOID)NewValue,
|
|||
|
(PVOID)CurrentValue)) == CurrentValue) {
|
|||
|
|
|||
|
return TRUE;
|
|||
|
}
|
|||
|
} else {
|
|||
|
//
|
|||
|
// Make sure the handle table entry is not freed
|
|||
|
//
|
|||
|
|
|||
|
if (CurrentValue == 0) {
|
|||
|
|
|||
|
return FALSE;
|
|||
|
}
|
|||
|
}
|
|||
|
ExpBlockOnLockedHandleEntry (HandleTable, HandleTableEntry);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
NTKERNELAPI
|
|||
|
VOID
|
|||
|
FORCEINLINE
|
|||
|
ExUnlockHandleTableEntry (
|
|||
|
PHANDLE_TABLE HandleTable,
|
|||
|
PHANDLE_TABLE_ENTRY HandleTableEntry
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
This routine unlocks the specified handle table entry. After the entry is
|
|||
|
unlocked the sign bit will be clear.
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
HandleTable - Supplies the handle table containing the entry being unlocked.
|
|||
|
|
|||
|
HandleTableEntry - Supplies the handle table entry being unlocked.
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
None.
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
LONG_PTR OldValue;
|
|||
|
|
|||
|
PAGED_CODE();
|
|||
|
|
|||
|
//
|
|||
|
// We are about to release a lock. Make sure we are protected from suspension.
|
|||
|
//
|
|||
|
ASSERT ((KeGetCurrentThread()->CombinedApcDisable != 0) || (KeGetCurrentIrql() == APC_LEVEL));
|
|||
|
|
|||
|
//
|
|||
|
// This routine does not need to loop and attempt the unlock opeation more
|
|||
|
// than once because by definition the caller has the entry already locked
|
|||
|
// and no one can be changing the value without the lock.
|
|||
|
//
|
|||
|
|
|||
|
|
|||
|
#if defined (_WIN64)
|
|||
|
|
|||
|
OldValue = InterlockedExchangeAdd64 ((PLONGLONG) &HandleTableEntry->Value, EXHANDLE_TABLE_ENTRY_LOCK_BIT);
|
|||
|
|
|||
|
|
|||
|
#else
|
|||
|
|
|||
|
OldValue = InterlockedOr ((LONG *) &HandleTableEntry->Value, EXHANDLE_TABLE_ENTRY_LOCK_BIT);
|
|||
|
|
|||
|
|
|||
|
#endif
|
|||
|
|
|||
|
EXASSERT ((OldValue&EXHANDLE_TABLE_ENTRY_LOCK_BIT) == 0);
|
|||
|
|
|||
|
//
|
|||
|
// Unblock any waiters waiting for this table entry.
|
|||
|
//
|
|||
|
ExUnblockPushLock (&HandleTable->HandleContentionEvent, NULL);
|
|||
|
|
|||
|
return;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
NTKERNELAPI
|
|||
|
VOID
|
|||
|
ExInitializeHandleTablePackage (
|
|||
|
VOID
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
This routine is called once at system initialization to setup the ex handle
|
|||
|
table package
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
None.
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
None.
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
//
|
|||
|
// Initialize the handle table synchronization resource and list head
|
|||
|
//
|
|||
|
|
|||
|
InitializeListHead( &HandleTableListHead );
|
|||
|
ExInitializePushLock( &HandleTableListLock );
|
|||
|
|
|||
|
return;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
NTKERNELAPI
|
|||
|
PHANDLE_TABLE
|
|||
|
ExCreateHandleTable (
|
|||
|
IN struct _EPROCESS *Process OPTIONAL
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
This function allocate and initialize a new new handle table
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
Process - Supplies an optional pointer to the process against which quota
|
|||
|
will be charged.
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
If a handle table is successfully created, then the address of the
|
|||
|
handle table is returned as the function value. Otherwize, a value
|
|||
|
NULL is returned.
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
PKTHREAD CurrentThread;
|
|||
|
PHANDLE_TABLE HandleTable;
|
|||
|
|
|||
|
PAGED_CODE();
|
|||
|
|
|||
|
CurrentThread = KeGetCurrentThread ();
|
|||
|
|
|||
|
//
|
|||
|
// Allocate and initialize a handle table descriptor
|
|||
|
//
|
|||
|
|
|||
|
HandleTable = ExpAllocateHandleTable( Process, TRUE );
|
|||
|
|
|||
|
if (HandleTable == NULL) {
|
|||
|
return NULL;
|
|||
|
}
|
|||
|
//
|
|||
|
// Insert the handle table in the handle table list.
|
|||
|
//
|
|||
|
|
|||
|
KeEnterCriticalRegionThread (CurrentThread);
|
|||
|
ExAcquirePushLockExclusive( &HandleTableListLock );
|
|||
|
|
|||
|
InsertTailList( &HandleTableListHead, &HandleTable->HandleTableList );
|
|||
|
|
|||
|
ExReleasePushLockExclusive( &HandleTableListLock );
|
|||
|
KeLeaveCriticalRegionThread (CurrentThread);
|
|||
|
|
|||
|
|
|||
|
//
|
|||
|
// And return to our caller
|
|||
|
//
|
|||
|
|
|||
|
return HandleTable;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
NTKERNELAPI
|
|||
|
VOID
|
|||
|
ExRemoveHandleTable (
|
|||
|
IN PHANDLE_TABLE HandleTable
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
This function removes the specified exhandle table from the list of
|
|||
|
exhandle tables. Used by PS and ATOM packages to make sure their handle
|
|||
|
tables are not in the list enumerated by the ExSnapShotHandleTables
|
|||
|
routine and the !handle debugger extension.
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
HandleTable - Supplies a pointer to a handle table
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
None.
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
PKTHREAD CurrentThread;
|
|||
|
|
|||
|
PAGED_CODE();
|
|||
|
|
|||
|
CurrentThread = KeGetCurrentThread ();
|
|||
|
|
|||
|
//
|
|||
|
// First, acquire the global handle table lock
|
|||
|
//
|
|||
|
|
|||
|
KeEnterCriticalRegionThread (CurrentThread);
|
|||
|
ExAcquirePushLockExclusive( &HandleTableListLock );
|
|||
|
|
|||
|
//
|
|||
|
// Remove the handle table from the handle table list. This routine is
|
|||
|
// written so that multiple calls to remove a handle table will not
|
|||
|
// corrupt the system.
|
|||
|
//
|
|||
|
|
|||
|
RemoveEntryList( &HandleTable->HandleTableList );
|
|||
|
InitializeListHead( &HandleTable->HandleTableList );
|
|||
|
|
|||
|
//
|
|||
|
// Now release the global lock and return to our caller
|
|||
|
//
|
|||
|
|
|||
|
ExReleasePushLockExclusive( &HandleTableListLock );
|
|||
|
KeLeaveCriticalRegionThread (CurrentThread);
|
|||
|
|
|||
|
return;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
NTKERNELAPI
|
|||
|
VOID
|
|||
|
ExDestroyHandleTable (
|
|||
|
IN PHANDLE_TABLE HandleTable,
|
|||
|
IN EX_DESTROY_HANDLE_ROUTINE DestroyHandleProcedure OPTIONAL
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
This function destroys the specified handle table.
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
HandleTable - Supplies a pointer to a handle table
|
|||
|
|
|||
|
DestroyHandleProcedure - Supplies a pointer to a function to call for each
|
|||
|
valid handle entry in the handle table.
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
None.
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
EXHANDLE Handle;
|
|||
|
PHANDLE_TABLE_ENTRY HandleTableEntry;
|
|||
|
|
|||
|
PAGED_CODE();
|
|||
|
|
|||
|
//
|
|||
|
// Remove the handle table from the handle table list
|
|||
|
//
|
|||
|
|
|||
|
ExRemoveHandleTable( HandleTable );
|
|||
|
|
|||
|
//
|
|||
|
// Iterate through the handle table and for each handle that is allocated
|
|||
|
// we'll invoke the call back. Note that this loop exits when we get a
|
|||
|
// null handle table entry. We know there will be no more possible
|
|||
|
// entries after the first null one is encountered because we allocate
|
|||
|
// memory of the handles in a dense fashion. But first test that we have
|
|||
|
// call back to use
|
|||
|
//
|
|||
|
|
|||
|
if (ARGUMENT_PRESENT(DestroyHandleProcedure)) {
|
|||
|
|
|||
|
for (Handle.Value = 0;
|
|||
|
(HandleTableEntry = ExpLookupHandleTableEntry( HandleTable, Handle )) != NULL;
|
|||
|
Handle.Value += HANDLE_VALUE_INC) {
|
|||
|
|
|||
|
//
|
|||
|
// Only do the callback if the entry is not free
|
|||
|
//
|
|||
|
|
|||
|
if ( ExpIsValidObjectEntry(HandleTableEntry) ) {
|
|||
|
|
|||
|
(*DestroyHandleProcedure)( Handle.GenericHandleOverlay );
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// Now free up the handle table memory and return to our caller
|
|||
|
//
|
|||
|
|
|||
|
ExpFreeHandleTable( HandleTable );
|
|||
|
|
|||
|
return;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
NTKERNELAPI
|
|||
|
VOID
|
|||
|
ExSweepHandleTable (
|
|||
|
IN PHANDLE_TABLE HandleTable,
|
|||
|
IN EX_ENUMERATE_HANDLE_ROUTINE EnumHandleProcedure,
|
|||
|
IN PVOID EnumParameter
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
This function sweeps a handle table in a unsynchronized manner.
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
HandleTable - Supplies a pointer to a handle table
|
|||
|
|
|||
|
EnumHandleProcedure - Supplies a pointer to a fucntion to call for
|
|||
|
each valid handle in the enumerated handle table.
|
|||
|
|
|||
|
EnumParameter - Supplies an uninterpreted 32-bit value that is passed
|
|||
|
to the EnumHandleProcedure each time it is called.
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
None.
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
EXHANDLE Handle;
|
|||
|
PHANDLE_TABLE_ENTRY HandleTableEntry;
|
|||
|
|
|||
|
PAGED_CODE();
|
|||
|
|
|||
|
//
|
|||
|
// Iterate through the handle table and for each handle that is allocated
|
|||
|
// we'll invoke the call back. Note that this loop exits when we get a
|
|||
|
// null handle table entry. We know there will be no more possible
|
|||
|
// entries after the first null one is encountered because we allocate
|
|||
|
// memory of the handles in a dense fashion.
|
|||
|
//
|
|||
|
Handle.Value = HANDLE_VALUE_INC;
|
|||
|
|
|||
|
while ((HandleTableEntry = ExpLookupHandleTableEntry( HandleTable, Handle )) != NULL) {
|
|||
|
|
|||
|
do {
|
|||
|
|
|||
|
//
|
|||
|
// Only do the callback if the entry is not free
|
|||
|
//
|
|||
|
//
|
|||
|
|
|||
|
if (ExpLockHandleTableEntry( HandleTable, HandleTableEntry )) {
|
|||
|
|
|||
|
(*EnumHandleProcedure)( HandleTableEntry,
|
|||
|
Handle.GenericHandleOverlay,
|
|||
|
EnumParameter );
|
|||
|
}
|
|||
|
Handle.Value += HANDLE_VALUE_INC;
|
|||
|
HandleTableEntry++;
|
|||
|
} while ((Handle.Value % (LOWLEVEL_COUNT * HANDLE_VALUE_INC)) != 0);
|
|||
|
// Skip past the first entry that's not a real entry
|
|||
|
Handle.Value += HANDLE_VALUE_INC;
|
|||
|
}
|
|||
|
|
|||
|
return;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
|
|||
|
NTKERNELAPI
|
|||
|
BOOLEAN
|
|||
|
ExEnumHandleTable (
|
|||
|
IN PHANDLE_TABLE HandleTable,
|
|||
|
IN EX_ENUMERATE_HANDLE_ROUTINE EnumHandleProcedure,
|
|||
|
IN PVOID EnumParameter,
|
|||
|
OUT PHANDLE Handle OPTIONAL
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
This function enumerates all the valid handles in a handle table.
|
|||
|
For each valid handle in the handle table, the specified eumeration
|
|||
|
function is called. If the enumeration function returns TRUE, then
|
|||
|
the enumeration is stopped, the current handle is returned to the
|
|||
|
caller via the optional Handle parameter, and this function returns
|
|||
|
TRUE to indicated that the enumeration stopped at a specific handle.
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
HandleTable - Supplies a pointer to a handle table.
|
|||
|
|
|||
|
EnumHandleProcedure - Supplies a pointer to a fucntion to call for
|
|||
|
each valid handle in the enumerated handle table.
|
|||
|
|
|||
|
EnumParameter - Supplies an uninterpreted 32-bit value that is passed
|
|||
|
to the EnumHandleProcedure each time it is called.
|
|||
|
|
|||
|
Handle - Supplies an optional pointer a variable that receives the
|
|||
|
Handle value that the enumeration stopped at. Contents of the
|
|||
|
variable only valid if this function returns TRUE.
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
If the enumeration stopped at a specific handle, then a value of TRUE
|
|||
|
is returned. Otherwise, a value of FALSE is returned.
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
PKTHREAD CurrentThread;
|
|||
|
BOOLEAN ResultValue;
|
|||
|
EXHANDLE LocalHandle;
|
|||
|
PHANDLE_TABLE_ENTRY HandleTableEntry;
|
|||
|
|
|||
|
PAGED_CODE();
|
|||
|
|
|||
|
CurrentThread = KeGetCurrentThread ();
|
|||
|
|
|||
|
//
|
|||
|
// Our initial return value is false until the enumeration callback
|
|||
|
// function tells us otherwise
|
|||
|
//
|
|||
|
|
|||
|
ResultValue = FALSE;
|
|||
|
|
|||
|
//
|
|||
|
// Iterate through the handle table and for each handle that is
|
|||
|
// allocated we'll invoke the call back. Note that this loop exits
|
|||
|
// when we get a null handle table entry. We know there will be no
|
|||
|
// more possible entries after the first null one is encountered
|
|||
|
// because we allocate memory for the handles in a dense fashion
|
|||
|
//
|
|||
|
|
|||
|
KeEnterCriticalRegionThread (CurrentThread);
|
|||
|
|
|||
|
for (LocalHandle.Value = 0; // does essentially the following "LocalHandle.Index = 0, LocalHandle.TagBits = 0;"
|
|||
|
(HandleTableEntry = ExpLookupHandleTableEntry( HandleTable, LocalHandle )) != NULL;
|
|||
|
LocalHandle.Value += HANDLE_VALUE_INC) {
|
|||
|
|
|||
|
//
|
|||
|
// Only do the callback if the entry is not free
|
|||
|
//
|
|||
|
|
|||
|
if ( ExpIsValidObjectEntry( HandleTableEntry ) ) {
|
|||
|
|
|||
|
//
|
|||
|
// Lock the handle table entry because we're about to give
|
|||
|
// it to the callback function, then release the entry
|
|||
|
// right after the call back.
|
|||
|
//
|
|||
|
|
|||
|
if (ExpLockHandleTableEntry( HandleTable, HandleTableEntry )) {
|
|||
|
|
|||
|
//
|
|||
|
// Invoke the callback, and if it returns true then set
|
|||
|
// the proper output values and break out of the loop.
|
|||
|
//
|
|||
|
|
|||
|
ResultValue = (*EnumHandleProcedure)( HandleTableEntry,
|
|||
|
LocalHandle.GenericHandleOverlay,
|
|||
|
EnumParameter );
|
|||
|
|
|||
|
ExUnlockHandleTableEntry( HandleTable, HandleTableEntry );
|
|||
|
|
|||
|
if (ResultValue) {
|
|||
|
if (ARGUMENT_PRESENT( Handle )) {
|
|||
|
|
|||
|
*Handle = LocalHandle.GenericHandleOverlay;
|
|||
|
}
|
|||
|
break;
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
KeLeaveCriticalRegionThread (CurrentThread);
|
|||
|
|
|||
|
|
|||
|
return ResultValue;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
NTKERNELAPI
|
|||
|
PHANDLE_TABLE
|
|||
|
ExDupHandleTable (
|
|||
|
IN struct _EPROCESS *Process OPTIONAL,
|
|||
|
IN PHANDLE_TABLE OldHandleTable,
|
|||
|
IN EX_DUPLICATE_HANDLE_ROUTINE DupHandleProcedure,
|
|||
|
IN ULONG_PTR Mask
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
This function creates a duplicate copy of the specified handle table.
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
Process - Supplies an optional to the process to charge quota to.
|
|||
|
|
|||
|
OldHandleTable - Supplies a pointer to a handle table.
|
|||
|
|
|||
|
DupHandleProcedure - Supplies an optional pointer to a function to call
|
|||
|
for each valid handle in the duplicated handle table.
|
|||
|
|
|||
|
Mask - Mask applied to the object pointer to work outif we need to duplicate
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
If the specified handle table is successfully duplicated, then the
|
|||
|
address of the new handle table is returned as the function value.
|
|||
|
Otherwize, a value NULL is returned.
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
PKTHREAD CurrentThread;
|
|||
|
PHANDLE_TABLE NewHandleTable;
|
|||
|
EXHANDLE Handle;
|
|||
|
PHANDLE_TABLE_ENTRY OldHandleTableEntry;
|
|||
|
PHANDLE_TABLE_ENTRY NewHandleTableEntry;
|
|||
|
BOOLEAN FreeEntry;
|
|||
|
NTSTATUS Status;
|
|||
|
|
|||
|
PAGED_CODE();
|
|||
|
|
|||
|
CurrentThread = KeGetCurrentThread ();
|
|||
|
|
|||
|
//
|
|||
|
// First allocate a new handle table. If this fails then
|
|||
|
// return immediately to our caller
|
|||
|
//
|
|||
|
|
|||
|
NewHandleTable = ExpAllocateHandleTable( Process, FALSE );
|
|||
|
|
|||
|
if (NewHandleTable == NULL) {
|
|||
|
|
|||
|
return NULL;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
//
|
|||
|
// Now we'll build up the new handle table. We do this by calling
|
|||
|
// allocating new handle table entries, and "fooling" the worker
|
|||
|
// routine to allocate keep on allocating until the next free
|
|||
|
// index needing pool are equal
|
|||
|
//
|
|||
|
while (NewHandleTable->NextHandleNeedingPool < OldHandleTable->NextHandleNeedingPool) {
|
|||
|
|
|||
|
//
|
|||
|
// Call the worker routine to grow the new handle table. If
|
|||
|
// not successful then free the new table as far as we got,
|
|||
|
// set our output variable and exit out here
|
|||
|
//
|
|||
|
if (!ExpAllocateHandleTableEntrySlow (NewHandleTable, FALSE)) {
|
|||
|
|
|||
|
ExpFreeHandleTable (NewHandleTable);
|
|||
|
return NULL;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// Make sure any table reads occur after the value we fetched from NextHandleNeedingPool
|
|||
|
//
|
|||
|
|
|||
|
KeMemoryBarrier ();
|
|||
|
|
|||
|
//
|
|||
|
// Now modify the new handle table to think it has zero handles
|
|||
|
// and set its free list to start on the same index as the old
|
|||
|
// free list
|
|||
|
//
|
|||
|
|
|||
|
NewHandleTable->HandleCount = 0;
|
|||
|
NewHandleTable->ExtraInfoPages = 0;
|
|||
|
NewHandleTable->FirstFree = 0;
|
|||
|
|
|||
|
//
|
|||
|
// Now for every valid index value we'll copy over the old entry into
|
|||
|
// the new entry
|
|||
|
//
|
|||
|
|
|||
|
|
|||
|
Handle.Value = HANDLE_VALUE_INC;
|
|||
|
|
|||
|
KeEnterCriticalRegionThread (CurrentThread);
|
|||
|
while ((NewHandleTableEntry = ExpLookupHandleTableEntry( NewHandleTable, Handle )) != NULL) {
|
|||
|
|
|||
|
//
|
|||
|
// Lookup the old entry.
|
|||
|
//
|
|||
|
|
|||
|
OldHandleTableEntry = ExpLookupHandleTableEntry( OldHandleTable, Handle );
|
|||
|
|
|||
|
do {
|
|||
|
|
|||
|
//
|
|||
|
// If the old entry is free then simply copy over the entire
|
|||
|
// old entry to the new entry. The lock command will tell us
|
|||
|
// if the entry is free.
|
|||
|
//
|
|||
|
if ((OldHandleTableEntry->Value&Mask) == 0 ||
|
|||
|
!ExpLockHandleTableEntry( OldHandleTable, OldHandleTableEntry )) {
|
|||
|
FreeEntry = TRUE;
|
|||
|
} else {
|
|||
|
|
|||
|
PHANDLE_TABLE_ENTRY_INFO EntryInfo;
|
|||
|
|
|||
|
//
|
|||
|
// Otherwise we have a non empty entry. So now copy it
|
|||
|
// over, and unlock the old entry. In both cases we bump
|
|||
|
// the handle count because either the entry is going into
|
|||
|
// the new table or we're going to remove it with Exp Free
|
|||
|
// Handle Table Entry which will decrement the handle count
|
|||
|
//
|
|||
|
|
|||
|
*NewHandleTableEntry = *OldHandleTableEntry;
|
|||
|
|
|||
|
//
|
|||
|
// Copy the entry info data, if any
|
|||
|
//
|
|||
|
|
|||
|
Status = STATUS_SUCCESS;
|
|||
|
EntryInfo = ExGetHandleInfo(OldHandleTable, Handle.GenericHandleOverlay, TRUE);
|
|||
|
|
|||
|
if (EntryInfo) {
|
|||
|
|
|||
|
Status = ExSetHandleInfo(NewHandleTable, Handle.GenericHandleOverlay, EntryInfo, TRUE);
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
//
|
|||
|
// Invoke the callback and if it returns true then we
|
|||
|
// unlock the new entry
|
|||
|
//
|
|||
|
|
|||
|
if (NT_SUCCESS (Status)) {
|
|||
|
if ((*DupHandleProcedure) (Process,
|
|||
|
OldHandleTable,
|
|||
|
OldHandleTableEntry,
|
|||
|
NewHandleTableEntry)) {
|
|||
|
|
|||
|
if (NewHandleTable->DebugInfo != NULL) {
|
|||
|
ExpUpdateDebugInfo(
|
|||
|
NewHandleTable,
|
|||
|
PsGetCurrentThread (),
|
|||
|
Handle.GenericHandleOverlay,
|
|||
|
HANDLE_TRACE_DB_OPEN);
|
|||
|
}
|
|||
|
//
|
|||
|
// Since there is no route to the new table yet we can just
|
|||
|
// clear the lock bit
|
|||
|
//
|
|||
|
NewHandleTableEntry->Value |= EXHANDLE_TABLE_ENTRY_LOCK_BIT;
|
|||
|
NewHandleTable->HandleCount += 1;
|
|||
|
FreeEntry = FALSE;
|
|||
|
} else {
|
|||
|
if (EntryInfo) {
|
|||
|
EntryInfo->AuditMask = 0;
|
|||
|
}
|
|||
|
|
|||
|
FreeEntry = TRUE;
|
|||
|
}
|
|||
|
} else {
|
|||
|
//
|
|||
|
// Duplicate routine doesn't want this handle duplicated so free it
|
|||
|
//
|
|||
|
ExUnlockHandleTableEntry( OldHandleTable, OldHandleTableEntry );
|
|||
|
FreeEntry = TRUE;
|
|||
|
}
|
|||
|
|
|||
|
}
|
|||
|
if (FreeEntry) {
|
|||
|
NewHandleTableEntry->Object = NULL;
|
|||
|
NewHandleTableEntry->NextFreeTableEntry =
|
|||
|
NewHandleTable->FirstFree;
|
|||
|
NewHandleTable->FirstFree = (ULONG) Handle.Value;
|
|||
|
}
|
|||
|
Handle.Value += HANDLE_VALUE_INC;
|
|||
|
NewHandleTableEntry++;
|
|||
|
OldHandleTableEntry++;
|
|||
|
|
|||
|
} while ((Handle.Value % (LOWLEVEL_COUNT * HANDLE_VALUE_INC)) != 0);
|
|||
|
|
|||
|
Handle.Value += HANDLE_VALUE_INC; // Skip past the first entry thats not a real entry
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// Insert the handle table in the handle table list.
|
|||
|
//
|
|||
|
|
|||
|
ExAcquirePushLockExclusive( &HandleTableListLock );
|
|||
|
|
|||
|
InsertTailList( &HandleTableListHead, &NewHandleTable->HandleTableList );
|
|||
|
|
|||
|
ExReleasePushLockExclusive( &HandleTableListLock );
|
|||
|
KeLeaveCriticalRegionThread (CurrentThread);
|
|||
|
|
|||
|
//
|
|||
|
// lastly return the new handle table to our caller
|
|||
|
//
|
|||
|
|
|||
|
return NewHandleTable;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
NTKERNELAPI
|
|||
|
NTSTATUS
|
|||
|
ExSnapShotHandleTables (
|
|||
|
IN PEX_SNAPSHOT_HANDLE_ENTRY SnapShotHandleEntry,
|
|||
|
IN OUT PSYSTEM_HANDLE_INFORMATION HandleInformation,
|
|||
|
IN ULONG Length,
|
|||
|
IN OUT PULONG RequiredLength
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
This function visits and invokes the specified callback for every valid
|
|||
|
handle that it can find off of the handle table.
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
SnapShotHandleEntry - Supplies a pointer to a function to call for
|
|||
|
each valid handle we encounter.
|
|||
|
|
|||
|
HandleInformation - Supplies a handle information structure to
|
|||
|
be filled in for each handle table we encounter. This routine
|
|||
|
fills in the handle count, but relies on a callback to fill in
|
|||
|
entry info fields.
|
|||
|
|
|||
|
Length - Supplies a parameter for the callback. In reality this is
|
|||
|
the total size, in bytes, of the Handle Information buffer.
|
|||
|
|
|||
|
RequiredLength - Supplies a parameter for the callback. In reality
|
|||
|
this is a final size in bytes used to store the requested
|
|||
|
information.
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
The last return status of the callback
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
NTSTATUS Status;
|
|||
|
PKTHREAD CurrentThread;
|
|||
|
PSYSTEM_HANDLE_TABLE_ENTRY_INFO HandleEntryInfo;
|
|||
|
PLIST_ENTRY NextEntry;
|
|||
|
PHANDLE_TABLE HandleTable;
|
|||
|
EXHANDLE Handle;
|
|||
|
PHANDLE_TABLE_ENTRY HandleTableEntry;
|
|||
|
|
|||
|
PAGED_CODE();
|
|||
|
|
|||
|
CurrentThread = KeGetCurrentThread ();
|
|||
|
|
|||
|
|
|||
|
Status = STATUS_SUCCESS;
|
|||
|
|
|||
|
//
|
|||
|
// Setup the output buffer pointer that the callback will maintain
|
|||
|
//
|
|||
|
|
|||
|
HandleEntryInfo = &HandleInformation->Handles[0];
|
|||
|
|
|||
|
//
|
|||
|
// Zero out the handle count
|
|||
|
//
|
|||
|
|
|||
|
HandleInformation->NumberOfHandles = 0;
|
|||
|
|
|||
|
//
|
|||
|
// Lock the handle table list exclusive and traverse the list of handle
|
|||
|
// tables.
|
|||
|
//
|
|||
|
|
|||
|
KeEnterCriticalRegionThread (CurrentThread);
|
|||
|
ExAcquirePushLockShared( &HandleTableListLock );
|
|||
|
|
|||
|
//
|
|||
|
// Iterate through all the handle tables in the system.
|
|||
|
//
|
|||
|
|
|||
|
for (NextEntry = HandleTableListHead.Flink;
|
|||
|
NextEntry != &HandleTableListHead;
|
|||
|
NextEntry = NextEntry->Flink) {
|
|||
|
|
|||
|
//
|
|||
|
// Get the address of the next handle table, lock the handle
|
|||
|
// table exclusive, and scan the list of handle entries.
|
|||
|
//
|
|||
|
|
|||
|
HandleTable = CONTAINING_RECORD( NextEntry,
|
|||
|
HANDLE_TABLE,
|
|||
|
HandleTableList );
|
|||
|
|
|||
|
|
|||
|
// Iterate through the handle table and for each handle that
|
|||
|
// is allocated we'll invoke the call back. Note that this
|
|||
|
// loop exits when we get a null handle table entry. We know
|
|||
|
// there will be no more possible entries after the first null
|
|||
|
// one is encountered because we allocate memory of the
|
|||
|
// handles in a dense fashion
|
|||
|
//
|
|||
|
|
|||
|
for (Handle.Value = 0;
|
|||
|
(HandleTableEntry = ExpLookupHandleTableEntry( HandleTable, Handle )) != NULL;
|
|||
|
Handle.Value += HANDLE_VALUE_INC) {
|
|||
|
|
|||
|
//
|
|||
|
// Only do the callback if the entry is not free
|
|||
|
//
|
|||
|
|
|||
|
if ( ExpIsValidObjectEntry(HandleTableEntry) ) {
|
|||
|
|
|||
|
//
|
|||
|
// Increment the handle count information in the
|
|||
|
// information buffer
|
|||
|
//
|
|||
|
|
|||
|
HandleInformation->NumberOfHandles += 1;
|
|||
|
|
|||
|
//
|
|||
|
// Lock the handle table entry because we're about to
|
|||
|
// give it to the callback function, then release the
|
|||
|
// entry right after the call back.
|
|||
|
//
|
|||
|
|
|||
|
if (ExpLockHandleTableEntry( HandleTable, HandleTableEntry )) {
|
|||
|
|
|||
|
Status = (*SnapShotHandleEntry)( &HandleEntryInfo,
|
|||
|
HandleTable->UniqueProcessId,
|
|||
|
HandleTableEntry,
|
|||
|
Handle.GenericHandleOverlay,
|
|||
|
Length,
|
|||
|
RequiredLength );
|
|||
|
|
|||
|
ExUnlockHandleTableEntry( HandleTable, HandleTableEntry );
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
ExReleasePushLockShared( &HandleTableListLock );
|
|||
|
KeLeaveCriticalRegionThread (CurrentThread);
|
|||
|
|
|||
|
return Status;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
NTKERNELAPI
|
|||
|
NTSTATUS
|
|||
|
ExSnapShotHandleTablesEx (
|
|||
|
IN PEX_SNAPSHOT_HANDLE_ENTRY_EX SnapShotHandleEntry,
|
|||
|
IN OUT PSYSTEM_HANDLE_INFORMATION_EX HandleInformation,
|
|||
|
IN ULONG Length,
|
|||
|
IN OUT PULONG RequiredLength
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
This function visits and invokes the specified callback for every valid
|
|||
|
handle that it can find off of the handle table.
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
SnapShotHandleEntry - Supplies a pointer to a function to call for
|
|||
|
each valid handle we encounter.
|
|||
|
|
|||
|
HandleInformation - Supplies a handle information structure to
|
|||
|
be filled in for each handle table we encounter. This routine
|
|||
|
fills in the handle count, but relies on a callback to fill in
|
|||
|
entry info fields.
|
|||
|
|
|||
|
Length - Supplies a parameter for the callback. In reality this is
|
|||
|
the total size, in bytes, of the Handle Information buffer.
|
|||
|
|
|||
|
RequiredLength - Supplies a parameter for the callback. In reality
|
|||
|
this is a final size in bytes used to store the requested
|
|||
|
information.
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
The last return status of the callback
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
NTSTATUS Status;
|
|||
|
PKTHREAD CurrentThread;
|
|||
|
PSYSTEM_HANDLE_TABLE_ENTRY_INFO_EX HandleEntryInfo;
|
|||
|
PLIST_ENTRY NextEntry;
|
|||
|
PHANDLE_TABLE HandleTable;
|
|||
|
EXHANDLE Handle;
|
|||
|
PHANDLE_TABLE_ENTRY HandleTableEntry;
|
|||
|
|
|||
|
PAGED_CODE();
|
|||
|
|
|||
|
CurrentThread = KeGetCurrentThread ();
|
|||
|
|
|||
|
Status = STATUS_SUCCESS;
|
|||
|
|
|||
|
|
|||
|
//
|
|||
|
// Setup the output buffer pointer that the callback will maintain
|
|||
|
//
|
|||
|
|
|||
|
HandleEntryInfo = &HandleInformation->Handles[0];
|
|||
|
|
|||
|
//
|
|||
|
// Zero out the handle count
|
|||
|
//
|
|||
|
|
|||
|
HandleInformation->NumberOfHandles = 0;
|
|||
|
|
|||
|
//
|
|||
|
// Lock the handle table list exclusive and traverse the list of handle
|
|||
|
// tables.
|
|||
|
//
|
|||
|
|
|||
|
KeEnterCriticalRegionThread (CurrentThread);
|
|||
|
ExAcquirePushLockShared( &HandleTableListLock );
|
|||
|
|
|||
|
//
|
|||
|
// Iterate through all the handle tables in the system.
|
|||
|
//
|
|||
|
|
|||
|
for (NextEntry = HandleTableListHead.Flink;
|
|||
|
NextEntry != &HandleTableListHead;
|
|||
|
NextEntry = NextEntry->Flink) {
|
|||
|
|
|||
|
//
|
|||
|
// Get the address of the next handle table, lock the handle
|
|||
|
// table exclusive, and scan the list of handle entries.
|
|||
|
//
|
|||
|
|
|||
|
HandleTable = CONTAINING_RECORD( NextEntry,
|
|||
|
HANDLE_TABLE,
|
|||
|
HandleTableList );
|
|||
|
|
|||
|
|
|||
|
// Iterate through the handle table and for each handle that
|
|||
|
// is allocated we'll invoke the call back. Note that this
|
|||
|
// loop exits when we get a null handle table entry. We know
|
|||
|
// there will be no more possible entries after the first null
|
|||
|
// one is encountered because we allocate memory of the
|
|||
|
// handles in a dense fashion
|
|||
|
//
|
|||
|
|
|||
|
for (Handle.Value = 0;
|
|||
|
(HandleTableEntry = ExpLookupHandleTableEntry( HandleTable, Handle )) != NULL;
|
|||
|
Handle.Value += HANDLE_VALUE_INC) {
|
|||
|
|
|||
|
//
|
|||
|
// Only do the callback if the entry is not free
|
|||
|
//
|
|||
|
|
|||
|
if ( ExpIsValidObjectEntry(HandleTableEntry) ) {
|
|||
|
|
|||
|
//
|
|||
|
// Increment the handle count information in the
|
|||
|
// information buffer
|
|||
|
//
|
|||
|
|
|||
|
HandleInformation->NumberOfHandles += 1;
|
|||
|
|
|||
|
//
|
|||
|
// Lock the handle table entry because we're about to
|
|||
|
// give it to the callback function, then release the
|
|||
|
// entry right after the call back.
|
|||
|
//
|
|||
|
|
|||
|
if (ExpLockHandleTableEntry( HandleTable, HandleTableEntry )) {
|
|||
|
|
|||
|
Status = (*SnapShotHandleEntry)( &HandleEntryInfo,
|
|||
|
HandleTable->UniqueProcessId,
|
|||
|
HandleTableEntry,
|
|||
|
Handle.GenericHandleOverlay,
|
|||
|
Length,
|
|||
|
RequiredLength );
|
|||
|
|
|||
|
ExUnlockHandleTableEntry( HandleTable, HandleTableEntry );
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
ExReleasePushLockShared( &HandleTableListLock );
|
|||
|
KeLeaveCriticalRegionThread (CurrentThread);
|
|||
|
|
|||
|
return Status;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
NTKERNELAPI
|
|||
|
HANDLE
|
|||
|
ExCreateHandle (
|
|||
|
IN PHANDLE_TABLE HandleTable,
|
|||
|
IN PHANDLE_TABLE_ENTRY HandleTableEntry
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
This function creates a handle entry in the specified handle table and
|
|||
|
returns a handle for the entry.
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
HandleTable - Supplies a pointer to a handle table
|
|||
|
|
|||
|
HandleEntry - Supplies a poiner to the handle entry for which a
|
|||
|
handle entry is created.
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
If the handle entry is successfully created, then value of the created
|
|||
|
handle is returned as the function value. Otherwise, a value of zero is
|
|||
|
returned.
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
EXHANDLE Handle;
|
|||
|
PETHREAD CurrentThread;
|
|||
|
PHANDLE_TABLE_ENTRY NewHandleTableEntry;
|
|||
|
|
|||
|
PAGED_CODE();
|
|||
|
|
|||
|
//
|
|||
|
// Set out output variable to zero (i.e., null) before going on
|
|||
|
//
|
|||
|
|
|||
|
//
|
|||
|
// Clears Handle.Index and Handle.TagBits
|
|||
|
//
|
|||
|
|
|||
|
Handle.GenericHandleOverlay = NULL;
|
|||
|
|
|||
|
|
|||
|
//
|
|||
|
// Allocate a new handle table entry, and get the handle value
|
|||
|
//
|
|||
|
|
|||
|
NewHandleTableEntry = ExpAllocateHandleTableEntry( HandleTable,
|
|||
|
&Handle );
|
|||
|
|
|||
|
//
|
|||
|
// If we really got a handle then copy over the template and unlock
|
|||
|
// the entry
|
|||
|
//
|
|||
|
|
|||
|
if (NewHandleTableEntry != NULL) {
|
|||
|
|
|||
|
CurrentThread = PsGetCurrentThread ();
|
|||
|
|
|||
|
//
|
|||
|
// We are about to create a locked entry so protect against suspension
|
|||
|
//
|
|||
|
KeEnterCriticalRegionThread (&CurrentThread->Tcb);
|
|||
|
|
|||
|
*NewHandleTableEntry = *HandleTableEntry;
|
|||
|
|
|||
|
//
|
|||
|
// If we are debugging handle operations then save away the details
|
|||
|
//
|
|||
|
if (HandleTable->DebugInfo != NULL) {
|
|||
|
ExpUpdateDebugInfo(HandleTable, CurrentThread, Handle.GenericHandleOverlay, HANDLE_TRACE_DB_OPEN);
|
|||
|
}
|
|||
|
|
|||
|
ExUnlockHandleTableEntry( HandleTable, NewHandleTableEntry );
|
|||
|
|
|||
|
KeLeaveCriticalRegionThread (&CurrentThread->Tcb);
|
|||
|
}
|
|||
|
|
|||
|
return Handle.GenericHandleOverlay;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
NTKERNELAPI
|
|||
|
BOOLEAN
|
|||
|
ExDestroyHandle (
|
|||
|
IN PHANDLE_TABLE HandleTable,
|
|||
|
IN HANDLE Handle,
|
|||
|
IN PHANDLE_TABLE_ENTRY HandleTableEntry OPTIONAL
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
This function removes a handle from a handle table.
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
HandleTable - Supplies a pointer to a handle table
|
|||
|
|
|||
|
Handle - Supplies the handle value of the entry to remove.
|
|||
|
|
|||
|
HandleTableEntry - Optionally supplies a pointer to the handle
|
|||
|
table entry being destroyed. If supplied the entry is
|
|||
|
assume to be locked.
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
If the specified handle is successfully removed, then a value of
|
|||
|
TRUE is returned. Otherwise, a value of FALSE is returned.
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
EXHANDLE LocalHandle;
|
|||
|
PETHREAD CurrentThread;
|
|||
|
PVOID Object;
|
|||
|
|
|||
|
PAGED_CODE();
|
|||
|
|
|||
|
LocalHandle.GenericHandleOverlay = Handle;
|
|||
|
|
|||
|
CurrentThread = PsGetCurrentThread ();
|
|||
|
|
|||
|
//
|
|||
|
// If the caller did not supply the optional handle table entry then
|
|||
|
// locate the entry via the supplied handle, make sure it is real, and
|
|||
|
// then lock the entry.
|
|||
|
//
|
|||
|
|
|||
|
KeEnterCriticalRegionThread (&CurrentThread->Tcb);
|
|||
|
|
|||
|
if (HandleTableEntry == NULL) {
|
|||
|
|
|||
|
HandleTableEntry = ExpLookupHandleTableEntry( HandleTable,
|
|||
|
LocalHandle );
|
|||
|
|
|||
|
if (!ExpIsValidObjectEntry(HandleTableEntry)) {
|
|||
|
|
|||
|
KeLeaveCriticalRegionThread (&CurrentThread->Tcb);
|
|||
|
return FALSE;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
if (!ExpLockHandleTableEntry( HandleTable, HandleTableEntry )) {
|
|||
|
|
|||
|
KeLeaveCriticalRegionThread (&CurrentThread->Tcb);
|
|||
|
return FALSE;
|
|||
|
}
|
|||
|
} else {
|
|||
|
EXASSERT ((HandleTableEntry->Value&EXHANDLE_TABLE_ENTRY_LOCK_BIT) == 0);
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
//
|
|||
|
// If we are debugging handle operations then save away the details
|
|||
|
//
|
|||
|
|
|||
|
if (HandleTable->DebugInfo != NULL) {
|
|||
|
ExpUpdateDebugInfo(HandleTable, CurrentThread, Handle, HANDLE_TRACE_DB_CLOSE);
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// At this point we have a locked handle table entry. Now mark it free
|
|||
|
// which does the implicit unlock. The system will not allocate it
|
|||
|
// again until we add it to the free list which we will do right after
|
|||
|
// we take out the lock
|
|||
|
//
|
|||
|
|
|||
|
Object = InterlockedExchangePointer (&HandleTableEntry->Object, NULL);
|
|||
|
|
|||
|
EXASSERT (Object != NULL);
|
|||
|
EXASSERT ((((ULONG_PTR)Object)&EXHANDLE_TABLE_ENTRY_LOCK_BIT) == 0);
|
|||
|
|
|||
|
//
|
|||
|
// Unblock any waiters waiting for this table entry.
|
|||
|
//
|
|||
|
ExUnblockPushLock (&HandleTable->HandleContentionEvent, NULL);
|
|||
|
|
|||
|
|
|||
|
ExpFreeHandleTableEntry( HandleTable,
|
|||
|
LocalHandle,
|
|||
|
HandleTableEntry );
|
|||
|
|
|||
|
KeLeaveCriticalRegionThread (&CurrentThread->Tcb);
|
|||
|
|
|||
|
return TRUE;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
NTKERNELAPI
|
|||
|
BOOLEAN
|
|||
|
ExChangeHandle (
|
|||
|
IN PHANDLE_TABLE HandleTable,
|
|||
|
IN HANDLE Handle,
|
|||
|
IN PEX_CHANGE_HANDLE_ROUTINE ChangeRoutine,
|
|||
|
IN ULONG_PTR Parameter
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
This function provides the capability to change the contents of the
|
|||
|
handle entry corrsponding to the specified handle.
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
HandleTable - Supplies a pointer to a handle table.
|
|||
|
|
|||
|
Handle - Supplies the handle for the handle entry that is changed.
|
|||
|
|
|||
|
ChangeRoutine - Supplies a pointer to a function that is called to
|
|||
|
perform the change.
|
|||
|
|
|||
|
Parameter - Supplies an uninterpreted parameter that is passed to
|
|||
|
the change routine.
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
If the operation was successfully performed, then a value of TRUE
|
|||
|
is returned. Otherwise, a value of FALSE is returned.
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
EXHANDLE LocalHandle;
|
|||
|
PKTHREAD CurrentThread;
|
|||
|
|
|||
|
PHANDLE_TABLE_ENTRY HandleTableEntry;
|
|||
|
BOOLEAN ReturnValue;
|
|||
|
|
|||
|
PAGED_CODE();
|
|||
|
|
|||
|
LocalHandle.GenericHandleOverlay = Handle;
|
|||
|
|
|||
|
CurrentThread = KeGetCurrentThread ();
|
|||
|
|
|||
|
//
|
|||
|
// Translate the input handle to a handle table entry and make
|
|||
|
// sure it is a valid handle.
|
|||
|
//
|
|||
|
|
|||
|
HandleTableEntry = ExpLookupHandleTableEntry( HandleTable,
|
|||
|
LocalHandle );
|
|||
|
|
|||
|
if ((HandleTableEntry == NULL) ||
|
|||
|
!ExpIsValidObjectEntry(HandleTableEntry)) {
|
|||
|
|
|||
|
return FALSE;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
|
|||
|
//
|
|||
|
// Try and lock the handle table entry, If this fails then that's
|
|||
|
// because someone freed the handle
|
|||
|
//
|
|||
|
|
|||
|
//
|
|||
|
// Make sure we can't get suspended and then invoke the callback
|
|||
|
//
|
|||
|
|
|||
|
KeEnterCriticalRegionThread (CurrentThread);
|
|||
|
|
|||
|
if (ExpLockHandleTableEntry( HandleTable, HandleTableEntry )) {
|
|||
|
|
|||
|
|
|||
|
ReturnValue = (*ChangeRoutine)( HandleTableEntry, Parameter );
|
|||
|
|
|||
|
ExUnlockHandleTableEntry( HandleTable, HandleTableEntry );
|
|||
|
|
|||
|
} else {
|
|||
|
ReturnValue = FALSE;
|
|||
|
}
|
|||
|
|
|||
|
KeLeaveCriticalRegionThread (CurrentThread);
|
|||
|
|
|||
|
return ReturnValue;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
NTKERNELAPI
|
|||
|
PHANDLE_TABLE_ENTRY
|
|||
|
ExMapHandleToPointer (
|
|||
|
IN PHANDLE_TABLE HandleTable,
|
|||
|
IN HANDLE Handle
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
This function maps a handle to a pointer to a handle table entry. If the
|
|||
|
map operation is successful then the handle table entry is locked when
|
|||
|
we return.
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
HandleTable - Supplies a pointer to a handle table.
|
|||
|
|
|||
|
Handle - Supplies the handle to be mapped to a handle entry.
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
If the handle was successfully mapped to a pointer to a handle entry,
|
|||
|
then the address of the handle table entry is returned as the function
|
|||
|
value with the entry locked. Otherwise, a value of NULL is returned.
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
EXHANDLE LocalHandle;
|
|||
|
PHANDLE_TABLE_ENTRY HandleTableEntry;
|
|||
|
|
|||
|
PAGED_CODE();
|
|||
|
|
|||
|
LocalHandle.GenericHandleOverlay = Handle;
|
|||
|
|
|||
|
if ((LocalHandle.Index & (LOWLEVEL_COUNT - 1)) == 0) {
|
|||
|
return NULL;
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// Translate the input handle to a handle table entry and make
|
|||
|
// sure it is a valid handle.
|
|||
|
//
|
|||
|
|
|||
|
HandleTableEntry = ExpLookupHandleTableEntry( HandleTable,
|
|||
|
LocalHandle );
|
|||
|
|
|||
|
if ((HandleTableEntry == NULL) ||
|
|||
|
!ExpLockHandleTableEntry( HandleTable, HandleTableEntry)) {
|
|||
|
//
|
|||
|
// If we are debugging handle operations then save away the details
|
|||
|
//
|
|||
|
|
|||
|
if (HandleTable->DebugInfo != NULL) {
|
|||
|
ExpUpdateDebugInfo(HandleTable, PsGetCurrentThread (), Handle, HANDLE_TRACE_DB_BADREF);
|
|||
|
}
|
|||
|
return NULL;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
//
|
|||
|
// Return the locked valid handle table entry
|
|||
|
//
|
|||
|
|
|||
|
return HandleTableEntry;
|
|||
|
}
|
|||
|
|
|||
|
NTKERNELAPI
|
|||
|
PHANDLE_TABLE_ENTRY
|
|||
|
ExMapHandleToPointerEx (
|
|||
|
IN PHANDLE_TABLE HandleTable,
|
|||
|
IN HANDLE Handle,
|
|||
|
IN KPROCESSOR_MODE PreviousMode
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
This function maps a handle to a pointer to a handle table entry. If the
|
|||
|
map operation is successful then the handle table entry is locked when
|
|||
|
we return.
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
HandleTable - Supplies a pointer to a handle table.
|
|||
|
|
|||
|
Handle - Supplies the handle to be mapped to a handle entry.
|
|||
|
|
|||
|
PreviousMode - Previous mode of caller
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
If the handle was successfully mapped to a pointer to a handle entry,
|
|||
|
then the address of the handle table entry is returned as the function
|
|||
|
value with the entry locked. Otherwise, a value of NULL is returned.
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
EXHANDLE LocalHandle;
|
|||
|
PHANDLE_TABLE_ENTRY HandleTableEntry = NULL;
|
|||
|
PETHREAD CurrentThread;
|
|||
|
|
|||
|
PAGED_CODE();
|
|||
|
|
|||
|
LocalHandle.GenericHandleOverlay = Handle;
|
|||
|
|
|||
|
//
|
|||
|
// Translate the input handle to a handle table entry and make
|
|||
|
// sure it is a valid handle.
|
|||
|
//
|
|||
|
|
|||
|
if (((LocalHandle.Index & (LOWLEVEL_COUNT - 1)) == 0) ||
|
|||
|
((HandleTableEntry = ExpLookupHandleTableEntry(HandleTable, LocalHandle)) == NULL) ||
|
|||
|
!ExpLockHandleTableEntry( HandleTable, HandleTableEntry)) {
|
|||
|
|
|||
|
//
|
|||
|
// If we are debugging handle operations then save away the details
|
|||
|
//
|
|||
|
|
|||
|
if (HandleTable->DebugInfo != NULL) {
|
|||
|
CurrentThread = PsGetCurrentThread ();
|
|||
|
ExpUpdateDebugInfo(HandleTable, CurrentThread, Handle, HANDLE_TRACE_DB_BADREF);
|
|||
|
|
|||
|
//
|
|||
|
// Since we have a non-null DebugInfo for the handle table of this
|
|||
|
// process it means application verifier was enabled for this process.
|
|||
|
//
|
|||
|
|
|||
|
if (PreviousMode == UserMode) {
|
|||
|
|
|||
|
if (!KeIsAttachedProcess()) {
|
|||
|
|
|||
|
//
|
|||
|
// If the current process is marked for verification
|
|||
|
// then we will raise an exception in user mode. In case
|
|||
|
// application verifier is enabled system wide we will
|
|||
|
// break first.
|
|||
|
//
|
|||
|
|
|||
|
if ((NtGlobalFlag & FLG_APPLICATION_VERIFIER)) {
|
|||
|
|
|||
|
DbgPrint ("AVRF: Invalid handle %p in process %p \n",
|
|||
|
Handle,
|
|||
|
PsGetCurrentProcess());
|
|||
|
|
|||
|
// DbgBreakPoint ();
|
|||
|
}
|
|||
|
|
|||
|
KeRaiseUserException (STATUS_INVALID_HANDLE);
|
|||
|
}
|
|||
|
} else {
|
|||
|
|
|||
|
//
|
|||
|
// We bugcheck for kernel handles only if we have the handle
|
|||
|
// exceptions flag set system-wide. This way a user enabling
|
|||
|
// application verifier for a process will not get bugchecks
|
|||
|
// only user mode errors.
|
|||
|
//
|
|||
|
|
|||
|
if ((NtGlobalFlag & FLG_ENABLE_HANDLE_EXCEPTIONS)) {
|
|||
|
|
|||
|
KeBugCheckEx(INVALID_KERNEL_HANDLE,
|
|||
|
(ULONG_PTR)Handle,
|
|||
|
(ULONG_PTR)HandleTable,
|
|||
|
(ULONG_PTR)HandleTableEntry,
|
|||
|
0x1);
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
return NULL;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
//
|
|||
|
// Return the locked valid handle table entry
|
|||
|
//
|
|||
|
|
|||
|
return HandleTableEntry;
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// Local Support Routine
|
|||
|
//
|
|||
|
|
|||
|
PVOID
|
|||
|
ExpAllocateTablePagedPool (
|
|||
|
IN PEPROCESS QuotaProcess OPTIONAL,
|
|||
|
IN SIZE_T NumberOfBytes
|
|||
|
)
|
|||
|
{
|
|||
|
PVOID PoolMemory;
|
|||
|
|
|||
|
PoolMemory = ExAllocatePoolWithTag( PagedPool,
|
|||
|
NumberOfBytes,
|
|||
|
'btbO' );
|
|||
|
if (PoolMemory != NULL) {
|
|||
|
|
|||
|
RtlZeroMemory( PoolMemory,
|
|||
|
NumberOfBytes );
|
|||
|
|
|||
|
if (ARGUMENT_PRESENT(QuotaProcess)) {
|
|||
|
|
|||
|
if (!NT_SUCCESS (PsChargeProcessPagedPoolQuota ( QuotaProcess,
|
|||
|
NumberOfBytes ))) {
|
|||
|
ExFreePool( PoolMemory );
|
|||
|
PoolMemory = NULL;
|
|||
|
}
|
|||
|
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
return PoolMemory;
|
|||
|
}
|
|||
|
|
|||
|
PVOID
|
|||
|
ExpAllocateTablePagedPoolNoZero (
|
|||
|
IN PEPROCESS QuotaProcess OPTIONAL,
|
|||
|
IN SIZE_T NumberOfBytes
|
|||
|
)
|
|||
|
{
|
|||
|
PVOID PoolMemory;
|
|||
|
|
|||
|
PoolMemory = ExAllocatePoolWithTag( PagedPool,
|
|||
|
NumberOfBytes,
|
|||
|
'btbO' );
|
|||
|
if (PoolMemory != NULL) {
|
|||
|
|
|||
|
if (ARGUMENT_PRESENT(QuotaProcess)) {
|
|||
|
|
|||
|
if (!NT_SUCCESS (PsChargeProcessPagedPoolQuota ( QuotaProcess,
|
|||
|
NumberOfBytes ))) {
|
|||
|
ExFreePool( PoolMemory );
|
|||
|
PoolMemory = NULL;
|
|||
|
}
|
|||
|
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
return PoolMemory;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
//
|
|||
|
// Local Support Routine
|
|||
|
//
|
|||
|
|
|||
|
VOID
|
|||
|
ExpFreeTablePagedPool (
|
|||
|
IN PEPROCESS QuotaProcess OPTIONAL,
|
|||
|
IN PVOID PoolMemory,
|
|||
|
IN SIZE_T NumberOfBytes
|
|||
|
)
|
|||
|
{
|
|||
|
|
|||
|
ExFreePool( PoolMemory );
|
|||
|
|
|||
|
if ( QuotaProcess ) {
|
|||
|
|
|||
|
PsReturnProcessPagedPoolQuota( QuotaProcess,
|
|||
|
NumberOfBytes
|
|||
|
);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
|
|||
|
PHANDLE_TRACE_DEBUG_INFO
|
|||
|
ExReferenceHandleDebugInfo (
|
|||
|
IN PHANDLE_TABLE HandleTable
|
|||
|
)
|
|||
|
{
|
|||
|
LONG RetVal;
|
|||
|
PHANDLE_TRACE_DEBUG_INFO DebugInfo;
|
|||
|
PKTHREAD CurrentThread;
|
|||
|
|
|||
|
CurrentThread = KeGetCurrentThread ();
|
|||
|
|
|||
|
ExpLockHandleTableShared (HandleTable, CurrentThread, 0);
|
|||
|
|
|||
|
DebugInfo = HandleTable->DebugInfo;
|
|||
|
|
|||
|
if (DebugInfo != NULL) {
|
|||
|
RetVal = InterlockedIncrement (&DebugInfo->RefCount);
|
|||
|
ASSERT (RetVal > 0);
|
|||
|
}
|
|||
|
|
|||
|
ExpUnlockHandleTableShared (HandleTable, CurrentThread, 0);
|
|||
|
|
|||
|
return DebugInfo;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
VOID
|
|||
|
ExDereferenceHandleDebugInfo (
|
|||
|
IN PHANDLE_TABLE HandleTable,
|
|||
|
IN PHANDLE_TRACE_DEBUG_INFO DebugInfo
|
|||
|
)
|
|||
|
{
|
|||
|
ULONG TraceSize;
|
|||
|
ULONG TableSize;
|
|||
|
LONG RetVal;
|
|||
|
|
|||
|
RetVal = InterlockedDecrement (&DebugInfo->RefCount);
|
|||
|
|
|||
|
ASSERT (RetVal >= 0);
|
|||
|
|
|||
|
if (RetVal == 0) {
|
|||
|
|
|||
|
TableSize = DebugInfo->TableSize;
|
|||
|
TraceSize = sizeof (*DebugInfo) + TableSize * sizeof (DebugInfo->TraceDb[0]) - sizeof (DebugInfo->TraceDb);
|
|||
|
|
|||
|
ExFreePool (DebugInfo);
|
|||
|
|
|||
|
if (HandleTable->QuotaProcess != NULL) {
|
|||
|
PsReturnProcessNonPagedPoolQuota (HandleTable->QuotaProcess,
|
|||
|
TraceSize);
|
|||
|
}
|
|||
|
|
|||
|
InterlockedExchangeAdd ((PLONG) &TotalTraceBuffers, -(LONG)TableSize);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
NTKERNELAPI
|
|||
|
NTSTATUS
|
|||
|
ExDisableHandleTracing (
|
|||
|
IN PHANDLE_TABLE HandleTable
|
|||
|
)
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
This routine turns off handle tracing for the specified table
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
HandleTable - Table to disable tracing in
|
|||
|
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
NTSTATUS - Status of operation
|
|||
|
|
|||
|
--*/
|
|||
|
{
|
|||
|
PHANDLE_TRACE_DEBUG_INFO DebugInfo;
|
|||
|
PKTHREAD CurrentThread;
|
|||
|
|
|||
|
CurrentThread = KeGetCurrentThread ();
|
|||
|
|
|||
|
ExpLockHandleTableExclusive (HandleTable, CurrentThread);
|
|||
|
|
|||
|
DebugInfo = HandleTable->DebugInfo;
|
|||
|
|
|||
|
HandleTable->DebugInfo = NULL;
|
|||
|
|
|||
|
ExpUnlockHandleTableExclusive (HandleTable, CurrentThread);
|
|||
|
|
|||
|
if (DebugInfo != NULL) {
|
|||
|
ExDereferenceHandleDebugInfo (HandleTable, DebugInfo);
|
|||
|
}
|
|||
|
|
|||
|
return STATUS_SUCCESS;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
NTKERNELAPI
|
|||
|
NTSTATUS
|
|||
|
ExEnableHandleTracing (
|
|||
|
IN PHANDLE_TABLE HandleTable,
|
|||
|
IN ULONG Slots
|
|||
|
)
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
This routine turns on handle tracing for the specified table
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
HandleTable - Table to enable tracing in
|
|||
|
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
NTSTATUS - Status of operation
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
PHANDLE_TRACE_DEBUG_INFO DebugInfo, OldDebugInfo;
|
|||
|
PEPROCESS Process;
|
|||
|
PKTHREAD CurrentThread;
|
|||
|
NTSTATUS Status;
|
|||
|
SIZE_T TotalNow;
|
|||
|
extern SIZE_T MmMaximumNonPagedPoolInBytes;
|
|||
|
SIZE_T TraceSize;
|
|||
|
LONG TotalSlots;
|
|||
|
|
|||
|
if (Slots == 0) {
|
|||
|
TotalSlots = HANDLE_TRACE_DB_DEFAULT_STACKS;
|
|||
|
} else {
|
|||
|
if (Slots < HANDLE_TRACE_DB_MIN_STACKS) {
|
|||
|
TotalSlots = HANDLE_TRACE_DB_MIN_STACKS;
|
|||
|
} else {
|
|||
|
TotalSlots = Slots;
|
|||
|
}
|
|||
|
|
|||
|
if (TotalSlots > HANDLE_TRACE_DB_MAX_STACKS) {
|
|||
|
TotalSlots = HANDLE_TRACE_DB_MAX_STACKS;
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// Round the value up to the next power of 2
|
|||
|
//
|
|||
|
|
|||
|
while ((TotalSlots & (TotalSlots - 1)) != 0) {
|
|||
|
TotalSlots |= (TotalSlots - 1);
|
|||
|
TotalSlots += 1;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// Total slots needs to be a power of two
|
|||
|
//
|
|||
|
ASSERT ((TotalSlots & (TotalSlots - 1)) == 0);
|
|||
|
ASSERT (TotalSlots > 0 && TotalSlots <= HANDLE_TRACE_DB_MAX_STACKS);
|
|||
|
|
|||
|
TraceSize = sizeof (*DebugInfo) + TotalSlots * sizeof (DebugInfo->TraceDb[0]) - sizeof (DebugInfo->TraceDb);
|
|||
|
|
|||
|
TotalNow = InterlockedExchangeAdd ((PLONG) &TotalTraceBuffers, TotalSlots);
|
|||
|
|
|||
|
//
|
|||
|
// See if we used more than 30% of nonpaged pool.
|
|||
|
//
|
|||
|
if ((SIZE_T)TotalNow * sizeof (DebugInfo->TraceDb[0]) > (MmMaximumNonPagedPoolInBytes * 30 / 100)) {
|
|||
|
Status = STATUS_INSUFFICIENT_RESOURCES;
|
|||
|
goto return_and_exit;
|
|||
|
}
|
|||
|
|
|||
|
Process = HandleTable->QuotaProcess;
|
|||
|
|
|||
|
if (Process) {
|
|||
|
Status = PsChargeProcessNonPagedPoolQuota (Process,
|
|||
|
TraceSize);
|
|||
|
if (!NT_SUCCESS (Status)) {
|
|||
|
goto return_and_exit;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// Allocate the handle debug database
|
|||
|
//
|
|||
|
DebugInfo = ExAllocatePoolWithTag (NonPagedPool,
|
|||
|
TraceSize,
|
|||
|
'dtbO');
|
|||
|
if (DebugInfo == NULL) {
|
|||
|
Status = STATUS_INSUFFICIENT_RESOURCES;
|
|||
|
goto quota_return_and_exit;
|
|||
|
}
|
|||
|
RtlZeroMemory (DebugInfo, TraceSize);
|
|||
|
|
|||
|
DebugInfo->RefCount = 1;
|
|||
|
DebugInfo->TableSize = TotalSlots;
|
|||
|
|
|||
|
ExInitializeFastMutex(&DebugInfo->CloseCompactionLock);
|
|||
|
|
|||
|
//
|
|||
|
// Since we are tracing then we should enforce strict FIFO
|
|||
|
// Only do this for tables with processes so we leave atom tables alone.
|
|||
|
//
|
|||
|
if (Process != NULL) {
|
|||
|
HandleTable->StrictFIFO = TRUE;
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// Put the new table in place releasing any existing table
|
|||
|
//
|
|||
|
|
|||
|
CurrentThread = KeGetCurrentThread ();
|
|||
|
|
|||
|
ExpLockHandleTableExclusive (HandleTable, CurrentThread);
|
|||
|
|
|||
|
OldDebugInfo = HandleTable->DebugInfo;
|
|||
|
|
|||
|
HandleTable->DebugInfo = DebugInfo;
|
|||
|
|
|||
|
ExpUnlockHandleTableExclusive (HandleTable, CurrentThread);
|
|||
|
|
|||
|
if (OldDebugInfo != NULL) {
|
|||
|
ExDereferenceHandleDebugInfo (HandleTable, OldDebugInfo);
|
|||
|
}
|
|||
|
|
|||
|
return STATUS_SUCCESS;
|
|||
|
|
|||
|
quota_return_and_exit:
|
|||
|
|
|||
|
if (Process) {
|
|||
|
PsReturnProcessNonPagedPoolQuota (Process,
|
|||
|
TraceSize);
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
return_and_exit:
|
|||
|
|
|||
|
InterlockedExchangeAdd ((PLONG) &TotalTraceBuffers, -TotalSlots);
|
|||
|
return Status;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
//
|
|||
|
// Local Support Routine
|
|||
|
//
|
|||
|
|
|||
|
PHANDLE_TABLE
|
|||
|
ExpAllocateHandleTable (
|
|||
|
IN PEPROCESS Process OPTIONAL,
|
|||
|
IN BOOLEAN DoInit
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
This worker routine will allocate and initialize a new handle table
|
|||
|
structure. The new structure consists of the basic handle table
|
|||
|
struct plus the first allocation needed to store handles. This is
|
|||
|
really one page divided up into the top level node, the first mid
|
|||
|
level node, and one bottom level node.
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
Process - Optionally supplies the process to charge quota for the
|
|||
|
handle table
|
|||
|
|
|||
|
DoInit - If FALSE then we are being called by duplicate and we don't need
|
|||
|
the free list built for the caller
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
A pointer to the new handle table or NULL if unsuccessful at getting
|
|||
|
pool.
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
PHANDLE_TABLE HandleTable;
|
|||
|
PHANDLE_TABLE_ENTRY HandleTableTable, HandleEntry;
|
|||
|
ULONG i, Idx;
|
|||
|
|
|||
|
PAGED_CODE();
|
|||
|
|
|||
|
//
|
|||
|
// If any alloation or quota failures happen we will catch it in the
|
|||
|
// following try-except clause and cleanup after outselves before
|
|||
|
// we return null
|
|||
|
//
|
|||
|
|
|||
|
//
|
|||
|
// First allocate the handle table, make sure we got one, charge quota
|
|||
|
// for it and then zero it out
|
|||
|
//
|
|||
|
|
|||
|
HandleTable = (PHANDLE_TABLE)ExAllocatePoolWithTag (PagedPool,
|
|||
|
sizeof(HANDLE_TABLE),
|
|||
|
'btbO');
|
|||
|
if (HandleTable == NULL) {
|
|||
|
return NULL;
|
|||
|
}
|
|||
|
|
|||
|
if (ARGUMENT_PRESENT(Process)) {
|
|||
|
|
|||
|
if (!NT_SUCCESS (PsChargeProcessPagedPoolQuota( Process,
|
|||
|
sizeof(HANDLE_TABLE)))) {
|
|||
|
ExFreePool( HandleTable );
|
|||
|
return NULL;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
RtlZeroMemory( HandleTable, sizeof(HANDLE_TABLE) );
|
|||
|
|
|||
|
|
|||
|
//
|
|||
|
// Now allocate space of the top level, one mid level and one bottom
|
|||
|
// level table structure. This will all fit on a page, maybe two.
|
|||
|
//
|
|||
|
|
|||
|
HandleTableTable = ExpAllocateTablePagedPoolNoZero ( Process,
|
|||
|
TABLE_PAGE_SIZE
|
|||
|
);
|
|||
|
|
|||
|
if ( HandleTableTable == NULL ) {
|
|||
|
|
|||
|
ExFreePool( HandleTable );
|
|||
|
|
|||
|
if (ARGUMENT_PRESENT(Process)) {
|
|||
|
|
|||
|
PsReturnProcessPagedPoolQuota (Process,
|
|||
|
sizeof(HANDLE_TABLE));
|
|||
|
}
|
|||
|
|
|||
|
return NULL;
|
|||
|
}
|
|||
|
|
|||
|
HandleTable->TableCode = (ULONG_PTR)HandleTableTable;
|
|||
|
|
|||
|
|
|||
|
//
|
|||
|
// We stamp with EX_ADDITIONAL_INFO_SIGNATURE to recognize in the future this
|
|||
|
// is a special information entry
|
|||
|
//
|
|||
|
|
|||
|
HandleEntry = &HandleTableTable[0];
|
|||
|
|
|||
|
HandleEntry->NextFreeTableEntry = EX_ADDITIONAL_INFO_SIGNATURE;
|
|||
|
HandleEntry->Value = 0;
|
|||
|
|
|||
|
//
|
|||
|
// For duplicate calls we skip building the free list as we rebuild it manually as
|
|||
|
// we traverse the old table we are duplicating
|
|||
|
//
|
|||
|
if (DoInit) {
|
|||
|
HandleEntry++;
|
|||
|
//
|
|||
|
// Now setup the free list. We do this by chaining together the free
|
|||
|
// entries such that each free entry give the next free index (i.e.,
|
|||
|
// like a fat chain). The chain is terminated with a 0. Note that
|
|||
|
// we'll skip handle zero because our callers will get that value
|
|||
|
// confused with null.
|
|||
|
//
|
|||
|
|
|||
|
|
|||
|
for (i = 1; i < LOWLEVEL_COUNT - 1; i += 1) {
|
|||
|
|
|||
|
HandleEntry->Value = 0;
|
|||
|
HandleEntry->NextFreeTableEntry = (i+1)*HANDLE_VALUE_INC;
|
|||
|
HandleEntry++;
|
|||
|
}
|
|||
|
HandleEntry->Value = 0;
|
|||
|
HandleEntry->NextFreeTableEntry = 0;
|
|||
|
|
|||
|
HandleTable->FirstFree = HANDLE_VALUE_INC;
|
|||
|
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
HandleTable->NextHandleNeedingPool = LOWLEVEL_COUNT * HANDLE_VALUE_INC;
|
|||
|
|
|||
|
//
|
|||
|
// Setup the necessary process information
|
|||
|
//
|
|||
|
|
|||
|
HandleTable->QuotaProcess = Process;
|
|||
|
HandleTable->UniqueProcessId = PsGetCurrentProcess()->UniqueProcessId;
|
|||
|
HandleTable->Flags = 0;
|
|||
|
|
|||
|
#if DBG && !EXHANDLE_EXTRA_CHECKS
|
|||
|
if (Process != NULL) {
|
|||
|
HandleTable->StrictFIFO = TRUE;
|
|||
|
}
|
|||
|
#endif
|
|||
|
|
|||
|
//
|
|||
|
// Initialize the handle table lock. This is only used by table expansion.
|
|||
|
//
|
|||
|
|
|||
|
for (Idx = 0; Idx < HANDLE_TABLE_LOCKS; Idx++) {
|
|||
|
ExInitializePushLock (&HandleTable->HandleTableLock[Idx]);
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// Initialize the blocker for handle entry lock contention.
|
|||
|
//
|
|||
|
|
|||
|
ExInitializePushLock (&HandleTable->HandleContentionEvent);
|
|||
|
|
|||
|
if (ExTraceAllTables) {
|
|||
|
ExEnableHandleTracing (HandleTable, 0);
|
|||
|
}
|
|||
|
//
|
|||
|
// And return to our caller
|
|||
|
//
|
|||
|
|
|||
|
return HandleTable;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
//
|
|||
|
// Local Support Routine
|
|||
|
//
|
|||
|
|
|||
|
VOID
|
|||
|
ExpFreeLowLevelTable (
|
|||
|
IN PEPROCESS QuotaProcess,
|
|||
|
IN PHANDLE_TABLE_ENTRY TableLevel1
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
This worker routine frees a low-level handle table
|
|||
|
and the additional info memory, if any.
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
HandleTable - Supplies the handle table being freed
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
None.
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
//
|
|||
|
// Check whether we have a pool allocated for the additional info
|
|||
|
//
|
|||
|
|
|||
|
if (TableLevel1[0].Object) {
|
|||
|
|
|||
|
ExpFreeTablePagedPool( QuotaProcess,
|
|||
|
TableLevel1[0].Object,
|
|||
|
LOWLEVEL_COUNT * sizeof(HANDLE_TABLE_ENTRY_INFO)
|
|||
|
);
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// Now free the low level table and return the quota for the process
|
|||
|
//
|
|||
|
|
|||
|
ExpFreeTablePagedPool( QuotaProcess,
|
|||
|
TableLevel1,
|
|||
|
TABLE_PAGE_SIZE
|
|||
|
);
|
|||
|
|
|||
|
//
|
|||
|
// And return to our caller
|
|||
|
//
|
|||
|
|
|||
|
return;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
//
|
|||
|
// Local Support Routine
|
|||
|
//
|
|||
|
|
|||
|
VOID
|
|||
|
ExpFreeHandleTable (
|
|||
|
IN PHANDLE_TABLE HandleTable
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
This worker routine tears down and frees the specified handle table.
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
HandleTable - Supplies the handle table being freed
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
None.
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
PEPROCESS Process;
|
|||
|
ULONG i,j;
|
|||
|
ULONG_PTR CapturedTable = HandleTable->TableCode;
|
|||
|
ULONG TableLevel = (ULONG)(CapturedTable & LEVEL_CODE_MASK);
|
|||
|
|
|||
|
PAGED_CODE();
|
|||
|
|
|||
|
//
|
|||
|
// Unmask the level bits
|
|||
|
//
|
|||
|
|
|||
|
CapturedTable = CapturedTable & ~LEVEL_CODE_MASK;
|
|||
|
Process = HandleTable->QuotaProcess;
|
|||
|
|
|||
|
//
|
|||
|
// We need to free all pages. We have 3 cases, depending on the number
|
|||
|
// of levels
|
|||
|
//
|
|||
|
|
|||
|
|
|||
|
if (TableLevel == 0) {
|
|||
|
|
|||
|
//
|
|||
|
// There is a single level handle table. We'll simply free the buffer
|
|||
|
//
|
|||
|
|
|||
|
PHANDLE_TABLE_ENTRY TableLevel1 = (PHANDLE_TABLE_ENTRY)CapturedTable;
|
|||
|
|
|||
|
ExpFreeLowLevelTable( Process, TableLevel1 );
|
|||
|
|
|||
|
} else if (TableLevel == 1) {
|
|||
|
|
|||
|
//
|
|||
|
// We have 2 levels in the handle table
|
|||
|
//
|
|||
|
|
|||
|
PHANDLE_TABLE_ENTRY *TableLevel2 = (PHANDLE_TABLE_ENTRY *)CapturedTable;
|
|||
|
|
|||
|
for (i = 0; i < MIDLEVEL_COUNT; i++) {
|
|||
|
|
|||
|
//
|
|||
|
// loop through the pointers to the low-level tables, and free each one
|
|||
|
//
|
|||
|
|
|||
|
if (TableLevel2[i] == NULL) {
|
|||
|
|
|||
|
break;
|
|||
|
}
|
|||
|
|
|||
|
ExpFreeLowLevelTable( Process, TableLevel2[i] );
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// Free the top level table
|
|||
|
//
|
|||
|
|
|||
|
ExpFreeTablePagedPool( Process,
|
|||
|
TableLevel2,
|
|||
|
PAGE_SIZE
|
|||
|
);
|
|||
|
|
|||
|
} else {
|
|||
|
|
|||
|
//
|
|||
|
// Here we handle the case where we have a 3 level handle table
|
|||
|
//
|
|||
|
|
|||
|
PHANDLE_TABLE_ENTRY **TableLevel3 = (PHANDLE_TABLE_ENTRY **)CapturedTable;
|
|||
|
|
|||
|
//
|
|||
|
// Iterates through the high-level pointers to mid-table
|
|||
|
//
|
|||
|
|
|||
|
for (i = 0; i < HIGHLEVEL_COUNT; i++) {
|
|||
|
|
|||
|
if (TableLevel3[i] == NULL) {
|
|||
|
|
|||
|
break;
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// Iterate through the mid-level table
|
|||
|
// and free every low-level page
|
|||
|
//
|
|||
|
|
|||
|
for (j = 0; j < MIDLEVEL_COUNT; j++) {
|
|||
|
|
|||
|
if (TableLevel3[i][j] == NULL) {
|
|||
|
|
|||
|
break;
|
|||
|
}
|
|||
|
|
|||
|
ExpFreeLowLevelTable( Process, TableLevel3[i][j] );
|
|||
|
}
|
|||
|
|
|||
|
ExpFreeTablePagedPool( Process,
|
|||
|
TableLevel3[i],
|
|||
|
PAGE_SIZE
|
|||
|
);
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// Free the top-level array
|
|||
|
//
|
|||
|
|
|||
|
ExpFreeTablePagedPool( Process,
|
|||
|
TableLevel3,
|
|||
|
HIGHLEVEL_SIZE
|
|||
|
);
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// Free any debug info if we have any.
|
|||
|
//
|
|||
|
|
|||
|
if (HandleTable->DebugInfo != NULL) {
|
|||
|
ExDereferenceHandleDebugInfo (HandleTable, HandleTable->DebugInfo);
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// Finally deallocate the handle table itself
|
|||
|
//
|
|||
|
|
|||
|
ExFreePool( HandleTable );
|
|||
|
|
|||
|
if (Process != NULL) {
|
|||
|
|
|||
|
PsReturnProcessPagedPoolQuota (Process,
|
|||
|
sizeof(HANDLE_TABLE));
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// And return to our caller
|
|||
|
//
|
|||
|
|
|||
|
return;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
//
|
|||
|
// Local Support Routine
|
|||
|
//
|
|||
|
|
|||
|
PHANDLE_TABLE_ENTRY
|
|||
|
ExpAllocateLowLevelTable (
|
|||
|
IN PHANDLE_TABLE HandleTable,
|
|||
|
IN BOOLEAN DoInit
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
This worker routine allocates a new low level table
|
|||
|
|
|||
|
Note: The caller must have already locked the handle table
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
HandleTable - Supplies the handle table being used
|
|||
|
|
|||
|
DoInit - If FALSE the caller (duplicate) doesn't need the free list maintained
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
Returns - a pointer to a low-level table if allocation is
|
|||
|
successful otherwise the return value is null.
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
ULONG k;
|
|||
|
PHANDLE_TABLE_ENTRY NewLowLevel = NULL, HandleEntry;
|
|||
|
ULONG BaseHandle;
|
|||
|
|
|||
|
//
|
|||
|
// Allocate the pool for lower level
|
|||
|
//
|
|||
|
|
|||
|
NewLowLevel = ExpAllocateTablePagedPoolNoZero( HandleTable->QuotaProcess,
|
|||
|
TABLE_PAGE_SIZE
|
|||
|
);
|
|||
|
|
|||
|
if (NewLowLevel == NULL) {
|
|||
|
|
|||
|
return NULL;
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// We stamp with EX_ADDITIONAL_INFO_SIGNATURE to recognize in the future this
|
|||
|
// is a special information entry
|
|||
|
//
|
|||
|
|
|||
|
HandleEntry = &NewLowLevel[0];
|
|||
|
|
|||
|
HandleEntry->NextFreeTableEntry = EX_ADDITIONAL_INFO_SIGNATURE;
|
|||
|
HandleEntry->Value = 0;
|
|||
|
|
|||
|
//
|
|||
|
// Initialize the free list within this page if the caller wants this
|
|||
|
//
|
|||
|
if (DoInit) {
|
|||
|
|
|||
|
HandleEntry++;
|
|||
|
|
|||
|
//
|
|||
|
// Now add the new entries to the free list. To do this we
|
|||
|
// chain the new free entries together. We are guaranteed to
|
|||
|
// have at least one new buffer. The second buffer we need
|
|||
|
// to check for.
|
|||
|
//
|
|||
|
// We reserve the first entry in the table to the structure with
|
|||
|
// additional info
|
|||
|
//
|
|||
|
//
|
|||
|
// Do the guaranteed first buffer
|
|||
|
//
|
|||
|
|
|||
|
BaseHandle = HandleTable->NextHandleNeedingPool + 2 * HANDLE_VALUE_INC;
|
|||
|
for (k = BaseHandle; k < BaseHandle + (LOWLEVEL_COUNT - 2) * HANDLE_VALUE_INC; k += HANDLE_VALUE_INC) {
|
|||
|
|
|||
|
HandleEntry->NextFreeTableEntry = k;
|
|||
|
HandleEntry->Value = 0;
|
|||
|
HandleEntry++;
|
|||
|
}
|
|||
|
HandleEntry->NextFreeTableEntry = 0;
|
|||
|
HandleEntry->Value = 0;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
return NewLowLevel;
|
|||
|
}
|
|||
|
|
|||
|
PHANDLE_TABLE_ENTRY *
|
|||
|
ExpAllocateMidLevelTable (
|
|||
|
IN PHANDLE_TABLE HandleTable,
|
|||
|
IN BOOLEAN DoInit,
|
|||
|
OUT PHANDLE_TABLE_ENTRY *pNewLowLevel
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
This worker routine allocates a mid-level table. This is an array with
|
|||
|
pointers to low-level tables.
|
|||
|
It will allocate also a low-level table and will save it in the first index
|
|||
|
|
|||
|
Note: The caller must have already locked the handle table
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
HandleTable - Supplies the handle table being used
|
|||
|
|
|||
|
DoInit - If FALSE the caller (duplicate) does not want the free list build
|
|||
|
|
|||
|
pNewLowLevel - Returns the new low level taible for later free list chaining
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
Returns a pointer to the new mid-level table allocated
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
PHANDLE_TABLE_ENTRY *NewMidLevel;
|
|||
|
PHANDLE_TABLE_ENTRY NewLowLevel;
|
|||
|
|
|||
|
NewMidLevel = ExpAllocateTablePagedPool( HandleTable->QuotaProcess,
|
|||
|
PAGE_SIZE
|
|||
|
);
|
|||
|
|
|||
|
if (NewMidLevel == NULL) {
|
|||
|
|
|||
|
return NULL;
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// If we need a new mid-level, we'll need a low-level too.
|
|||
|
// We'll create one and if success we'll save it at the first position
|
|||
|
//
|
|||
|
|
|||
|
NewLowLevel = ExpAllocateLowLevelTable( HandleTable, DoInit );
|
|||
|
|
|||
|
if (NewLowLevel == NULL) {
|
|||
|
|
|||
|
ExpFreeTablePagedPool( HandleTable->QuotaProcess,
|
|||
|
NewMidLevel,
|
|||
|
PAGE_SIZE
|
|||
|
);
|
|||
|
|
|||
|
return NULL;
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// Set the low-level table at the first index
|
|||
|
//
|
|||
|
|
|||
|
NewMidLevel[0] = NewLowLevel;
|
|||
|
*pNewLowLevel = NewLowLevel;
|
|||
|
|
|||
|
return NewMidLevel;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
|
|||
|
BOOLEAN
|
|||
|
ExpAllocateHandleTableEntrySlow (
|
|||
|
IN PHANDLE_TABLE HandleTable,
|
|||
|
IN BOOLEAN DoInit
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
This worker routine allocates a new handle table entry for the specified
|
|||
|
handle table.
|
|||
|
|
|||
|
Note: The caller must have already locked the handle table
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
HandleTable - Supplies the handle table being used
|
|||
|
|
|||
|
DoInit - If FALSE then the caller (duplicate) doesn't need the free list built
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
BOOLEAN - TRUE, Retry the fast allocation path, FALSE, We failed to allocate memory
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
ULONG i,j;
|
|||
|
|
|||
|
PHANDLE_TABLE_ENTRY NewLowLevel;
|
|||
|
PHANDLE_TABLE_ENTRY *NewMidLevel;
|
|||
|
PHANDLE_TABLE_ENTRY **NewHighLevel;
|
|||
|
ULONG NewFree, OldFree;
|
|||
|
ULONG OldIndex;
|
|||
|
PVOID OldValue;
|
|||
|
|
|||
|
ULONG_PTR CapturedTable = HandleTable->TableCode;
|
|||
|
ULONG TableLevel = (ULONG)(CapturedTable & LEVEL_CODE_MASK);
|
|||
|
|
|||
|
PAGED_CODE();
|
|||
|
|
|||
|
//
|
|||
|
// Initializing NewLowLevel is not needed for
|
|||
|
// correctness but without it the compiler cannot compile this code
|
|||
|
// W4 to check for use of uninitialized variables.
|
|||
|
//
|
|||
|
|
|||
|
NewLowLevel = NULL;
|
|||
|
|
|||
|
CapturedTable = CapturedTable & ~LEVEL_CODE_MASK;
|
|||
|
|
|||
|
|
|||
|
if ( TableLevel == 0 ) {
|
|||
|
|
|||
|
//
|
|||
|
// We have a single level. We need to ad a mid-layer
|
|||
|
// to the process handle table
|
|||
|
//
|
|||
|
|
|||
|
NewMidLevel = ExpAllocateMidLevelTable( HandleTable, DoInit, &NewLowLevel );
|
|||
|
|
|||
|
if (NewMidLevel == NULL) {
|
|||
|
return FALSE;
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// Since ExpAllocateMidLevelTable initialize the
|
|||
|
// first position with a new table, we need to move it in
|
|||
|
// the second position, and store in the first position the current one
|
|||
|
//
|
|||
|
|
|||
|
NewMidLevel[1] = NewMidLevel[0];
|
|||
|
NewMidLevel[0] = (PHANDLE_TABLE_ENTRY)CapturedTable;
|
|||
|
|
|||
|
//
|
|||
|
// Encode the current level and set it to the handle table process
|
|||
|
//
|
|||
|
|
|||
|
CapturedTable = ((ULONG_PTR)NewMidLevel) | 1;
|
|||
|
|
|||
|
OldValue = InterlockedExchangePointer( (PVOID *)&HandleTable->TableCode, (PVOID)CapturedTable );
|
|||
|
|
|||
|
|
|||
|
} else if (TableLevel == 1) {
|
|||
|
|
|||
|
//
|
|||
|
// We have a 2 levels handle table
|
|||
|
//
|
|||
|
|
|||
|
PHANDLE_TABLE_ENTRY *TableLevel2 = (PHANDLE_TABLE_ENTRY *)CapturedTable;
|
|||
|
|
|||
|
//
|
|||
|
// Test whether the index we need to create is still in the
|
|||
|
// range for a 2 layers table
|
|||
|
//
|
|||
|
|
|||
|
i = HandleTable->NextHandleNeedingPool / (LOWLEVEL_COUNT * HANDLE_VALUE_INC);
|
|||
|
|
|||
|
if (i < MIDLEVEL_COUNT) {
|
|||
|
|
|||
|
//
|
|||
|
// We just need to allocate a new low-level
|
|||
|
// table
|
|||
|
//
|
|||
|
|
|||
|
NewLowLevel = ExpAllocateLowLevelTable( HandleTable, DoInit );
|
|||
|
|
|||
|
if (NewLowLevel == NULL) {
|
|||
|
return FALSE;
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// Set the new one to the table, at appropriate position
|
|||
|
//
|
|||
|
|
|||
|
OldValue = InterlockedExchangePointer( (PVOID *) (&TableLevel2[i]), NewLowLevel );
|
|||
|
EXASSERT (OldValue == NULL);
|
|||
|
|
|||
|
} else {
|
|||
|
|
|||
|
//
|
|||
|
// We exhausted the 2 level domain. We need to insert a new one
|
|||
|
//
|
|||
|
|
|||
|
NewHighLevel = ExpAllocateTablePagedPool( HandleTable->QuotaProcess,
|
|||
|
HIGHLEVEL_SIZE
|
|||
|
);
|
|||
|
|
|||
|
if (NewHighLevel == NULL) {
|
|||
|
|
|||
|
return FALSE;
|
|||
|
}
|
|||
|
|
|||
|
NewMidLevel = ExpAllocateMidLevelTable( HandleTable, DoInit, &NewLowLevel );
|
|||
|
|
|||
|
if (NewMidLevel == NULL) {
|
|||
|
|
|||
|
ExpFreeTablePagedPool( HandleTable->QuotaProcess,
|
|||
|
NewHighLevel,
|
|||
|
HIGHLEVEL_SIZE
|
|||
|
);
|
|||
|
|
|||
|
return FALSE;
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// Initialize the first index with the previous mid-level layer
|
|||
|
//
|
|||
|
|
|||
|
NewHighLevel[0] = (PHANDLE_TABLE_ENTRY*)CapturedTable;
|
|||
|
NewHighLevel[1] = NewMidLevel;
|
|||
|
|
|||
|
//
|
|||
|
// Encode the level into the table pointer
|
|||
|
//
|
|||
|
|
|||
|
CapturedTable = ((ULONG_PTR)NewHighLevel) | 2;
|
|||
|
|
|||
|
//
|
|||
|
// Change the handle table pointer with this one
|
|||
|
//
|
|||
|
|
|||
|
OldValue = InterlockedExchangePointer( (PVOID *)&HandleTable->TableCode, (PVOID)CapturedTable );
|
|||
|
|
|||
|
}
|
|||
|
|
|||
|
} else if (TableLevel == 2) {
|
|||
|
|
|||
|
//
|
|||
|
// we have already a table with 3 levels
|
|||
|
//
|
|||
|
|
|||
|
ULONG RemainingIndex;
|
|||
|
PHANDLE_TABLE_ENTRY **TableLevel3 = (PHANDLE_TABLE_ENTRY **)CapturedTable;
|
|||
|
|
|||
|
i = HandleTable->NextHandleNeedingPool / (MIDLEVEL_THRESHOLD * HANDLE_VALUE_INC);
|
|||
|
|
|||
|
//
|
|||
|
// Check whether we exhausted all possible indexes.
|
|||
|
//
|
|||
|
|
|||
|
if (i >= HIGHLEVEL_COUNT) {
|
|||
|
|
|||
|
return FALSE;
|
|||
|
}
|
|||
|
|
|||
|
if (TableLevel3[i] == NULL) {
|
|||
|
|
|||
|
//
|
|||
|
// The new available handle points to a free mid-level entry
|
|||
|
// We need then to allocate a new one and save it in that position
|
|||
|
//
|
|||
|
|
|||
|
NewMidLevel = ExpAllocateMidLevelTable( HandleTable, DoInit, &NewLowLevel );
|
|||
|
|
|||
|
if (NewMidLevel == NULL) {
|
|||
|
|
|||
|
return FALSE;
|
|||
|
}
|
|||
|
|
|||
|
OldValue = InterlockedExchangePointer( (PVOID *) &(TableLevel3[i]), NewMidLevel );
|
|||
|
EXASSERT (OldValue == NULL);
|
|||
|
|
|||
|
} else {
|
|||
|
|
|||
|
//
|
|||
|
// We have already a mid-level table. We just need to add a new low-level one
|
|||
|
// at the end
|
|||
|
//
|
|||
|
|
|||
|
RemainingIndex = (HandleTable->NextHandleNeedingPool / HANDLE_VALUE_INC) -
|
|||
|
i * MIDLEVEL_THRESHOLD;
|
|||
|
j = RemainingIndex / LOWLEVEL_COUNT;
|
|||
|
|
|||
|
NewLowLevel = ExpAllocateLowLevelTable( HandleTable, DoInit );
|
|||
|
|
|||
|
if (NewLowLevel == NULL) {
|
|||
|
|
|||
|
return FALSE;
|
|||
|
}
|
|||
|
|
|||
|
OldValue = InterlockedExchangePointer( (PVOID *)(&TableLevel3[i][j]) , NewLowLevel );
|
|||
|
EXASSERT (OldValue == NULL);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// This must be done after the table pointers so that new created handles
|
|||
|
// are valid before being freed.
|
|||
|
//
|
|||
|
OldIndex = InterlockedExchangeAdd ((PLONG) &HandleTable->NextHandleNeedingPool,
|
|||
|
LOWLEVEL_COUNT * HANDLE_VALUE_INC);
|
|||
|
|
|||
|
|
|||
|
if (DoInit) {
|
|||
|
//
|
|||
|
// Generate a new sequence number since this is a push
|
|||
|
//
|
|||
|
OldIndex += HANDLE_VALUE_INC + GetNextSeq();
|
|||
|
|
|||
|
//
|
|||
|
// Now free the handles. These are all ready to be accepted by the lookup logic now.
|
|||
|
//
|
|||
|
while (1) {
|
|||
|
OldFree = HandleTable->FirstFree;
|
|||
|
NewLowLevel[LOWLEVEL_COUNT - 1].NextFreeTableEntry = OldFree;
|
|||
|
|
|||
|
//
|
|||
|
// These are new entries that have never existed before. We can't have an A-B-A problem
|
|||
|
// with these so we don't need to take any locks
|
|||
|
//
|
|||
|
|
|||
|
|
|||
|
NewFree = InterlockedCompareExchange ((PLONG)&HandleTable->FirstFree,
|
|||
|
OldIndex,
|
|||
|
OldFree);
|
|||
|
if (NewFree == OldFree) {
|
|||
|
break;
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
return TRUE;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
VOID
|
|||
|
ExSetHandleTableStrictFIFO (
|
|||
|
IN PHANDLE_TABLE HandleTable
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
This routine marks a handle table so that handle allocation is done in
|
|||
|
a strict FIFO order.
|
|||
|
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
HandleTable - Supplies the handle table being changed to FIFO
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
None.
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
HandleTable->StrictFIFO = TRUE;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
//
|
|||
|
// Local Support Routine
|
|||
|
//
|
|||
|
|
|||
|
//
|
|||
|
// The following is a global variable only present in the checked builds
|
|||
|
// to help catch apps that reuse handle values after they're closed.
|
|||
|
//
|
|||
|
|
|||
|
#if DBG
|
|||
|
BOOLEAN ExReuseHandles = 1;
|
|||
|
#endif //DBG
|
|||
|
|
|||
|
VOID
|
|||
|
ExpFreeHandleTableEntry (
|
|||
|
IN PHANDLE_TABLE HandleTable,
|
|||
|
IN EXHANDLE Handle,
|
|||
|
IN PHANDLE_TABLE_ENTRY HandleTableEntry
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
This worker routine returns the specified handle table entry to the free
|
|||
|
list for the handle table.
|
|||
|
|
|||
|
Note: The caller must have already locked the handle table
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
HandleTable - Supplies the parent handle table being modified
|
|||
|
|
|||
|
Handle - Supplies the handle of the entry being freed
|
|||
|
|
|||
|
HandleTableEntry - Supplies the table entry being freed
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
None.
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
PHANDLE_TABLE_ENTRY_INFO EntryInfo;
|
|||
|
ULONG OldFree, NewFree, *Free;
|
|||
|
PKTHREAD CurrentThread;
|
|||
|
ULONG Idx;
|
|||
|
ULONG SeqInc;
|
|||
|
|
|||
|
PAGED_CODE();
|
|||
|
|
|||
|
EXASSERT (HandleTableEntry->Object == NULL);
|
|||
|
EXASSERT (HandleTableEntry == ExpLookupHandleTableEntry (HandleTable, Handle));
|
|||
|
|
|||
|
//
|
|||
|
// Clear the AuditMask flags if these are present into the table
|
|||
|
//
|
|||
|
|
|||
|
EntryInfo = ExGetHandleInfo(HandleTable, Handle.GenericHandleOverlay, TRUE);
|
|||
|
|
|||
|
if (EntryInfo) {
|
|||
|
|
|||
|
EntryInfo->AuditMask = 0;
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// A free is simply a push onto the free table entry stack, or in the
|
|||
|
// debug case we'll sometimes just float the entry to catch apps who
|
|||
|
// reuse a recycled handle value.
|
|||
|
//
|
|||
|
|
|||
|
InterlockedDecrement (&HandleTable->HandleCount);
|
|||
|
CurrentThread = KeGetCurrentThread ();
|
|||
|
|
|||
|
NewFree = (ULONG) Handle.Value & ~(HANDLE_VALUE_INC - 1);
|
|||
|
|
|||
|
#if DBG
|
|||
|
if (ExReuseHandles) {
|
|||
|
#endif //DBG
|
|||
|
|
|||
|
if (!HandleTable->StrictFIFO) {
|
|||
|
|
|||
|
|
|||
|
//
|
|||
|
// We are pushing potentialy old entries onto the free list.
|
|||
|
// Prevent the A-B-A problem by shifting to an alternate list
|
|||
|
// read this element has the list head out of the loop.
|
|||
|
//
|
|||
|
Idx = (NewFree>>2) % HANDLE_TABLE_LOCKS;
|
|||
|
if (ExTryAcquireReleasePushLockExclusive (&HandleTable->HandleTableLock[Idx])) {
|
|||
|
SeqInc = GetNextSeq();
|
|||
|
Free = &HandleTable->FirstFree;
|
|||
|
} else {
|
|||
|
SeqInc = 0;
|
|||
|
Free = &HandleTable->LastFree;
|
|||
|
}
|
|||
|
} else {
|
|||
|
SeqInc = 0;
|
|||
|
Free = &HandleTable->LastFree;
|
|||
|
}
|
|||
|
|
|||
|
while (1) {
|
|||
|
|
|||
|
|
|||
|
OldFree = *Free;
|
|||
|
HandleTableEntry->NextFreeTableEntry = OldFree;
|
|||
|
|
|||
|
|
|||
|
if ((ULONG)InterlockedCompareExchange ((PLONG)Free,
|
|||
|
NewFree + SeqInc,
|
|||
|
OldFree) == OldFree) {
|
|||
|
|
|||
|
EXASSERT ((OldFree & FREE_HANDLE_MASK) < HandleTable->NextHandleNeedingPool);
|
|||
|
|
|||
|
break;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
#if DBG
|
|||
|
} else {
|
|||
|
|
|||
|
HandleTableEntry->NextFreeTableEntry = 0;
|
|||
|
}
|
|||
|
#endif //DBG
|
|||
|
|
|||
|
|
|||
|
return;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
//
|
|||
|
// Local Support Routine
|
|||
|
//
|
|||
|
|
|||
|
PHANDLE_TABLE_ENTRY
|
|||
|
ExpLookupHandleTableEntry (
|
|||
|
IN PHANDLE_TABLE HandleTable,
|
|||
|
IN EXHANDLE tHandle
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
This routine looks up and returns the table entry for the
|
|||
|
specified handle value.
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
HandleTable - Supplies the handle table being queried
|
|||
|
|
|||
|
tHandle - Supplies the handle value being queried
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
Returns a pointer to the corresponding table entry for the input
|
|||
|
handle. Or NULL if the handle value is invalid (i.e., too large
|
|||
|
for the tables current allocation.
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
ULONG_PTR i,j,k;
|
|||
|
ULONG_PTR CapturedTable;
|
|||
|
ULONG TableLevel;
|
|||
|
PHANDLE_TABLE_ENTRY Entry = NULL;
|
|||
|
EXHANDLE Handle;
|
|||
|
|
|||
|
PUCHAR TableLevel1;
|
|||
|
PUCHAR TableLevel2;
|
|||
|
PUCHAR TableLevel3;
|
|||
|
|
|||
|
ULONG_PTR MaxHandle;
|
|||
|
|
|||
|
PAGED_CODE();
|
|||
|
|
|||
|
|
|||
|
//
|
|||
|
// Extract the handle index
|
|||
|
//
|
|||
|
Handle = tHandle;
|
|||
|
|
|||
|
Handle.TagBits = 0;
|
|||
|
|
|||
|
MaxHandle = *(volatile ULONG *) &HandleTable->NextHandleNeedingPool;
|
|||
|
|
|||
|
//
|
|||
|
// See if this can be a valid handle given the table levels.
|
|||
|
//
|
|||
|
if (Handle.Value >= MaxHandle) {
|
|||
|
return NULL;
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// Now fetch the table address and level bits. We must preserve the
|
|||
|
// ordering here.
|
|||
|
//
|
|||
|
CapturedTable = *(volatile ULONG_PTR *) &HandleTable->TableCode;
|
|||
|
|
|||
|
//
|
|||
|
// we need to capture the current table. This routine is lock free
|
|||
|
// so another thread may change the table at HandleTable->TableCode
|
|||
|
//
|
|||
|
|
|||
|
TableLevel = (ULONG)(CapturedTable & LEVEL_CODE_MASK);
|
|||
|
CapturedTable = CapturedTable - TableLevel;
|
|||
|
|
|||
|
//
|
|||
|
// The lookup code depends on number of levels we have
|
|||
|
//
|
|||
|
|
|||
|
switch (TableLevel) {
|
|||
|
|
|||
|
case 0:
|
|||
|
|
|||
|
//
|
|||
|
// We have a simple index into the array, for a single level
|
|||
|
// handle table
|
|||
|
//
|
|||
|
|
|||
|
|
|||
|
TableLevel1 = (PUCHAR) CapturedTable;
|
|||
|
|
|||
|
//
|
|||
|
// The index for this level is already scaled by a factor of 4. Take advantage of this
|
|||
|
//
|
|||
|
|
|||
|
Entry = (PHANDLE_TABLE_ENTRY) &TableLevel1[Handle.Value *
|
|||
|
(sizeof (HANDLE_TABLE_ENTRY) / HANDLE_VALUE_INC)];
|
|||
|
|
|||
|
break;
|
|||
|
|
|||
|
case 1:
|
|||
|
|
|||
|
//
|
|||
|
// we have a 2 level handle table. We need to get the upper index
|
|||
|
// and lower index into the array
|
|||
|
//
|
|||
|
|
|||
|
|
|||
|
TableLevel2 = (PUCHAR) CapturedTable;
|
|||
|
|
|||
|
i = Handle.Value % (LOWLEVEL_COUNT * HANDLE_VALUE_INC);
|
|||
|
|
|||
|
Handle.Value -= i;
|
|||
|
j = Handle.Value / ((LOWLEVEL_COUNT * HANDLE_VALUE_INC) / sizeof (PHANDLE_TABLE_ENTRY));
|
|||
|
|
|||
|
TableLevel1 = (PUCHAR) *(PHANDLE_TABLE_ENTRY *) &TableLevel2[j];
|
|||
|
Entry = (PHANDLE_TABLE_ENTRY) &TableLevel1[i * (sizeof (HANDLE_TABLE_ENTRY) / HANDLE_VALUE_INC)];
|
|||
|
|
|||
|
break;
|
|||
|
|
|||
|
case 2:
|
|||
|
|
|||
|
//
|
|||
|
// We have here a three level handle table.
|
|||
|
//
|
|||
|
|
|||
|
|
|||
|
TableLevel3 = (PUCHAR) CapturedTable;
|
|||
|
|
|||
|
i = Handle.Value % (LOWLEVEL_COUNT * HANDLE_VALUE_INC);
|
|||
|
|
|||
|
Handle.Value -= i;
|
|||
|
|
|||
|
k = Handle.Value / ((LOWLEVEL_COUNT * HANDLE_VALUE_INC) / sizeof (PHANDLE_TABLE_ENTRY));
|
|||
|
|
|||
|
j = k % (MIDLEVEL_COUNT * sizeof (PHANDLE_TABLE_ENTRY));
|
|||
|
|
|||
|
k -= j;
|
|||
|
|
|||
|
k /= MIDLEVEL_COUNT;
|
|||
|
|
|||
|
|
|||
|
TableLevel2 = (PUCHAR) *(PHANDLE_TABLE_ENTRY *) &TableLevel3[k];
|
|||
|
TableLevel1 = (PUCHAR) *(PHANDLE_TABLE_ENTRY *) &TableLevel2[j];
|
|||
|
Entry = (PHANDLE_TABLE_ENTRY) &TableLevel1[i * (sizeof (HANDLE_TABLE_ENTRY) / HANDLE_VALUE_INC)];
|
|||
|
|
|||
|
break;
|
|||
|
|
|||
|
default :
|
|||
|
_assume (0);
|
|||
|
}
|
|||
|
|
|||
|
return Entry;
|
|||
|
}
|
|||
|
|
|||
|
NTKERNELAPI
|
|||
|
NTSTATUS
|
|||
|
ExSetHandleInfo (
|
|||
|
IN PHANDLE_TABLE HandleTable,
|
|||
|
IN HANDLE Handle,
|
|||
|
IN PHANDLE_TABLE_ENTRY_INFO EntryInfo,
|
|||
|
IN BOOLEAN EntryLocked
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
The routine set the entry info for the specified handle table
|
|||
|
|
|||
|
Note: the handle entry must be locked when this function is called
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
HandleTable - Supplies the handle table being queried
|
|||
|
|
|||
|
Handle - Supplies the handle value being queried
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
PKTHREAD CurrentThread;
|
|||
|
PHANDLE_TABLE_ENTRY InfoStructure;
|
|||
|
EXHANDLE ExHandle;
|
|||
|
NTSTATUS Status;
|
|||
|
PHANDLE_TABLE_ENTRY TableEntry;
|
|||
|
PHANDLE_TABLE_ENTRY_INFO InfoTable;
|
|||
|
|
|||
|
Status = STATUS_UNSUCCESSFUL;
|
|||
|
TableEntry = NULL;
|
|||
|
CurrentThread = NULL;
|
|||
|
|
|||
|
ExHandle.GenericHandleOverlay = Handle;
|
|||
|
ExHandle.Index &= ~(LOWLEVEL_COUNT - 1);
|
|||
|
|
|||
|
if (!EntryLocked) {
|
|||
|
CurrentThread = KeGetCurrentThread ();
|
|||
|
KeEnterCriticalRegionThread (CurrentThread);
|
|||
|
TableEntry = ExMapHandleToPointer(HandleTable, Handle);
|
|||
|
|
|||
|
if (TableEntry == NULL) {
|
|||
|
KeLeaveCriticalRegionThread (CurrentThread);
|
|||
|
|
|||
|
return STATUS_UNSUCCESSFUL;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// The info structure is at the first position in each low-level table
|
|||
|
//
|
|||
|
|
|||
|
InfoStructure = ExpLookupHandleTableEntry( HandleTable,
|
|||
|
ExHandle
|
|||
|
);
|
|||
|
|
|||
|
if (InfoStructure == NULL || InfoStructure->NextFreeTableEntry != EX_ADDITIONAL_INFO_SIGNATURE) {
|
|||
|
|
|||
|
if ( TableEntry ) {
|
|||
|
ExUnlockHandleTableEntry( HandleTable, TableEntry );
|
|||
|
KeLeaveCriticalRegionThread (CurrentThread);
|
|||
|
}
|
|||
|
|
|||
|
return STATUS_UNSUCCESSFUL;
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// Check whether we need to allocate a new table
|
|||
|
//
|
|||
|
InfoTable = InfoStructure->InfoTable;
|
|||
|
if (InfoTable == NULL) {
|
|||
|
//
|
|||
|
// Nobody allocated the Infotable so far.
|
|||
|
// We'll do it right now
|
|||
|
//
|
|||
|
|
|||
|
InfoTable = ExpAllocateTablePagedPool (HandleTable->QuotaProcess,
|
|||
|
LOWLEVEL_COUNT * sizeof(HANDLE_TABLE_ENTRY_INFO));
|
|||
|
|
|||
|
if (InfoTable) {
|
|||
|
|
|||
|
//
|
|||
|
// Update the number of pages for extra info. If somebody beat us to it then free the
|
|||
|
// new table
|
|||
|
//
|
|||
|
if (InterlockedCompareExchangePointer (&InfoStructure->InfoTable,
|
|||
|
InfoTable,
|
|||
|
NULL) == NULL) {
|
|||
|
|
|||
|
InterlockedIncrement(&HandleTable->ExtraInfoPages);
|
|||
|
|
|||
|
} else {
|
|||
|
ExpFreeTablePagedPool (HandleTable->QuotaProcess,
|
|||
|
InfoTable,
|
|||
|
LOWLEVEL_COUNT * sizeof(HANDLE_TABLE_ENTRY_INFO));
|
|||
|
InfoTable = InfoStructure->InfoTable;
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
if (InfoTable != NULL) {
|
|||
|
|
|||
|
//
|
|||
|
// Calculate the index and copy the structure
|
|||
|
//
|
|||
|
|
|||
|
ExHandle.GenericHandleOverlay = Handle;
|
|||
|
|
|||
|
InfoTable[ExHandle.Index % LOWLEVEL_COUNT] = *EntryInfo;
|
|||
|
|
|||
|
Status = STATUS_SUCCESS;
|
|||
|
}
|
|||
|
|
|||
|
if ( TableEntry ) {
|
|||
|
|
|||
|
ExUnlockHandleTableEntry( HandleTable, TableEntry );
|
|||
|
KeLeaveCriticalRegionThread (CurrentThread);
|
|||
|
}
|
|||
|
|
|||
|
return Status;
|
|||
|
}
|
|||
|
|
|||
|
NTKERNELAPI
|
|||
|
PHANDLE_TABLE_ENTRY_INFO
|
|||
|
ExpGetHandleInfo (
|
|||
|
IN PHANDLE_TABLE HandleTable,
|
|||
|
IN HANDLE Handle,
|
|||
|
IN BOOLEAN EntryLocked
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
The routine reads the entry info for the specified handle table
|
|||
|
|
|||
|
Note: the handle entry must be locked when this function is called
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
HandleTable - Supplies the handle table being queried
|
|||
|
|
|||
|
Handle - Supplies the handle value being queried
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
PHANDLE_TABLE_ENTRY InfoStructure;
|
|||
|
EXHANDLE ExHandle;
|
|||
|
PHANDLE_TABLE_ENTRY TableEntry = NULL;
|
|||
|
|
|||
|
ExHandle.GenericHandleOverlay = Handle;
|
|||
|
ExHandle.Index &= ~(LOWLEVEL_COUNT - 1);
|
|||
|
|
|||
|
if (!EntryLocked) {
|
|||
|
|
|||
|
TableEntry = ExMapHandleToPointer(HandleTable, Handle);
|
|||
|
|
|||
|
if (TableEntry == NULL) {
|
|||
|
|
|||
|
return NULL;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// The info structure is at the first position in each low-level table
|
|||
|
//
|
|||
|
|
|||
|
InfoStructure = ExpLookupHandleTableEntry( HandleTable,
|
|||
|
ExHandle
|
|||
|
);
|
|||
|
|
|||
|
if (InfoStructure == NULL || InfoStructure->NextFreeTableEntry != EX_ADDITIONAL_INFO_SIGNATURE ||
|
|||
|
InfoStructure->InfoTable == NULL) {
|
|||
|
|
|||
|
if ( TableEntry ) {
|
|||
|
|
|||
|
ExUnlockHandleTableEntry( HandleTable, TableEntry );
|
|||
|
}
|
|||
|
|
|||
|
return NULL;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
//
|
|||
|
// Return a pointer to the info structure
|
|||
|
//
|
|||
|
|
|||
|
ExHandle.GenericHandleOverlay = Handle;
|
|||
|
|
|||
|
return &(InfoStructure->InfoTable[ExHandle.Index % LOWLEVEL_COUNT]);
|
|||
|
}
|
|||
|
|
|||
|
#if DBG
|
|||
|
ULONG g_ulExpUpdateDebugInfoDebugLevel = 0;
|
|||
|
#endif
|
|||
|
void ExpUpdateDebugInfo(
|
|||
|
PHANDLE_TABLE HandleTable,
|
|||
|
PETHREAD CurrentThread,
|
|||
|
HANDLE Handle,
|
|||
|
ULONG Type)
|
|||
|
{
|
|||
|
BOOLEAN LockAcquired = FALSE;
|
|||
|
PHANDLE_TRACE_DEBUG_INFO DebugInfo;
|
|||
|
|
|||
|
DebugInfo = ExReferenceHandleDebugInfo (HandleTable);
|
|||
|
|
|||
|
if (DebugInfo == NULL) {
|
|||
|
return;
|
|||
|
}
|
|||
|
|
|||
|
#if DBG
|
|||
|
if (g_ulExpUpdateDebugInfoDebugLevel > 10)
|
|||
|
{
|
|||
|
DbgPrint ("ExpUpdateDebugInfo() BitMaskFlags=0x%x, CurrentStackIndex=%d, Handle=0x%p, Type=%d \n",
|
|||
|
DebugInfo->BitMaskFlags,
|
|||
|
DebugInfo->CurrentStackIndex,
|
|||
|
Handle,
|
|||
|
Type);
|
|||
|
}
|
|||
|
#endif
|
|||
|
if (DebugInfo->BitMaskFlags & (HANDLE_TRACE_DEBUG_INFO_CLEAN_DEBUG_INFO | HANDLE_TRACE_DEBUG_INFO_COMPACT_CLOSE_HANDLE)) {
|
|||
|
//
|
|||
|
// we wish to presenve lock-free behavior in the non-compation path
|
|||
|
// so we lock only in this path
|
|||
|
//
|
|||
|
ExAcquireFastMutex(&DebugInfo->CloseCompactionLock);
|
|||
|
LockAcquired = TRUE;
|
|||
|
}
|
|||
|
|
|||
|
if (DebugInfo->BitMaskFlags & HANDLE_TRACE_DEBUG_INFO_CLEAN_DEBUG_INFO) {
|
|||
|
//
|
|||
|
// clean debug info, but not the fast mutex!
|
|||
|
//
|
|||
|
|
|||
|
ASSERT(LockAcquired);
|
|||
|
|
|||
|
DebugInfo->BitMaskFlags &= ~HANDLE_TRACE_DEBUG_INFO_CLEAN_DEBUG_INFO;
|
|||
|
DebugInfo->BitMaskFlags &= ~HANDLE_TRACE_DEBUG_INFO_WAS_WRAPPED_AROUND;
|
|||
|
DebugInfo->BitMaskFlags |= HANDLE_TRACE_DEBUG_INFO_WAS_SOMETIME_CLEANED;
|
|||
|
DebugInfo->CurrentStackIndex = 0;
|
|||
|
RtlZeroMemory (DebugInfo->TraceDb,
|
|||
|
sizeof (*DebugInfo) +
|
|||
|
DebugInfo->TableSize * sizeof (DebugInfo->TraceDb[0]) -
|
|||
|
sizeof (DebugInfo->TraceDb));
|
|||
|
}
|
|||
|
|
|||
|
if (
|
|||
|
(DebugInfo->BitMaskFlags & HANDLE_TRACE_DEBUG_INFO_COMPACT_CLOSE_HANDLE) &&
|
|||
|
(Type == HANDLE_TRACE_DB_CLOSE)
|
|||
|
){
|
|||
|
//
|
|||
|
// i am asuming that either:
|
|||
|
// 1) this flag was set from the beginning, so there are no close items.
|
|||
|
// 2) this flag was set via KD, in which case HANDLE_TRACE_DEBUG_INFO_CLEAN_DEBUG_INFO
|
|||
|
// must have been set also, so again there are no close items
|
|||
|
//
|
|||
|
ULONG uiMaxNumOfItemsInTraceDb;
|
|||
|
ULONG uiNextItem;
|
|||
|
|
|||
|
ASSERT(LockAcquired);
|
|||
|
|
|||
|
//
|
|||
|
// look for the matching open item, remove them from the list, and compact the list
|
|||
|
//
|
|||
|
uiMaxNumOfItemsInTraceDb = (DebugInfo->BitMaskFlags & HANDLE_TRACE_DEBUG_INFO_WAS_WRAPPED_AROUND) ? DebugInfo->TableSize : DebugInfo->CurrentStackIndex ;
|
|||
|
for (uiNextItem = 1; uiNextItem <= uiMaxNumOfItemsInTraceDb; uiNextItem++) {
|
|||
|
//
|
|||
|
// if HANDLE_TRACE_DEBUG_INFO_COMPACT_CLOSE_HANDLE is on
|
|||
|
// there could be no HANDLE_TRACE_DB_CLOSE items
|
|||
|
// This ASSERT can fire if the HANDLE_TRACE_DEBUG_INFO_COMPACT_CLOSE_HANDLE flag
|
|||
|
// is set dynamically, so another thread was not using the locks to add
|
|||
|
// items to the list, and could have added a HANDLE_TRACE_DB_CLOSE item.
|
|||
|
//
|
|||
|
ASSERT(DebugInfo->TraceDb[uiNextItem%DebugInfo->TableSize].Type != HANDLE_TRACE_DB_CLOSE);
|
|||
|
|
|||
|
if (
|
|||
|
(DebugInfo->TraceDb[uiNextItem%DebugInfo->TableSize].Type == HANDLE_TRACE_DB_OPEN) &&
|
|||
|
(DebugInfo->TraceDb[uiNextItem%DebugInfo->TableSize].Handle == Handle)
|
|||
|
) {
|
|||
|
//
|
|||
|
// found the matching open, compact the list
|
|||
|
//
|
|||
|
ULONG IndexToMoveBack;
|
|||
|
DebugInfo->CurrentStackIndex--;
|
|||
|
IndexToMoveBack = DebugInfo->CurrentStackIndex % DebugInfo->TableSize;
|
|||
|
if (0 != IndexToMoveBack)
|
|||
|
{
|
|||
|
DebugInfo->TraceDb[uiNextItem%DebugInfo->TableSize] = DebugInfo->TraceDb[IndexToMoveBack];
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
//
|
|||
|
// the list is already empty
|
|||
|
//
|
|||
|
}
|
|||
|
break;
|
|||
|
}
|
|||
|
}
|
|||
|
if (!(uiNextItem <= uiMaxNumOfItemsInTraceDb)) {
|
|||
|
//
|
|||
|
// a matching open was not found.
|
|||
|
// this must mean that we wrapped around, or cleaned the list sometime after
|
|||
|
// it was created
|
|||
|
// or that a duplicated handle was created before handle tracking was initiated
|
|||
|
// so we cannot ASSERT for that
|
|||
|
//
|
|||
|
/*
|
|||
|
ASSERT(
|
|||
|
(DebugInfo->BitMaskFlags & HANDLE_TRACE_DEBUG_INFO_WAS_SOMETIME_CLEANED) ||
|
|||
|
(DebugInfo->BitMaskFlags & HANDLE_TRACE_DEBUG_INFO_WAS_WRAPPED_AROUND)
|
|||
|
);
|
|||
|
*/
|
|||
|
//
|
|||
|
// just ignore this HANDLE_TRACE_DB_CLOSE
|
|||
|
//
|
|||
|
}
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
PHANDLE_TRACE_DB_ENTRY DebugEntry;
|
|||
|
ULONG Index = ((ULONG) InterlockedIncrement ((PLONG)&DebugInfo->CurrentStackIndex))
|
|||
|
% DebugInfo->TableSize;
|
|||
|
ASSERT((Type != HANDLE_TRACE_DB_CLOSE) || (!(DebugInfo->BitMaskFlags & HANDLE_TRACE_DEBUG_INFO_COMPACT_CLOSE_HANDLE)));
|
|||
|
if (0 == Index) {
|
|||
|
//
|
|||
|
// this is a wraparound of the db, mark it as such, and if there's
|
|||
|
// a debugger attached break into it
|
|||
|
//
|
|||
|
DebugInfo->BitMaskFlags |= HANDLE_TRACE_DEBUG_INFO_WAS_WRAPPED_AROUND;
|
|||
|
if(DebugInfo->BitMaskFlags & HANDLE_TRACE_DEBUG_INFO_BREAK_ON_WRAP_AROUND)
|
|||
|
{
|
|||
|
__try {
|
|||
|
DbgBreakPoint();
|
|||
|
}
|
|||
|
__except(1) {
|
|||
|
NOTHING;
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
DebugEntry = &DebugInfo->TraceDb[Index];
|
|||
|
DebugEntry->ClientId = CurrentThread->Cid;
|
|||
|
DebugEntry->Handle = Handle;
|
|||
|
DebugEntry->Type = Type;
|
|||
|
Index = RtlWalkFrameChain (DebugEntry->StackTrace, HANDLE_TRACE_DB_STACK_SIZE, 0);
|
|||
|
RtlWalkFrameChain (&DebugEntry->StackTrace[Index], HANDLE_TRACE_DB_STACK_SIZE - Index, 1);
|
|||
|
}
|
|||
|
|
|||
|
if (LockAcquired)
|
|||
|
{
|
|||
|
ExReleaseFastMutex(&DebugInfo->CloseCompactionLock);
|
|||
|
}
|
|||
|
|
|||
|
ExDereferenceHandleDebugInfo (HandleTable, DebugInfo);
|
|||
|
}
|
|||
|
|
|||
|
VOID
|
|||
|
ExHandleTest (
|
|||
|
)
|
|||
|
{
|
|||
|
PHANDLE_TABLE HandleTable;
|
|||
|
ULONG i, j, k;
|
|||
|
#define MAX_ALLOCS 20
|
|||
|
PHANDLE_TABLE_ENTRY HandleEntryArray[MAX_ALLOCS];
|
|||
|
EXHANDLE Handle[MAX_ALLOCS];
|
|||
|
LARGE_INTEGER CurrentTime;
|
|||
|
|
|||
|
HandleTable = PsGetCurrentProcess ()->ObjectTable;
|
|||
|
HandleTable->StrictFIFO = 0;
|
|||
|
|
|||
|
k = 0;
|
|||
|
for (i = 0; i < 100000; i++) {
|
|||
|
KeQuerySystemTime (&CurrentTime);
|
|||
|
for (j = 0; j < MAX_ALLOCS; j++) {
|
|||
|
//
|
|||
|
// Clears Handle.Index and Handle.TagBits
|
|||
|
//
|
|||
|
|
|||
|
Handle[j].GenericHandleOverlay = NULL;
|
|||
|
|
|||
|
|
|||
|
//
|
|||
|
// Allocate a new handle table entry, and get the handle value
|
|||
|
//
|
|||
|
|
|||
|
HandleEntryArray[j] = ExpAllocateHandleTableEntry (HandleTable,
|
|||
|
&Handle[j]);
|
|||
|
if (HandleEntryArray[j] != NULL) {
|
|||
|
HandleEntryArray[j]->NextFreeTableEntry = 0x88888888;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
for (j = MAX_ALLOCS; j > 0; j--) {
|
|||
|
k = k + CurrentTime.LowPart;
|
|||
|
k = k % j;
|
|||
|
CurrentTime.QuadPart >>= 3;
|
|||
|
if (HandleEntryArray[k] != NULL) {
|
|||
|
//
|
|||
|
// Free the entry
|
|||
|
//
|
|||
|
ExpFreeHandleTableEntry (HandleTable,
|
|||
|
Handle[k],
|
|||
|
HandleEntryArray[k]);
|
|||
|
HandleEntryArray[k] = HandleEntryArray[j-1];
|
|||
|
HandleEntryArray[j-1] = NULL;
|
|||
|
Handle[k] = Handle[j-1];
|
|||
|
Handle[j-1].GenericHandleOverlay = NULL;
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
}
|