windows-nt/Source/XPSP1/NT/inetsrv/iis/iisrearc/ul/drv/hash.cxx

1287 lines
35 KiB
C++
Raw Normal View History

2020-09-26 03:20:57 -05:00
/*++
Copyright (c) 1998-2001 Microsoft Corporation
Module Name:
hash.cxx
Abstract:
Contains the HTTP response cache hash table logic.
Author:
Alex Chen (alexch) 28-Mar-2001
Revision History:
George V. Reilly (GeorgeRe) 09-May-2001
Cleaned up and tuned up
--*/
#include "precomp.h"
#include "hashp.h"
// Global Variables
ULONG g_UlHashTableBits;
ULONG g_UlHashTableSize;
ULONG g_UlHashTableMask;
ULONG g_UlHashIndexShift;
//
// Optimization: Use the space of (g_UlCacheLineSize - sizeof (HASHBUCKET))
// to store a few records (Hash, pUriCacheEntry) such that we can scan the
// records first before jumping to the single list for searching.
//
// g_UlNumOfHashUriKeys: The number of stored records in the space.
//
ULONG g_UlNumOfHashUriKeys;
/***************************************************************************++
Routine Description:
This routine determine the hash table size according to
(1) user-define value (reading from registry) or
(2) system memory size estimation, if (1) is not defined
Arguments:
HashTableBits - The number of buckets is (1 << HashTableBits)
--***************************************************************************/
VOID
UlpGetHashTableSize(
IN LONG HashTableBits
)
{
SYSTEM_BASIC_INFORMATION sbi;
ULONG TotalPhysicalMemMB;
NTSTATUS Status;
Status = NtQuerySystemInformation(
SystemBasicInformation,
&sbi,
sizeof(sbi),
NULL);
ASSERT(NT_SUCCESS(Status));
// Capture total physical memory, in terms of megabytes
TotalPhysicalMemMB = PAGES_TO_MEGABYTES(sbi.NumberOfPhysicalPages);
//
// HashTableBits is equal to DEFAULT_HASH_TABLE_BITS
// if it is not defined in the registry
//
if (HashTableBits != DEFAULT_HASH_TABLE_BITS)
{
// Use the registry value
// BUGBUG: We must check for reasonable values, so that a
// malicious or careless user doesn't cause us to eat up
// all of (Non)PagedPool.
g_UlHashTableBits = HashTableBits;
}
else
{
//
// Registry value REGISTRY_HASH_TABLE_BITS is not defined,
// use system memory size estimation instead
//
MM_SYSTEMSIZE SystemSize = MmQuerySystemSize();
if (SystemSize == MmSmallSystem)
{
// Hash Table Size: 4K buckets
g_UlHashTableBits = 12;
}
else if (SystemSize == MmMediumSystem)
{
// Hash Table Size: 16K buckets
g_UlHashTableBits = 14;
}
else
{
// Hash Table Size: 64K buckets
// BUGBUG: A 64KB server is considered an MmLargeSystem.
// Tune g_UlHashTableBits according to physical memory
g_UlHashTableBits = 16;
}
}
g_UlHashIndexShift = g_UlCacheLineBits;
#ifdef HASH_TEST
g_UlHashTableBits = 3;
#endif
g_UlHashTableSize = (1 << g_UlHashTableBits);
g_UlHashTableMask = g_UlHashTableSize - 1;
} // UlpGetHashTableSize
/***************************************************************************++
Routine Description:
Validates that a locked HASHBUCKET is `compact'. If there are less than
g_UlNumOfHashUriKeys entries in a bucket, they are clumped together at
the beginning of the records array, and all the empty slots are at the
end. All empty slots must have Hash == HASH_INVALID_SIGNATURE and
pUriCacheEntry == NULL. Conversely, all the non-empty slots at the
beginning of the array must have point to valid UL_URI_CACHE_ENTRYs
and must have Hash == correct hash signature, which cannot be
HASH_INVALID_SIGNATURE. If the single list pointer is non-NULL, then
the records array must be full.
If the HASHBUCKET is compact, then we can abort a search for a key as
soon as we see HASH_INVALID_SIGNATURE. This invariant speeds up Find and
Insert at the cost of making Delete and Flush a little more
complex. Since we expect to do a lot more Finds than Deletes or Inserts,
this is an acceptable tradeoff.
Storing hash signatures means that we have a very fast test that
eliminates almost all false positives. We very seldom find two keys
that have matching hash signatures, but different strings.
Arguments:
pBucket - The hash bucket
--***************************************************************************/
BOOLEAN
UlpHashBucketIsCompact(
IN const PHASHBUCKET pBucket)
{
PUL_URI_CACHE_ENTRY pUriCacheEntry;
PUL_URI_CACHE_ENTRY pPrevUriCacheEntry = NULL;
PHASHURIKEY pHashUriKey = UlpHashTableUriKeyFromBucket(pBucket);
ULONG i, j, Entries = 0;
// First, validate the records array
for (i = 0; i < g_UlNumOfHashUriKeys; i++)
{
ULONG Hash = pHashUriKey[i].Hash;
if (HASH_INVALID_SIGNATURE == Hash)
{
// There are no more valid entries in the records array
// and no singly linked list
ASSERT(NULL == pBucket->pUriCacheEntry);
pPrevUriCacheEntry = NULL;
for (j = i; j < g_UlNumOfHashUriKeys; j++)
{
ASSERT(NULL == pHashUriKey[j].pUriCacheEntry);
ASSERT(HASH_INVALID_SIGNATURE == pHashUriKey[j].Hash);
}
}
else
{
// non-empty slot
++Entries;
pUriCacheEntry = pHashUriKey[i].pUriCacheEntry;
ASSERT(IS_VALID_URI_CACHE_ENTRY(pUriCacheEntry));
ASSERT(pUriCacheEntry->Cached);
ASSERT(Hash == pUriCacheEntry->UriKey.Hash);
ASSERT(Hash == HashRandomizeBits(
HashStringNoCaseW(
pUriCacheEntry->UriKey.pUri,
0
)));
ASSERT(pPrevUriCacheEntry != pUriCacheEntry);
pPrevUriCacheEntry = pUriCacheEntry;
}
}
// Next, validate the singly linked list
for (pUriCacheEntry = pBucket->pUriCacheEntry;
NULL != pUriCacheEntry;
pUriCacheEntry
= (PUL_URI_CACHE_ENTRY) pUriCacheEntry->BucketEntry.Next)
{
++Entries;
ASSERT(IS_VALID_URI_CACHE_ENTRY(pUriCacheEntry));
ASSERT(pUriCacheEntry->Cached);
ASSERT(pUriCacheEntry->UriKey.Hash
== HashRandomizeBits(
HashStringNoCaseW(
pUriCacheEntry->UriKey.pUri,
0
)));
ASSERT(pPrevUriCacheEntry != pUriCacheEntry);
pPrevUriCacheEntry = pUriCacheEntry;
}
ASSERT(Entries == pBucket->Entries);
return TRUE;
} // UlpHashBucketIsCompact
/***************************************************************************++
Routine Description:
This routine initialize the hash table
Arguments:
pHashTable - The hash table
PoolType - Specifies the type of pool memory to allocate
HashTableBits - The number of buckets is (1 << HashTableBits)
Returns:
NTSTATUS - STATUS_SUCCESS or STATUS_NO_MEMORY
--***************************************************************************/
NTSTATUS
UlInitializeHashTable(
IN OUT PHASHTABLE pHashTable,
IN POOL_TYPE PoolType,
IN LONG HashTableBits
)
{
ULONG i;
ULONG_PTR CacheLineMask, CacheLineSize = g_UlCacheLineSize;
//
// First, get the hash table size from the registry.
// If not defined in the registry, determine the hash table
// size by the system memory size
//
UlpGetHashTableSize(HashTableBits);
#ifdef HASH_TEST
CacheLineSize = 64;
g_UlHashIndexShift = 6;
#endif
CacheLineMask = CacheLineSize - 1;
ASSERT((CacheLineSize & CacheLineMask) == 0); // power of 2
ASSERT(CacheLineSize == (1U << g_UlHashIndexShift));
pHashTable->Signature = UL_HASH_TABLE_POOL_TAG;
pHashTable->PoolType = PoolType;
pHashTable->NumberOfBytes = g_UlHashTableSize * CacheLineSize;
ASSERT(CacheLineSize > sizeof(HASHBUCKET));
// number of keys stored in the initial clump
g_UlNumOfHashUriKeys = (((ULONG) CacheLineSize - sizeof (HASHBUCKET))
/ sizeof(HASHURIKEY));
pHashTable->pBuckets = NULL;
#ifdef HASH_TEST
g_UlNumOfHashUriKeys = 3;
#endif
ASSERT((sizeof(HASHBUCKET) + g_UlNumOfHashUriKeys * sizeof(HASHURIKEY))
<= (1U << g_UlHashIndexShift));
// Allocate the memory
pHashTable->pAllocMem
= (PHASHBUCKET) UL_ALLOCATE_POOL(
PoolType,
pHashTable->NumberOfBytes + CacheLineMask,
UL_HASH_TABLE_POOL_TAG
);
if (NULL == pHashTable->pAllocMem)
{
pHashTable->Signature = MAKE_FREE_TAG(UL_HASH_TABLE_POOL_TAG);
return STATUS_NO_MEMORY;
}
// Align the memory the cache line size boundary
pHashTable->pBuckets
= (PHASHBUCKET)((((ULONG_PTR)(pHashTable->pAllocMem)) + CacheLineMask)
& ~CacheLineMask);
// Initialize each bucket and padding array elements
for (i=0; i < g_UlHashTableSize; i++)
{
PHASHBUCKET pBucket = UlpHashTableIndexedBucket(pHashTable, i);
PHASHURIKEY pHashUriKey;
ULONG j;
UlInitializeRWSpinLock(&pBucket->RWSpinLock);
pBucket->pUriCacheEntry = NULL;
pBucket->Entries = 0;
pHashUriKey = UlpHashTableUriKeyFromBucket(pBucket);
for (j = 0; j < g_UlNumOfHashUriKeys ;j++)
{
pHashUriKey[j].Hash = HASH_INVALID_SIGNATURE;
pHashUriKey[j].pUriCacheEntry = NULL;
}
ASSERT(UlpHashBucketIsCompact(pBucket));
}
return STATUS_SUCCESS;
} // UlInitializeHashTable
/***************************************************************************++
Routine Description:
This routine terminates the hash table
(flush all entries and free the table).
Arguments:
pHashTable - The hash table
--***************************************************************************/
VOID
UlTerminateHashTable(
IN PHASHTABLE pHashTable
)
{
if ( pHashTable->pAllocMem != NULL )
{
ASSERT(IS_VALID_HASHTABLE(pHashTable));
// Clear the hash table (delete all entries)
UlClearHashTable(pHashTable);
// Free the hash table buckets
UL_FREE_POOL(pHashTable->pAllocMem, UL_HASH_TABLE_POOL_TAG);
pHashTable->Signature = MAKE_FREE_TAG(UL_HASH_TABLE_POOL_TAG);
pHashTable->pAllocMem = pHashTable->pBuckets = NULL;
}
} // UlTerminateHashTable
/***************************************************************************++
Routine Description:
This routine does a cache lookup on a hash table
to see if there is a valid entry corresponding to the request key.
Increment the reference counter of the entry by 1 inside the lock
protection to ensure this entry will be still alive after returning
this entry back.
Arguments:
pHashTable - The hash table
pUriKey - the search key
Returns:
PUL_URI_CACHE_ENTRY - pointer to entry or NULL
--***************************************************************************/
PUL_URI_CACHE_ENTRY
UlGetFromHashTable(
IN PHASHTABLE pHashTable,
IN PURI_KEY pUriKey
)
{
PUL_URI_CACHE_ENTRY pUriCacheEntry;
PHASHBUCKET pBucket;
PHASHURIKEY pHashUriKey;
ULONG i;
HASH_PAGED_CODE(pHashTable);
ASSERT(IS_VALID_HASHTABLE(pHashTable));
pBucket = UlpHashTableBucketFromUriKey(pHashTable, pUriKey);
UlAcquireRWSpinLockShared(&pBucket->RWSpinLock);
ASSERT(UlpHashBucketIsCompact(pBucket));
pHashUriKey = UlpHashTableUriKeyFromBucket(pBucket);
// Scan the records array first
for (i = 0; i < g_UlNumOfHashUriKeys; i++)
{
ULONG Hash = pHashUriKey[i].Hash;
if (HASH_INVALID_SIGNATURE == Hash)
{
// There are no more valid entries in the bucket
ASSERT(NULL == pBucket->pUriCacheEntry);
ASSERT(NULL == pHashUriKey[i].pUriCacheEntry);
pUriCacheEntry = NULL;
goto unlock;
}
if (Hash == pUriKey->Hash)
{
pUriCacheEntry = pHashUriKey[i].pUriCacheEntry;
ASSERT(NULL != pUriCacheEntry);
if (UlpEqualUriKeys(&pUriCacheEntry->UriKey, pUriKey))
{
goto addref;
}
}
}
ASSERT(i == g_UlNumOfHashUriKeys);
// Jump to the single list for searching
for (pUriCacheEntry = pBucket->pUriCacheEntry;
NULL != pUriCacheEntry;
pUriCacheEntry
= (PUL_URI_CACHE_ENTRY) pUriCacheEntry->BucketEntry.Next)
{
if (pUriCacheEntry->UriKey.Hash == pUriKey->Hash
&& UlpEqualUriKeys(&pUriCacheEntry->UriKey, pUriKey))
{
goto addref;
}
}
// Not found
ASSERT(NULL == pUriCacheEntry);
goto unlock;
addref:
ASSERT(NULL != pUriCacheEntry);
REFERENCE_URI_CACHE_ENTRY(pUriCacheEntry, CHECKOUT);
unlock:
ASSERT(UlpHashBucketIsCompact(pBucket));
UlReleaseRWSpinLockShared(&pBucket->RWSpinLock);
return pUriCacheEntry;
} // UlGetFromHashTable
/***************************************************************************++
Routine Description:
This routine does a cache lookup on a hash table
to see if there is a valid entry corresponding to the request URI,
if found, delete this entry. However, increment the reference counter
of the entry by 1 insde the lock protection to ensure this entry will be
still alive after returning this entry back.
Arguments:
pHashTable - The hash table
pUriKey - the search key
Returns:
PUL_URI_CACHE_ENTRY - pointer to entry removed from table or NULL
--***************************************************************************/
PUL_URI_CACHE_ENTRY
UlDeleteFromHashTable(
IN PHASHTABLE pHashTable,
IN PURI_KEY pUriKey
)
{
PUL_URI_CACHE_ENTRY pUriCacheEntry;
PUL_URI_CACHE_ENTRY PrevUriCacheEntry;
PHASHBUCKET pBucket;
PHASHURIKEY pHashUriKey;
ULONG i;
HASH_PAGED_CODE(pHashTable);
ASSERT(IS_VALID_HASHTABLE(pHashTable));
pBucket = UlpHashTableBucketFromUriKey(pHashTable, pUriKey);
UlAcquireRWSpinLockExclusive(&pBucket->RWSpinLock);
ASSERT(UlpHashBucketIsCompact(pBucket));
pHashUriKey = UlpHashTableUriKeyFromBucket(pBucket);
// Scan the records array first
for (i = 0; i < g_UlNumOfHashUriKeys; i++)
{
ULONG Hash = pHashUriKey[i].Hash;
if (HASH_INVALID_SIGNATURE == Hash)
{
ASSERT(NULL == pBucket->pUriCacheEntry);
ASSERT(NULL == pHashUriKey[i].pUriCacheEntry);
pUriCacheEntry = NULL;
goto unlock;
}
if (Hash == pUriKey->Hash)
{
pUriCacheEntry = pHashUriKey[i].pUriCacheEntry;
ASSERT(NULL != pUriCacheEntry);
if (UlpEqualUriKeys(&pUriCacheEntry->UriKey, pUriKey))
{
--pBucket->Entries;
if (pBucket->pUriCacheEntry)
{
// If there exists an entry in the single list,
// move it to the array
pHashUriKey[i].Hash
= pBucket->pUriCacheEntry->UriKey.Hash;
pHashUriKey[i].pUriCacheEntry = pBucket->pUriCacheEntry;
pBucket->pUriCacheEntry
= (PUL_URI_CACHE_ENTRY)
pBucket->pUriCacheEntry->BucketEntry.Next;
}
else
{
// if this is not the last entry in the records array,
// move the last entry to this slot
ULONG j;
for (j = g_UlNumOfHashUriKeys; --j >= i; )
{
if (NULL != pHashUriKey[j].pUriCacheEntry)
{
ASSERT(HASH_INVALID_SIGNATURE
!= pHashUriKey[j].Hash);
ASSERT(j >= i);
pHashUriKey[i].Hash = pHashUriKey[j].Hash;
pHashUriKey[i].pUriCacheEntry
= pHashUriKey[j].pUriCacheEntry;
// Zap the last entry. Correct even if j == i
pHashUriKey[j].Hash = HASH_INVALID_SIGNATURE;
pHashUriKey[j].pUriCacheEntry = NULL;
goto unlock;
}
else
{
ASSERT(HASH_INVALID_SIGNATURE
== pHashUriKey[j].Hash);
}
}
// We can't get here, since pHashUriKey[i] should
// have terminated the loop even if there wasn't
// any non-empty slot following it.
ASSERT(! "Overshot the deleted entry");
}
goto unlock;
}
}
}
ASSERT(i == g_UlNumOfHashUriKeys);
// Jump to the single list for searching
pUriCacheEntry = pBucket->pUriCacheEntry;
PrevUriCacheEntry = NULL;
while (NULL != pUriCacheEntry)
{
if (pUriCacheEntry->UriKey.Hash == pUriKey->Hash
&& UlpEqualUriKeys(&pUriCacheEntry->UriKey, pUriKey))
{
if (PrevUriCacheEntry == NULL)
{
// Delete First Entry
pBucket->pUriCacheEntry
= (PUL_URI_CACHE_ENTRY) pUriCacheEntry->BucketEntry.Next;
}
else
{
PrevUriCacheEntry->BucketEntry.Next
= pUriCacheEntry->BucketEntry.Next;
}
--pBucket->Entries;
goto unlock;
}
PrevUriCacheEntry = pUriCacheEntry;
pUriCacheEntry
= (PUL_URI_CACHE_ENTRY) pUriCacheEntry->BucketEntry.Next;
}
// Not found
ASSERT(NULL == pUriCacheEntry);
unlock:
ASSERT((LONG) pBucket->Entries >= 0);
ASSERT(UlpHashBucketIsCompact(pBucket));
UlReleaseRWSpinLockExclusive(&pBucket->RWSpinLock);
return pUriCacheEntry;
} // UlDeleteFromHashTable
/***************************************************************************++
Routine Description:
This routine does a cache lookup on a hash table
to see if a given entry exists, if not found, add this entry to the
hash table. Increment the reference counter of the entry by 1 insde the
lock protection.
Arguments:
pHashTable - The hash table
pUriCacheEntry - the given entry
Returns
ULC_RETCODE - ULC_SUCCESS or ULC_KEY_EXISTS
--***************************************************************************/
ULC_RETCODE
UlAddToHashTable(
IN PHASHTABLE pHashTable,
IN PUL_URI_CACHE_ENTRY pUriCacheEntry
)
{
PUL_URI_CACHE_ENTRY pTmpUriCacheEntry;
PURI_KEY pUriKey = &(pUriCacheEntry->UriKey);
PHASHBUCKET pBucket;
PHASHURIKEY pHashUriKey;
LONG EmptySlot = INVALID_SLOT_INDEX;
ULONG i;
ULC_RETCODE rc;
HASH_PAGED_CODE(pHashTable);
ASSERT(IS_VALID_HASHTABLE(pHashTable));
ASSERT(IS_VALID_URI_CACHE_ENTRY(pUriCacheEntry));
ASSERT(pUriCacheEntry->Cached);
pBucket = UlpHashTableBucketFromUriKey(pHashTable, pUriKey);
UlAcquireRWSpinLockExclusive(&pBucket->RWSpinLock);
ASSERT(UlpHashBucketIsCompact(pBucket));
pHashUriKey = UlpHashTableUriKeyFromBucket(pBucket);
// Scan the records array first
for (i = 0; i < g_UlNumOfHashUriKeys; i++)
{
ULONG Hash = pHashUriKey[i].Hash;
if (HASH_INVALID_SIGNATURE == Hash)
{
ASSERT(NULL == pBucket->pUriCacheEntry);
ASSERT(NULL == pHashUriKey[i].pUriCacheEntry);
EmptySlot = (LONG) i;
goto insert;
}
if (Hash == pUriKey->Hash)
{
pTmpUriCacheEntry = pHashUriKey[i].pUriCacheEntry;
ASSERT(NULL != pTmpUriCacheEntry);
if (UlpEqualUriKeys(&pTmpUriCacheEntry->UriKey, pUriKey))
{
// duplicate key exists
pUriCacheEntry->Cached = FALSE;
rc = ULC_KEY_EXISTS;
goto unlock;
}
}
}
ASSERT(i == g_UlNumOfHashUriKeys);
ASSERT(EmptySlot == INVALID_SLOT_INDEX);
// Jump to the single list for searching
for (pTmpUriCacheEntry = pBucket->pUriCacheEntry;
NULL != pTmpUriCacheEntry;
pTmpUriCacheEntry
= (PUL_URI_CACHE_ENTRY) pTmpUriCacheEntry->BucketEntry.Next)
{
if (pTmpUriCacheEntry->UriKey.Hash == pUriKey->Hash
&& UlpEqualUriKeys(&pTmpUriCacheEntry->UriKey, pUriKey))
{
// duplicate key exists
pUriCacheEntry->Cached = FALSE;
rc = ULC_KEY_EXISTS;
goto unlock;
}
}
insert:
//
// Not found, no duplicate key in hash table
//
if (EmptySlot != INVALID_SLOT_INDEX)
{
ASSERT(0 <= EmptySlot && EmptySlot < (LONG) g_UlNumOfHashUriKeys);
// First, try to add this entry to the array if there is an empty slot.
pHashUriKey[EmptySlot].Hash = pUriKey->Hash;
pHashUriKey[EmptySlot].pUriCacheEntry = pUriCacheEntry;
}
else
{
// Otherwise, add this entry to the head of the single list
pUriCacheEntry->BucketEntry.Next
= (PSINGLE_LIST_ENTRY) pBucket->pUriCacheEntry;
pBucket->pUriCacheEntry = pUriCacheEntry;
}
REFERENCE_URI_CACHE_ENTRY(pUriCacheEntry, ADD);
ASSERT((LONG) pBucket->Entries >= 0);
++pBucket->Entries;
rc = ULC_SUCCESS;
unlock:
ASSERT(UlpHashBucketIsCompact(pBucket));
UlReleaseRWSpinLockExclusive(&pBucket->RWSpinLock);
return rc;
} // UlAddToHashTable
/***************************************************************************++
Routine Description:
Removes entries based on a caller-specified filter. The caller
provides a predicate function which takes a cache entry as a
parameter. The function will be called for each item in the cache.
If the function returns ULC_DELETE, the item will be removed.
Otherwise the item will remain in the cache.
All removals are done on a hash table bucket.
Walk through all the entries under this bucket.
Assume bucket exclusive lock is held.
Arguments:
pBucket - The hash table bucket
pFilterRoutine - A pointer to the filter function
pContext - A parameter to the filter function
pDeletedCount - A pointer to the number of deleted entries on this bucket
bStop - A pointer to a boolean variable returned to caller
(TRUE if the filter function asks for action stop)
--***************************************************************************/
BOOLEAN
UlpFilterFlushHashBucket(
IN PHASHBUCKET pBucket,
IN PUL_URI_FILTER pFilterRoutine,
IN PVOID pContext,
OUT PULONG pDeletedCount
)
{
PUL_URI_CACHE_ENTRY pUriCacheEntry;
PUL_URI_CACHE_ENTRY pPrevUriCacheEntry;
PUL_URI_CACHE_ENTRY pTmpUriCacheEntry;
UL_CACHE_PREDICATE result;
LONG ReferenceCount;
PHASHURIKEY pHashUriKey;
ULONG i;
LONG LastSlot;
BOOLEAN bStop = FALSE;
// Check if bucket exclusive lock is held
ASSERT( UlIsLockedExclusive(&pBucket->RWSpinLock) );
ASSERT(UlpHashBucketIsCompact(pBucket));
// Scan the single list first
pUriCacheEntry = pBucket->pUriCacheEntry;
pPrevUriCacheEntry = NULL;
while (NULL != pUriCacheEntry)
{
BOOLEAN bDelete = FALSE;
result = (*pFilterRoutine)(pUriCacheEntry, pContext);
switch (result)
{
case ULC_ABORT:
bStop = TRUE;
goto end;
case ULC_NO_ACTION:
// nothing to do
break;
case ULC_PERFORM:
case ULC_PERFORM_STOP:
case ULC_DELETE:
case ULC_DELETE_STOP:
{
// Delete this entry
bDelete = TRUE;
ASSERT(pBucket->Entries > 0);
--pBucket->Entries;
pTmpUriCacheEntry = pUriCacheEntry;
if (NULL == pPrevUriCacheEntry)
{
// Delete First Entry
pBucket->pUriCacheEntry
= (PUL_URI_CACHE_ENTRY)
pUriCacheEntry->BucketEntry.Next;
pUriCacheEntry = pBucket->pUriCacheEntry;
}
else
{
pPrevUriCacheEntry->BucketEntry.Next
= pUriCacheEntry->BucketEntry.Next;
pUriCacheEntry
= (PUL_URI_CACHE_ENTRY)
pPrevUriCacheEntry->BucketEntry.Next;
}
ASSERT(UlpHashBucketIsCompact(pBucket));
DEREFERENCE_URI_CACHE_ENTRY(pTmpUriCacheEntry, FILTER);
++(*pDeletedCount);
if (result == ULC_PERFORM_STOP || result == ULC_DELETE_STOP)
{
bStop = TRUE;
goto end;
}
break;
}
default:
break;
}
if (!bDelete)
{
pPrevUriCacheEntry = pUriCacheEntry;
pUriCacheEntry
= (PUL_URI_CACHE_ENTRY) pUriCacheEntry->BucketEntry.Next;
}
}
pHashUriKey = UlpHashTableUriKeyFromBucket(pBucket);
//
// Now, scan the records array.
//
// Because we keep the records array compact, we need to keep
// track of the last valid slot, so that we can move its contents
// to the slot that's being deleted.
//
LastSlot = INVALID_SLOT_INDEX;
if (NULL == pBucket->pUriCacheEntry)
{
for (i = g_UlNumOfHashUriKeys; i-- > 0; )
{
if (NULL != pHashUriKey[i].pUriCacheEntry)
{
ASSERT(HASH_INVALID_SIGNATURE != pHashUriKey[i].Hash);
LastSlot = (LONG) i;
break;
}
else
{
ASSERT(HASH_INVALID_SIGNATURE == pHashUriKey[i].Hash);
}
}
// Is records array completely empty?
if (LastSlot == INVALID_SLOT_INDEX)
goto end;
}
else
{
// final slot cannot be empty
ASSERT(HASH_INVALID_SIGNATURE
!= pHashUriKey[g_UlNumOfHashUriKeys-1].Hash);
}
// Walk through the records array
for (i = 0; i < g_UlNumOfHashUriKeys; i++)
{
pUriCacheEntry = pHashUriKey[i].pUriCacheEntry;
if (NULL == pUriCacheEntry)
{
ASSERT(HASH_INVALID_SIGNATURE == pHashUriKey[i].Hash);
goto end;
}
else
{
ASSERT(HASH_INVALID_SIGNATURE != pHashUriKey[i].Hash);
}
result = (*pFilterRoutine)(pUriCacheEntry, pContext);
switch (result)
{
case ULC_ABORT:
bStop = TRUE;
goto end;
case ULC_NO_ACTION:
// nothing to do
break;
case ULC_PERFORM:
case ULC_PERFORM_STOP:
case ULC_DELETE:
case ULC_DELETE_STOP:
{
// Delete this entry
ASSERT(pBucket->Entries > 0);
--pBucket->Entries;
if (NULL != pBucket->pUriCacheEntry)
{
// If there exists an entry in the single list,
// move it to the array
ASSERT(LastSlot == INVALID_SLOT_INDEX);
pHashUriKey[i].Hash
= pBucket->pUriCacheEntry->UriKey.Hash;
pHashUriKey[i].pUriCacheEntry = pBucket->pUriCacheEntry;
pBucket->pUriCacheEntry
= (PUL_URI_CACHE_ENTRY)
pBucket->pUriCacheEntry->BucketEntry.Next;
if (NULL == pBucket->pUriCacheEntry)
{
LastSlot = g_UlNumOfHashUriKeys - 1;
ASSERT(HASH_INVALID_SIGNATURE
!= pHashUriKey[LastSlot].Hash);
}
}
else
{
// Move the entry in the last slot to this position,
// zap the last slot, and move LastSlot backwards
if (LastSlot != INVALID_SLOT_INDEX
&& (LONG) i < LastSlot)
{
ASSERT(HASH_INVALID_SIGNATURE
!= pHashUriKey[LastSlot].Hash);
pHashUriKey[i].Hash = pHashUriKey[LastSlot].Hash;
pHashUriKey[i].pUriCacheEntry
= pHashUriKey[LastSlot].pUriCacheEntry;
pHashUriKey[LastSlot].Hash = HASH_INVALID_SIGNATURE;
pHashUriKey[LastSlot].pUriCacheEntry = NULL;
if (--LastSlot == i)
LastSlot = INVALID_SLOT_INDEX;
else
ASSERT(HASH_INVALID_SIGNATURE
!= pHashUriKey[LastSlot].Hash);
}
else
{
// Just reset this array element
pHashUriKey[i].Hash = HASH_INVALID_SIGNATURE;
pHashUriKey[i].pUriCacheEntry = NULL;
LastSlot = INVALID_SLOT_INDEX;
}
}
ASSERT(UlpHashBucketIsCompact(pBucket));
DEREFERENCE_URI_CACHE_ENTRY(pUriCacheEntry, FILTER);
++(*pDeletedCount);
if (result == ULC_PERFORM_STOP || result == ULC_DELETE_STOP)
{
bStop = TRUE;
goto end;
}
break;
}
default:
break;
}
}
end:
ASSERT(UlpHashBucketIsCompact(pBucket));
return bStop;
} // UlpFilterFlushHashBucket
/***************************************************************************++
Routine Description:
Removes entries based on a caller-specified filter. The caller
provides a predicate function which takes a cache entry as a
parameter. The function will be called with each item in the cache.
If the function returns ULC_DELETE, the item will be removed.
Otherwise the item will remain in the cache.
Arguments:
pHashTable - The hash table
pFilterRoutine - A pointer to the filter function
pContext - A parameter to the filter function
pDeletedCount - A pointer to the number of deleted entries
Returns:
ULONG - Number of entries flushed from the table
--***************************************************************************/
ULONG
UlFilterFlushHashTable(
IN PHASHTABLE pHashTable,
IN PUL_URI_FILTER pFilterRoutine,
IN PVOID pContext
)
{
ULONG i;
BOOLEAN bStop = FALSE;
ULONG DeletedCount = 0;
HASH_PAGED_CODE(pHashTable);
ASSERT(IS_VALID_HASHTABLE(pHashTable));
//
// Scan and delete (if matching the filter) each bucket
// of the cache table.
//
for (i = 0; !bStop && i < g_UlHashTableSize; i++)
{
PHASHBUCKET pBucket = UlpHashTableIndexedBucket(pHashTable, i);
ULONG DeletedInBucket = 0;
UlAcquireRWSpinLockExclusive(&pBucket->RWSpinLock);
bStop = UlpFilterFlushHashBucket(
pBucket,
pFilterRoutine,
pContext,
&DeletedInBucket
);
UlReleaseRWSpinLockExclusive(&pBucket->RWSpinLock);
DeletedCount += DeletedInBucket;
}
return DeletedCount;
} // UlFilterFlushHashTable
/***************************************************************************++
Routine Description:
Removes all entries on a bucket.
Assume bucket exclusive lock is held.
Arguments:
pBucket - The hash table bucket
--***************************************************************************/
VOID
UlpClearHashBucket(
IN PHASHBUCKET pBucket
)
{
PUL_URI_CACHE_ENTRY pUriCacheEntry;
PUL_URI_CACHE_ENTRY pTmpUriCacheEntry;
PHASHURIKEY pHashUriKey;
ULONG i;
LONG ReferenceCount;
// Check if bucket exclusive lock is held
ASSERT( UlIsLockedExclusive(&pBucket->RWSpinLock) );
ASSERT(UlpHashBucketIsCompact(pBucket));
// Scan the single list first
pUriCacheEntry = pBucket->pUriCacheEntry;
while (NULL != pUriCacheEntry)
{
pTmpUriCacheEntry = pUriCacheEntry;
pBucket->pUriCacheEntry
= (PUL_URI_CACHE_ENTRY) pUriCacheEntry->BucketEntry.Next;
pUriCacheEntry = pBucket->pUriCacheEntry;
DEREFERENCE_URI_CACHE_ENTRY(pTmpUriCacheEntry, CLEAR);
}
ASSERT(NULL == pBucket->pUriCacheEntry);
pHashUriKey = UlpHashTableUriKeyFromBucket(pBucket);
// Scan the records array
for (i = 0; i < g_UlNumOfHashUriKeys; i++)
{
pUriCacheEntry = pHashUriKey[i].pUriCacheEntry;
if (NULL == pUriCacheEntry)
{
ASSERT(HASH_INVALID_SIGNATURE == pHashUriKey[i].Hash);
break;
}
else
{
ASSERT(HASH_INVALID_SIGNATURE != pHashUriKey[i].Hash);
}
DEREFERENCE_URI_CACHE_ENTRY(pUriCacheEntry, CLEAR);
}
pBucket->Entries = 0;
ASSERT(UlpHashBucketIsCompact(pBucket));
} // UlpClearHashBucket
/***************************************************************************++
Routine Description:
Removes all entries of the hash table.
Arguments:
pHashTable - The hash table
--***************************************************************************/
VOID
UlClearHashTable(
IN PHASHTABLE pHashTable
)
{
ULONG i;
PHASHBUCKET pBucket;
HASH_PAGED_CODE(pHashTable);
ASSERT(IS_VALID_HASHTABLE(pHashTable));
for (i = 0; i < g_UlHashTableSize ;i++)
{
pBucket = UlpHashTableIndexedBucket(pHashTable, i);
UlAcquireRWSpinLockExclusive(&pBucket->RWSpinLock);
UlpClearHashBucket(pBucket);
UlReleaseRWSpinLockExclusive(&pBucket->RWSpinLock);
}
} // UlClearHashTable