windows-nt/Source/XPSP1/NT/inetsrv/iis/iisrearc/ul/drv/cache.cxx
2020-09-26 16:20:57 +08:00

2056 lines
52 KiB
C++

/*++
Copyright (c) 1998-2001 Microsoft Corporation
Module Name:
cache.cxx
Abstract:
Contains the HTTP response cache logic.
Author:
Michael Courage (mcourage) 17-May-1999
Revision History:
--*/
#include "precomp.h"
#include "cachep.h"
PTRACE_LOG g_UriTraceLog;
BOOLEAN g_InitUriCacheCalled;
//
// Global hash table
//
HASHTABLE g_UriCacheTable;
LIST_ENTRY g_ZombieListHead;
UL_URI_CACHE_CONFIG g_UriCacheConfig;
UL_URI_CACHE_STATS g_UriCacheStats;
UL_SPIN_LOCK g_UriCacheSpinLock;
//
// Scavenger stuff.
//
UL_SPIN_LOCK g_UriScavengerSpinLock;
BOOLEAN g_UriScavengerInitialized;
KDPC g_UriScavengerDpc;
KTIMER g_UriScavengerTimer;
KEVENT g_UriScavengerTerminationEvent;
UL_WORK_ITEM g_UriScavengerWorkItem;
LONG g_UriScavengerRunning;
#ifdef ALLOC_PRAGMA
#pragma alloc_text( INIT, UlInitializeUriCache )
#pragma alloc_text( PAGE, UlTerminateUriCache )
#pragma alloc_text( INIT, UlpInitializeScavenger )
#pragma alloc_text( PAGE, UlCheckCachePreconditions )
#pragma alloc_text( PAGE, UlCheckCacheResponseConditions )
#pragma alloc_text( PAGE, UlCheckoutUriCacheEntry )
#pragma alloc_text( PAGE, UlCheckinUriCacheEntry )
#pragma alloc_text( PAGE, UlFlushCache )
#pragma alloc_text( PAGE, UlpFlushFilterAll )
#pragma alloc_text( PAGE, UlFlushCacheByProcess )
#pragma alloc_text( PAGE, UlpFlushFilterProcess )
#pragma alloc_text( PAGE, UlFlushCacheByUri )
#pragma alloc_text( PAGE, UlpFlushUri )
#pragma alloc_text( PAGE, UlAddCacheEntry )
#pragma alloc_text( PAGE, UlpFilteredFlushUriCache )
#pragma alloc_text( PAGE, UlpAddZombie )
#pragma alloc_text( PAGE, UlpClearZombieList )
#pragma alloc_text( PAGE, UlpDestroyUriCacheEntry )
#pragma alloc_text( PAGE, UlpFlushFilterScavenger )
#pragma alloc_text( PAGE, UlpQueryTranslateHeader )
#endif // ALLOC_PRAGMA
#if 0
NOT PAGEABLE -- UlpCheckTableSpace
NOT PAGEABLE -- UlpCheckSpaceAndAddEntryStats
NOT PAGEABLE -- UlpRemoveEntryStats
NOT PAGEABLE -- UlpTerminateScavenger
NOT PAGEABLE -- UlpScavengerDpcRoutine
NOT PAGEABLE -- UlpSetScavengerTimer
NOT PAGEABLE -- UlpScavenger
#endif
/***************************************************************************++
Routine Description:
Performs global initialization of the URI cache.
Return Value:
NTSTATUS - Completion status.
--***************************************************************************/
NTSTATUS
UlInitializeUriCache(
PUL_CONFIG pConfig
)
{
NTSTATUS Status = STATUS_SUCCESS;
//
// Sanity check.
//
PAGED_CODE();
ASSERT( !g_InitUriCacheCalled );
UlTrace(URI_CACHE, ("Http!UlInitializeUriCache\n"));
if ( !g_InitUriCacheCalled )
{
PUL_URI_CACHE_CONFIG pUriConfig = &pConfig->UriConfig;
g_UriCacheConfig.EnableCache = pUriConfig->EnableCache;
g_UriCacheConfig.MaxCacheUriCount = pUriConfig->MaxCacheUriCount;
g_UriCacheConfig.MaxCacheMegabyteCount =
min((ULONG)pConfig->LargeMemMegabytes, pUriConfig->MaxCacheMegabyteCount);
//
// Don't want to scavenge more than once every ten seconds.
// In particular, do not want to scavenge every 0 seconds, as the
// machine will become completely unresponsive.
//
g_UriCacheConfig.ScavengerPeriod =
max(pUriConfig->ScavengerPeriod, 10);
g_UriCacheConfig.MaxUriBytes = pUriConfig->MaxUriBytes;
g_UriCacheConfig.HashTableBits = pUriConfig->HashTableBits;
RtlZeroMemory(&g_UriCacheStats, sizeof(g_UriCacheStats));
InitializeListHead(&g_ZombieListHead);
UlInitializeSpinLock( &g_UriCacheSpinLock, "g_UriCacheSpinLock" );
if (g_UriCacheConfig.EnableCache)
{
Status = UlInitializeResource(
&g_pUlNonpagedData->UriZombieResource,
"UriZombieResource",
0,
UL_ZOMBIE_RESOURCE_TAG
);
if (NT_SUCCESS(Status))
{
Status = UlInitializeHashTable(
&g_UriCacheTable,
PagedPool,
g_UriCacheConfig.HashTableBits
);
if (NT_SUCCESS(Status))
{
ASSERT(IS_VALID_HASHTABLE(&g_UriCacheTable));
CREATE_REF_TRACE_LOG( g_UriTraceLog,
2048 - REF_TRACE_OVERHEAD, 0 );
UlpInitializeScavenger();
g_InitUriCacheCalled = TRUE;
}
}
else
{
UlDeleteResource(&g_pUlNonpagedData->UriZombieResource);
}
}
else
{
UlTrace(URI_CACHE, ("URI Cache disabled.\n"));
g_InitUriCacheCalled = TRUE;
}
}
else
{
UlTrace(URI_CACHE, ("URI CACHE INITIALIZED TWICE!\n"));
}
return Status;
} // UlInitializeUriCache
/***************************************************************************++
Routine Description:
Performs global termination of the URI cache.
--***************************************************************************/
VOID
UlTerminateUriCache(
VOID
)
{
NTSTATUS Status;
//
// Sanity check.
//
PAGED_CODE();
UlTrace(URI_CACHE, ("Http!UlTerminateUriCache\n"));
if (g_InitUriCacheCalled && g_UriCacheConfig.EnableCache)
{
// Must terminate the scavenger before destroying the hash table
UlpTerminateScavenger();
UlTerminateHashTable(&g_UriCacheTable);
Status = UlDeleteResource(&g_pUlNonpagedData->UriZombieResource);
ASSERT(NT_SUCCESS(Status));
DESTROY_REF_TRACE_LOG( g_UriTraceLog );
g_UriTraceLog = NULL;
}
g_InitUriCacheCalled = FALSE;
} // UlTerminateUriCache
/***************************************************************************++
Routine Description:
This routine checks a request (and its connection) to see if it's
ok to serve this request from the cache. Basically we only accept
simple GET requests with no conditional headers.
Arguments:
pHttpConn - The connection to be checked
Return Value:
BOOLEAN - True if it's ok to serve from cache
--***************************************************************************/
BOOLEAN
UlCheckCachePreconditions(
PUL_INTERNAL_REQUEST pRequest,
PUL_HTTP_CONNECTION pHttpConn
)
{
URI_PRECONDITION Precondition = URI_PRE_OK;
//
// Sanity check
//
PAGED_CODE();
ASSERT( UL_IS_VALID_HTTP_CONNECTION(pHttpConn) );
ASSERT( UL_IS_VALID_INTERNAL_REQUEST(pRequest) );
if (!g_UriCacheConfig.EnableCache)
{
Precondition = URI_PRE_DISABLED;
}
else if (pRequest->ParseState != ParseDoneState)
{
Precondition = URI_PRE_ENTITY_BODY;
}
else if (pRequest->Verb != HttpVerbGET)
{
Precondition = URI_PRE_VERB;
}
else if (HTTP_NOT_EQUAL_VERSION(pRequest->Version, 1, 1)
&& HTTP_NOT_EQUAL_VERSION(pRequest->Version, 1, 0))
{
Precondition = URI_PRE_PROTOCOL;
}
// check for Translate: f (DAV)
else if ( UlpQueryTranslateHeader(pRequest) )
{
Precondition = URI_PRE_TRANSLATE;
}
// check for Authorization header
else if (pRequest->HeaderValid[HttpHeaderAuthorization])
{
Precondition = URI_PRE_AUTHORIZATION;
}
//
// check for some of the If-* headers
// NOTE: See UlpCheckCacheControlHeaders for handling of other If-* headers
//
else if (pRequest->HeaderValid[HttpHeaderIfRange])
{
Precondition = URI_PRE_CONDITIONAL;
}
// CODEWORK: check for other evil headers
else if (pRequest->HeaderValid[HttpHeaderRange])
{
Precondition = URI_PRE_OTHER_HEADER;
}
UlTrace(URI_CACHE, (
"Http!UlCheckCachePreconditions(req = %p, httpconn = %p)\n"
" OkToServeFromCache = %d, Precondition = %d\n",
pRequest,
pHttpConn,
(URI_PRE_OK == Precondition) ? 1 : 0,
Precondition
));
//
// update stats
//
if (URI_PRE_OK != Precondition) {
InterlockedIncrement((PLONG) &g_UriCacheStats.MissPreconditionCount);
}
return (URI_PRE_OK == Precondition);
} // UlCheckCachePreconditions
/***************************************************************************++
Routine Description:
This routine checks a response to see if it's cacheable. Basically
we'll take it if:
* the cache policy is right
* the size is small enough
* there is room in the cache
* we get the response all at once
Arguments:
pHttpConn - The connection to be checked
Return Value:
BOOLEAN - True if it's ok to serve from cache
--***************************************************************************/
BOOLEAN
UlCheckCacheResponseConditions(
PUL_INTERNAL_REQUEST pRequest,
PUL_INTERNAL_RESPONSE pResponse,
ULONG Flags,
HTTP_CACHE_POLICY CachePolicy
)
{
URI_PRECONDITION Precondition = URI_PRE_OK;
//
// Sanity check
//
PAGED_CODE();
ASSERT( UL_IS_VALID_INTERNAL_REQUEST(pRequest) );
ASSERT( UL_IS_VALID_INTERNAL_RESPONSE(pResponse) );
if (pRequest->CachePreconditions == FALSE) {
Precondition = URI_PRE_REQUEST;
}
// check policy
else if (CachePolicy.Policy == HttpCachePolicyNocache) {
Precondition = URI_PRE_POLICY;
}
// check size of response
else if ((pResponse->ResponseLength - pResponse->HeaderLength) >
g_UriCacheConfig.MaxUriBytes) {
Precondition = URI_PRE_SIZE;
}
// check for full response
else if (Flags & HTTP_SEND_RESPONSE_FLAG_MORE_DATA) {
Precondition = URI_PRE_FRAGMENT;
}
// check available cache table space
else if (!UlpCheckTableSpace(pResponse->ResponseLength)) {
Precondition = URI_PRE_NOMEMORY;
}
// Check for bogus responses
// BUGBUG: this check should be in ioctl.cxx
else if ((pResponse->ResponseLength < 1) || (pResponse->ChunkCount < 2)) {
Precondition = URI_PRE_BOGUS;
}
UlTrace(URI_CACHE, (
"Http!UlCheckCacheResponseConditions(pRequest = %p, pResponse = %p)\n"
" OkToCache = %d, Precondition = %d\n",
pRequest,
pResponse,
(URI_PRE_OK == Precondition),
Precondition
));
return (URI_PRE_OK == Precondition);
} // UlCheckCacheResponseConditions
/***************************************************************************++
Routine Description:
This routine does a cache lookup to see if there is a valid entry
corresponding to the request URI.
Arguments:
pRequest - The request
Return Value:
PUL_URI_CACHE_ENTRY - Pointer to the entry, if found. NULL otherwise.
--***************************************************************************/
PUL_URI_CACHE_ENTRY
UlCheckoutUriCacheEntry(
PUL_INTERNAL_REQUEST pRequest
)
{
PUL_URI_CACHE_ENTRY pUriCacheEntry = NULL;
ULONG BucketNumber;
URI_KEY Key;
//
// Sanity check
//
PAGED_CODE();
//
// find bucket
//
Key.Hash = pRequest->CookedUrl.Hash;
Key.Length = pRequest->CookedUrl.Length;
Key.pUri = pRequest->CookedUrl.pUrl;
ASSERT(!g_UriCacheConfig.EnableCache
|| IS_VALID_HASHTABLE(&g_UriCacheTable));
pUriCacheEntry = UlGetFromHashTable(&g_UriCacheTable, &Key);
if (pUriCacheEntry != NULL)
{
ASSERT( IS_VALID_URI_CACHE_ENTRY(pUriCacheEntry) );
ASSERT(! pUriCacheEntry->Zombie);
InterlockedIncrement((PLONG) &pUriCacheEntry->HitCount);
// reset scavenger counter
pUriCacheEntry->ScavengerTicks = 0;
UlTrace(URI_CACHE, (
"Http!UlCheckoutUriCacheEntry(pUriCacheEntry %p, '%ls') "
"refcount = %d\n",
pUriCacheEntry, pUriCacheEntry->UriKey.pUri,
pUriCacheEntry->ReferenceCount
));
//
// update stats
//
InterlockedIncrement((PLONG) &g_UriCacheStats.HitCount);
UlIncCounter(HttpGlobalCounterUriCacheHits);
}
else
{
InterlockedIncrement((PLONG) &g_UriCacheStats.MissTableCount);
UlIncCounter(HttpGlobalCounterUriCacheMisses);
}
return pUriCacheEntry;
} // UlCheckoutUriCacheEntry
/***************************************************************************++
Routine Description:
Decrements the refcount on a cache entry. Cleans up non-cached
entries.
Arguments:
pUriCacheEntry - the entry to deref
--***************************************************************************/
VOID
UlCheckinUriCacheEntry(
PUL_URI_CACHE_ENTRY pUriCacheEntry
)
{
LONG ReferenceCount;
BOOLEAN Cached;
//
// Sanity check
//
PAGED_CODE();
ASSERT(!g_UriCacheConfig.EnableCache
|| IS_VALID_HASHTABLE(&g_UriCacheTable));
ASSERT( IS_VALID_URI_CACHE_ENTRY(pUriCacheEntry) );
//
// check to see if it was cached while we have a reference
//
Cached = pUriCacheEntry->Cached;
//
// decrement count
//
ReferenceCount = DEREFERENCE_URI_CACHE_ENTRY(pUriCacheEntry, CHECKIN);
ASSERT(ReferenceCount >= 0);
if (ReferenceCount == 0)
{
// CODEWORK: add this entry to the zombie list, instead of
// deleting synchronously
// Make sure the zombie list gets periodically purged, even
// if the ScavengerPeriod is huge.
// The refcount of a cached entry can be zero if the entry
// was flushed from the cache while the entry was being
// sent back to a client.
}
else
{
// If the reference was not marked as cached, its refcount
// must now be zero
ASSERT(Cached);
}
} // UlCheckinUriCacheEntry
/***************************************************************************++
Routine Description:
Removes all cache entries
--***************************************************************************/
VOID
UlFlushCache(
VOID
)
{
//
// sanity check
//
PAGED_CODE();
ASSERT(!g_UriCacheConfig.EnableCache
|| IS_VALID_HASHTABLE(&g_UriCacheTable));
if (g_UriCacheConfig.EnableCache) {
UlTrace(URI_CACHE, (
"Http!UlFlushCache()\n"
));
UlpFilteredFlushUriCache(UlpFlushFilterAll, NULL);
}
} // UlFlushCache
/***************************************************************************++
Routine Description:
A filter for UlFlushCache. Called by UlpFilteredFlushUriCache.
Arguments:
pUriCacheEntry - the entry to check
pContext - ignored
--***************************************************************************/
UL_CACHE_PREDICATE
UlpFlushFilterAll(
IN PUL_URI_CACHE_ENTRY pUriCacheEntry,
IN PVOID pContext
)
{
PURI_FILTER_CONTEXT pUriFilterContext = (PURI_FILTER_CONTEXT) pContext;
//
// Sanity check
//
PAGED_CODE();
ASSERT( IS_VALID_URI_CACHE_ENTRY(pUriCacheEntry) );
ASSERT( pUriFilterContext != NULL
&& URI_FILTER_CONTEXT_POOL_TAG == pUriFilterContext->Signature
&& pUriFilterContext->pCallerContext == NULL );
UlTrace(URI_CACHE, (
"Http!UlpFlushFilterAll(pUriCacheEntry %p '%ls') refcount = %d\n",
pUriCacheEntry, pUriCacheEntry->UriKey.pUri,
pUriCacheEntry->ReferenceCount));
return UlpZombifyEntry(
TRUE,
pUriCacheEntry,
pUriFilterContext
);
} // UlpFlushFilterAll
/***************************************************************************++
Routine Description:
Removes any cache entries that were created by the given process.
Arguments:
pProcess - a process that is going away
--***************************************************************************/
VOID
UlFlushCacheByProcess(
PUL_APP_POOL_PROCESS pProcess
)
{
//
// sanity check
//
PAGED_CODE();
ASSERT( IS_VALID_AP_PROCESS(pProcess) );
ASSERT(!g_UriCacheConfig.EnableCache
|| IS_VALID_HASHTABLE(&g_UriCacheTable));
if (g_UriCacheConfig.EnableCache) {
UlTrace(URI_CACHE, (
"Http!UlFlushCacheByProcess(proc = %p)\n",
pProcess
));
UlpFilteredFlushUriCache(UlpFlushFilterProcess, pProcess);
}
} // UlFlushCacheByProcess
/***************************************************************************++
Routine Description:
Removes any cache entries that were created by the given process
whose url has a given prefix.
Arguments:
pUri - the uri prefix to match against
Length - length of the prefix
Flags - HTTP_FLUSH_RESPONSE_FLAG_RECURSIVE indicates a tree flush
pProcess - the process that made the call
--***************************************************************************/
VOID
UlFlushCacheByUri(
IN PWSTR pUri,
IN ULONG Length,
IN ULONG Flags,
PUL_APP_POOL_PROCESS pProcess
)
{
//
// sanity check
//
PAGED_CODE();
ASSERT( IS_VALID_AP_PROCESS(pProcess) );
ASSERT(!g_UriCacheConfig.EnableCache
|| IS_VALID_HASHTABLE(&g_UriCacheTable));
if (g_UriCacheConfig.EnableCache)
{
UlTrace(URI_CACHE, (
"Http!UlFlushCacheByUri(\n"
" uri = '%S'\n"
" len = %d\n"
" flags = %08x\n"
" proc = %p\n",
pUri,
Length,
Flags,
pProcess
));
if (Flags & HTTP_FLUSH_RESPONSE_FLAG_RECURSIVE) {
//
// CODEWORK: restrict the flush to the ones they actually
// asked for!
//
UlpFilteredFlushUriCache(UlpFlushFilterProcess, pProcess);
} else {
UlpFlushUri(
pUri,
Length,
pProcess
);
UlpClearZombieList();
}
}
} // UlFlushCacheByUri
/***************************************************************************++
Routine Description:
Removes a single URI from the table if the name and process match an
entry.
Arguments:
--***************************************************************************/
VOID
UlpFlushUri(
IN PWSTR pUri,
IN ULONG Length,
PUL_APP_POOL_PROCESS pProcess
)
{
PUL_URI_CACHE_ENTRY pUriCacheEntry = NULL;
URI_KEY Key;
LONG ReferenceCount;
//
// Sanity check
//
PAGED_CODE();
//
// find bucket
//
Key.Hash = HashRandomizeBits(HashStringNoCaseW(pUri, 0));
Key.Length = Length;
Key.pUri = pUri;
pUriCacheEntry = UlDeleteFromHashTable(&g_UriCacheTable, &Key);
if (NULL != pUriCacheEntry)
{
ASSERT( IS_VALID_URI_CACHE_ENTRY(pUriCacheEntry) );
UlTrace(URI_CACHE, (
"Http!UlpFlushUri(pUriCacheEntry %p '%ls') refcount = %d\n",
pUriCacheEntry, pUriCacheEntry->UriKey.pUri,
pUriCacheEntry->ReferenceCount));
DEREFERENCE_URI_CACHE_ENTRY(pUriCacheEntry, FLUSH);
//
// Perfmon counters
//
UlIncCounter(HttpGlobalCounterTotalFlushedUris);
}
} // UlpFlushUri
/***************************************************************************++
Routine Description:
A filter for UlFlushCacheByProcess. Called by UlpFilteredFlushUriCache.
Arguments:
pUriCacheEntry - the entry to check
pContext - pointer to the UL_APP_POOL_PROCESS that's going away
--***************************************************************************/
UL_CACHE_PREDICATE
UlpFlushFilterProcess(
IN PUL_URI_CACHE_ENTRY pUriCacheEntry,
IN PVOID pContext
)
{
PURI_FILTER_CONTEXT pUriFilterContext = (PURI_FILTER_CONTEXT) pContext;
PUL_APP_POOL_PROCESS pProcess;
//
// Sanity check
//
PAGED_CODE();
ASSERT( IS_VALID_URI_CACHE_ENTRY(pUriCacheEntry) );
ASSERT( pUriFilterContext != NULL
&& URI_FILTER_CONTEXT_POOL_TAG == pUriFilterContext->Signature
&& pUriFilterContext->pCallerContext != NULL );
pProcess = (PUL_APP_POOL_PROCESS) pUriFilterContext->pCallerContext;
ASSERT( IS_VALID_AP_PROCESS(pProcess) );
return UlpZombifyEntry(
(pProcess == pUriCacheEntry->pProcess),
pUriCacheEntry,
pUriFilterContext
);
} // UlpFlushFilterProcess
/***************************************************************************++
Routine Description:
Checks the hash table to make sure there is room for one more
entry of a given size.
This routine must be called with the table resource held.
Arguments:
EntrySize - the size in bytes of the entry to be added
--***************************************************************************/
BOOLEAN
UlpCheckTableSpace(
IN ULONGLONG EntrySize
)
{
ULONG UriCount;
ULONGLONG ByteCount;
UriCount = g_UriCacheStats.UriCount + 1;
ByteCount = g_UriCacheStats.ByteCount + EntrySize;
//
// CODEWORK: MaxCacheMegabyteCount of zero should mean adaptive limit,
// but for now I'll take it to mean "no limit".
//
if (g_UriCacheConfig.MaxCacheMegabyteCount == 0) {
ByteCount = 0;
}
//
// MaxCacheUriCount of zero means no limit on number of URIs cached
//
if (g_UriCacheConfig.MaxCacheUriCount == 0) {
UriCount = 0;
}
if (
UriCount <= g_UriCacheConfig.MaxCacheUriCount &&
ByteCount <= (g_UriCacheConfig.MaxCacheMegabyteCount << MEGABYTE_SHIFT)
)
{
return TRUE;
} else {
UlTrace(URI_CACHE, (
"Http!UlpCheckTableSpace(%d) FALSE\n"
" UriCount = %d\n"
" ByteCount = %I64d (%dMB)\n"
" MaxCacheUriCount = %d\n"
" MaxCacheMegabyteCount = %dMB\n",
EntrySize,
g_UriCacheStats.UriCount,
g_UriCacheStats.ByteCount,
g_UriCacheStats.ByteCount >> MEGABYTE_SHIFT,
g_UriCacheConfig.MaxCacheUriCount,
g_UriCacheConfig.MaxCacheMegabyteCount
));
return FALSE;
}
} // UlpCheckTableSpace
/***************************************************************************++
Routine Description:
Tries to add a cache entry to the hash table.
Arguments:
pUriCacheEntry - the entry to be added
--***************************************************************************/
VOID
UlAddCacheEntry(
PUL_URI_CACHE_ENTRY pUriCacheEntry
)
{
ULC_RETCODE rc = ULC_SUCCESS;
//
// Sanity check
//
PAGED_CODE();
ASSERT(!g_UriCacheConfig.EnableCache
|| IS_VALID_HASHTABLE(&g_UriCacheTable));
ASSERT( IS_VALID_URI_CACHE_ENTRY(pUriCacheEntry) );
ASSERT(! pUriCacheEntry->Zombie);
pUriCacheEntry->BucketEntry.Next = NULL;
pUriCacheEntry->Cached = FALSE;
// First, check if still has space for storing the cache entry
if (UlpCheckSpaceAndAddEntryStats(pUriCacheEntry))
{
pUriCacheEntry->Cached = TRUE;
//
// Insert this record into the hash table
// Check first to see if the key already presents
//
rc = UlAddToHashTable(&g_UriCacheTable, pUriCacheEntry);
}
UlTrace(URI_CACHE, (
"Http!UlAddCacheEntry(urientry %p '%ls') %s added to table. "
"RefCount=%d, lkrc=%d.\n",
pUriCacheEntry, pUriCacheEntry->UriKey.pUri,
pUriCacheEntry->Cached ? "was" : "was not",
pUriCacheEntry->ReferenceCount,
rc
));
} // UlAddCacheEntry
/***************************************************************************++
Routine Description:
Check to see if we have space to add this cache entry and if so update
cache statistics to reflect the addition of an entry. This has to be
done together inside a lock.
Arguments:
pUriCacheEntry - entry being added
--***************************************************************************/
BOOLEAN
UlpCheckSpaceAndAddEntryStats(
PUL_URI_CACHE_ENTRY pUriCacheEntry
)
{
KIRQL OldIrql;
ULONG EntrySize;
//
// Sanity check
//
ASSERT( IS_VALID_URI_CACHE_ENTRY(pUriCacheEntry) );
EntrySize = pUriCacheEntry->HeaderLength + pUriCacheEntry->ContentLength;
if (!pUriCacheEntry->LongTermCacheable)
{
//
// Schedule the scavenger immediately, but only do it once
//
if (FALSE == InterlockedExchange(&g_UriScavengerRunning, TRUE))
{
UL_QUEUE_WORK_ITEM(
&g_UriScavengerWorkItem,
&UlpScavenger
);
}
return FALSE;
}
UlAcquireSpinLock( &g_UriCacheSpinLock, &OldIrql );
if (UlpCheckTableSpace(EntrySize))
{
g_UriCacheStats.UriCount++;
g_UriCacheStats.UriAddedTotal++;
g_UriCacheStats.UriCountMax = MAX(
g_UriCacheStats.UriCountMax,
g_UriCacheStats.UriCount
);
g_UriCacheStats.ByteCount += EntrySize;
g_UriCacheStats.ByteCountMax = MAX(
g_UriCacheStats.ByteCountMax,
g_UriCacheStats.ByteCount
);
UlReleaseSpinLock( &g_UriCacheSpinLock, OldIrql );
UlTrace(URI_CACHE, (
"Http!UlpCheckSpaceAndAddEntryStats (urientry %p '%ls')\n",
pUriCacheEntry, pUriCacheEntry->UriKey.pUri
));
//
// Perfmon counters
//
UlIncCounter(HttpGlobalCounterCurrentUrisCached);
UlIncCounter(HttpGlobalCounterTotalUrisCached);
return TRUE;
}
UlReleaseSpinLock( &g_UriCacheSpinLock, OldIrql );
return FALSE;
} // UlpCheckSpaceAndAddEntryStats
/***************************************************************************++
Routine Description:
Updates cache statistics to reflect the removal of an entry
Arguments:
pUriCacheEntry - entry being added
--***************************************************************************/
VOID
UlpRemoveEntryStats(
PUL_URI_CACHE_ENTRY pUriCacheEntry
)
{
KIRQL OldIrql;
ULONG EntrySize;
//
// Sanity check
//
ASSERT( IS_VALID_URI_CACHE_ENTRY(pUriCacheEntry) );
ASSERT( pUriCacheEntry->Cached );
EntrySize = pUriCacheEntry->HeaderLength + pUriCacheEntry->ContentLength;
UlAcquireSpinLock( &g_UriCacheSpinLock, &OldIrql );
g_UriCacheStats.UriCount--;
g_UriCacheStats.ByteCount -= EntrySize;
UlReleaseSpinLock( &g_UriCacheSpinLock, OldIrql );
UlTrace(URI_CACHE, (
"Http!UlpRemoveEntryStats (urientry %p '%ls')\n",
pUriCacheEntry, pUriCacheEntry->UriKey.pUri
));
//
// Perfmon counters
//
UlDecCounter(HttpGlobalCounterCurrentUrisCached);
} // UlpRemoveEntryStats
/***************************************************************************++
Routine Description:
Helper function for the filter callbacks indirectly invoked by
UlpFilteredFlushUriCache. Adds deleteable entries to a temporary
list.
Arguments:
MustZombify - if TRUE, add entry to the private zombie list
pUriCacheEntry - entry to zombify
pUriFilterContext - contains private list
--***************************************************************************/
UL_CACHE_PREDICATE
UlpZombifyEntry(
BOOLEAN MustZombify,
IN PUL_URI_CACHE_ENTRY pUriCacheEntry,
IN PURI_FILTER_CONTEXT pUriFilterContext
)
{
ASSERT( IS_VALID_URI_CACHE_ENTRY(pUriCacheEntry) );
ASSERT(URI_FILTER_CONTEXT_POOL_TAG == pUriFilterContext->Signature);
ASSERT(! pUriCacheEntry->Zombie);
ASSERT(NULL == pUriCacheEntry->ZombieListEntry.Flink);
if (MustZombify)
{
//
// Temporarily bump the refcount up so that it won't go down
// to zero when it's removed from the hash table, automatically
// invoking UlpDestroyUriCacheEntry, which we are trying to defer.
//
pUriCacheEntry->ZombieAddReffed = TRUE;
REFERENCE_URI_CACHE_ENTRY(pUriCacheEntry, ZOMBIFY);
InsertTailList(
&pUriFilterContext->ZombieListHead,
&pUriCacheEntry->ZombieListEntry);
pUriCacheEntry->Zombie = TRUE;
//
// reset timer so we can track how long an entry is on the list
//
pUriCacheEntry->ScavengerTicks = 0;
++ pUriFilterContext->ZombieCount;
// now remove it from the hash table
return ULC_DELETE;
}
// do not remove pUriCacheEntry from table
return ULC_NO_ACTION;
} // UlpZombifyEntry
/***************************************************************************++
Routine Description:
Adds a list of entries to the global zombie list, then calls
UlpClearZombieList. This cleans up the list of deferred deletions
built up by UlpFilteredFlushUriCache.
Runs at passive level.
Arguments:
pWorkItem - workitem within a URI_FILTER_CONTEXT containing private list
--***************************************************************************/
VOID
UlpZombifyList(
IN PUL_WORK_ITEM pWorkItem
)
{
PAGED_CODE();
ASSERT(NULL != pWorkItem);
PURI_FILTER_CONTEXT pUriFilterContext
= CONTAINING_RECORD(pWorkItem, URI_FILTER_CONTEXT, WorkItem);
ASSERT(URI_FILTER_CONTEXT_POOL_TAG == pUriFilterContext->Signature);
UlTrace(URI_CACHE, (
"http!UlpZombifyList, ctxt = %p\n",
pUriFilterContext
));
UlAcquireResourceExclusive(&g_pUlNonpagedData->UriZombieResource, TRUE);
//
// Splice the entire private list into the head of the Zombie list
//
ASSERT(! IsListEmpty(&pUriFilterContext->ZombieListHead));
PLIST_ENTRY pContextHead = pUriFilterContext->ZombieListHead.Flink;
PLIST_ENTRY pContextTail = pUriFilterContext->ZombieListHead.Blink;
PLIST_ENTRY pZombieHead = g_ZombieListHead.Flink;
pContextTail->Flink = pZombieHead;
pZombieHead->Blink = pContextTail;
g_ZombieListHead.Flink = pContextHead;
pContextHead->Blink = &g_ZombieListHead;
// Update stats
g_UriCacheStats.ZombieCount += pUriFilterContext->ZombieCount;
g_UriCacheStats.ZombieCountMax = MAX(g_UriCacheStats.ZombieCount,
g_UriCacheStats.ZombieCountMax);
#if DBG
PLIST_ENTRY pEntry;
ULONG ZombieCount;
// Walk forwards through the zombie list and check that it contains
// exactly as many valid zombied UriCacheEntries as we expect.
for (pEntry = g_ZombieListHead.Flink, ZombieCount = 0;
pEntry != &g_ZombieListHead;
pEntry = pEntry->Flink, ++ZombieCount)
{
PUL_URI_CACHE_ENTRY pUriCacheEntry
= CONTAINING_RECORD(pEntry, UL_URI_CACHE_ENTRY, ZombieListEntry);
ASSERT( IS_VALID_URI_CACHE_ENTRY(pUriCacheEntry) );
ASSERT(pUriCacheEntry->Zombie);
ASSERT(pUriCacheEntry->ZombieAddReffed
? pUriCacheEntry->ScavengerTicks == 0
: pUriCacheEntry->ScavengerTicks > 0);
ASSERT(ZombieCount < g_UriCacheStats.ZombieCount);
}
ASSERT(ZombieCount == g_UriCacheStats.ZombieCount);
// And backwards too, like Ginger Rogers
for (pEntry = g_ZombieListHead.Blink, ZombieCount = 0;
pEntry != &g_ZombieListHead;
pEntry = pEntry->Blink, ++ZombieCount)
{
PUL_URI_CACHE_ENTRY pUriCacheEntry
= CONTAINING_RECORD(pEntry, UL_URI_CACHE_ENTRY, ZombieListEntry);
ASSERT( IS_VALID_URI_CACHE_ENTRY(pUriCacheEntry) );
ASSERT(pUriCacheEntry->Zombie);
ASSERT(pUriCacheEntry->ZombieAddReffed
? pUriCacheEntry->ScavengerTicks == 0
: pUriCacheEntry->ScavengerTicks > 0);
ASSERT(ZombieCount < g_UriCacheStats.ZombieCount);
}
ASSERT(ZombieCount == g_UriCacheStats.ZombieCount);
#endif // DBG
UlReleaseResource(&g_pUlNonpagedData->UriZombieResource);
UL_FREE_POOL_WITH_SIG(pUriFilterContext, URI_FILTER_CONTEXT_POOL_TAG);
// Now purge those entries, if there are no outstanding references
UlpClearZombieList();
} // UlpZombifyList
/***************************************************************************++
Routine Description:
Removes entries based on a caller specified filter. The caller
provides a boolean function which takes a cache entry as a
parameter. The function will be called with each item in the cache.
The function should conclude with a call to UlpZombifyEntry, passing
in whether or not the item should be deleted. See sample usage
elsewhere in this file.
Arguments:
pFilterRoutine - A pointer to the filter function
pCallerContext - a parameter to the filter function
--***************************************************************************/
VOID
UlpFilteredFlushUriCache(
IN PUL_URI_FILTER pFilterRoutine,
IN PVOID pCallerContext
)
{
PURI_FILTER_CONTEXT pUriFilterContext;
LONG ZombieCount = 0;
//
// sanity check
//
PAGED_CODE();
ASSERT( NULL != pFilterRoutine );
//
// Perfmon counters
//
UlIncCounter(HttpGlobalCounterUriCacheFlushes);
pUriFilterContext = UL_ALLOCATE_STRUCT(
NonPagedPool,
URI_FILTER_CONTEXT,
URI_FILTER_CONTEXT_POOL_TAG);
if (pUriFilterContext == NULL)
return;
pUriFilterContext->Signature = URI_FILTER_CONTEXT_POOL_TAG;
InitializeListHead(&pUriFilterContext->ZombieListHead);
pUriFilterContext->pCallerContext = pCallerContext;
pUriFilterContext->ZombieCount = 0;
UlTrace(URI_CACHE, (
"Http!UlFilteredFlushUriCache(filt = %p, ctxt = %p)\n",
pFilterRoutine, pUriFilterContext
));
if (IS_VALID_HASHTABLE(&g_UriCacheTable))
{
ZombieCount = UlFilterFlushHashTable(
&g_UriCacheTable,
pFilterRoutine,
pUriFilterContext
);
ASSERT(ZombieCount == pUriFilterContext->ZombieCount);
if (0 != ZombieCount)
{
UlAddCounter(HttpGlobalCounterTotalFlushedUris, ZombieCount);
UL_QUEUE_WORK_ITEM(
&pUriFilterContext->WorkItem,
UlpZombifyList
);
}
else
{
UL_FREE_POOL_WITH_SIG(pUriFilterContext,
URI_FILTER_CONTEXT_POOL_TAG);
}
UlTrace(URI_CACHE, (
"Http!UlFilteredFlushUriCache(filt = %p, caller ctxt = %p)"
" Zombified: %d\n",
pFilterRoutine,
pCallerContext,
ZombieCount
));
}
} // UlpFilteredFlushUriCache
/***************************************************************************++
Routine Description:
Scans the zombie list for entries whose refcount has dropped to "zero".
(The calling routine is generally expected to have added a reference
(and set the ZombieAddReffed field within the entries), so that
otherwise unreferenced entries will actually have a refcount of one. It
works this way because we don't want the scavenger directly triggering
calls to UlpDestroyUriCacheEntry)
--***************************************************************************/
VOID
UlpClearZombieList(
VOID
)
{
ULONG ZombiesFreed = 0;
ULONG ZombiesSpared = 0;
PLIST_ENTRY pCurrent;
LONG ReferenceCount;
//
// sanity check
//
PAGED_CODE();
UlAcquireResourceExclusive(&g_pUlNonpagedData->UriZombieResource, TRUE);
pCurrent = g_ZombieListHead.Flink;
while (pCurrent != &g_ZombieListHead)
{
PUL_URI_CACHE_ENTRY pUriCacheEntry
= CONTAINING_RECORD(pCurrent, UL_URI_CACHE_ENTRY, ZombieListEntry);
ASSERT( IS_VALID_URI_CACHE_ENTRY(pUriCacheEntry) );
ASSERT(pUriCacheEntry->Zombie);
//
// get next entry now, because we might destroy this one
//
pCurrent = pCurrent->Flink;
//
// ReferenceCount is modified with interlocked ops, but in
// this case we know the ReferenceCount can't go up on
// a Zombie, and if an entry hits one just after we look
// at it, we'll just get it on the next pass
//
if (pUriCacheEntry->ZombieAddReffed)
{
bool LastRef = (pUriCacheEntry->ReferenceCount == 1);
if (LastRef)
{
RemoveEntryList(&pUriCacheEntry->ZombieListEntry);
pUriCacheEntry->ZombieListEntry.Flink = NULL;
++ ZombiesFreed;
ASSERT(g_UriCacheStats.ZombieCount > 0);
-- g_UriCacheStats.ZombieCount;
}
else
{
// track age of zombie
++ pUriCacheEntry->ScavengerTicks;
++ ZombiesSpared;
}
pUriCacheEntry->ZombieAddReffed = FALSE;
DEREFERENCE_URI_CACHE_ENTRY(pUriCacheEntry, UNZOMBIFY);
}
else
{
ASSERT(pUriCacheEntry->ScavengerTicks > 0);
// If we've released the zombie reference on it and it's still
// on the zombie list, somebody'd better have a reference.
ASSERT(pUriCacheEntry->ReferenceCount > 0);
// track age of zombie
++ pUriCacheEntry->ScavengerTicks;
++ ZombiesSpared;
if (pUriCacheEntry->ScavengerTicks > ZOMBIE_AGE_THRESHOLD)
{
UlTrace(URI_CACHE, (
"Http!UlpClearZombieList()\n"
" WARNING: %p '%ls' (refs = %d) "
"has been a zombie for %d ticks!\n",
pUriCacheEntry, pUriCacheEntry->UriKey.pUri,
pUriCacheEntry->ReferenceCount,
pUriCacheEntry->ScavengerTicks
));
}
}
}
ASSERT(ZombiesSpared == g_UriCacheStats.ZombieCount);
ASSERT((g_UriCacheStats.ZombieCount == 0)
== IsListEmpty(&g_ZombieListHead));
UlReleaseResource(&g_pUlNonpagedData->UriZombieResource);
UlTrace(URI_CACHE, (
"Http!UlpClearZombieList(): Freed = %d, Remaining = %d.\n",
ZombiesFreed,
ZombiesSpared
));
} // UlpClearZombieList
/***************************************************************************++
Routine Description:
Frees a URI entry to the pool. Removes references to other objects.
Arguments:
pTracker - Supplies the UL_READ_TRACKER to manipulate.
--***************************************************************************/
VOID
UlpDestroyUriCacheEntry(
PUL_URI_CACHE_ENTRY pUriCacheEntry
)
{
//
// Sanity check
//
PAGED_CODE();
ASSERT( IS_VALID_URI_CACHE_ENTRY(pUriCacheEntry) );
// CODEWORK: real cleanup will need to release
// config & process references.
ASSERT(0 == pUriCacheEntry->ReferenceCount);
UlTrace(URI_CACHE, (
"Http!UlpDestroyUriCacheEntry: Entry %p, '%ls', Refs=%d\n",
pUriCacheEntry, pUriCacheEntry->UriKey.pUri,
pUriCacheEntry->ReferenceCount
));
//
// Release the UL_URL_CONFIG_GROUP_INFO block
//
UlpConfigGroupInfoRelease(&pUriCacheEntry->ConfigInfo);
UlLargeMemFree(pUriCacheEntry->pResponseMdl);
//
// Remove from g_ZombieListHead if neccessary
//
if (pUriCacheEntry->ZombieListEntry.Flink != NULL)
{
ASSERT(pUriCacheEntry->Zombie);
ASSERT(! pUriCacheEntry->ZombieAddReffed);
UlAcquireResourceExclusive(
&g_pUlNonpagedData->UriZombieResource,
TRUE);
RemoveEntryList(&pUriCacheEntry->ZombieListEntry);
ASSERT(g_UriCacheStats.ZombieCount > 0);
-- g_UriCacheStats.ZombieCount;
UlReleaseResource(&g_pUlNonpagedData->UriZombieResource);
}
UL_FREE_POOL_WITH_SIG(
pUriCacheEntry,
UL_URI_CACHE_ENTRY_POOL_TAG
);
} // UlpDestroyUriCacheEntry
/***************************************************************************++
Routine Description:
Initializes the cache scavenger.
--***************************************************************************/
VOID
UlpInitializeScavenger(
VOID
)
{
UlTrace(URI_CACHE, (
"Http!UlpInitializeScavenger\n"
));
g_UriScavengerInitialized = TRUE;
g_UriScavengerRunning = FALSE;
UlInitializeSpinLock(
&g_UriScavengerSpinLock,
"g_UriScavengerSpinLock"
);
KeInitializeDpc(
&g_UriScavengerDpc, // DPC object
&UlpScavengerDpcRoutine, // DPC routine
NULL // context
);
KeInitializeTimer(
&g_UriScavengerTimer
);
KeInitializeEvent(
&g_UriScavengerTerminationEvent,
NotificationEvent,
FALSE
);
UlpSetScavengerTimer();
} // UlpInitializeScavenger
/***************************************************************************++
Routine Description:
Increments the current chunk pointer in the tracker and initializes
some of the "from file" related tracker fields if necessary.
Arguments:
pTracker - Supplies the UL_READ_TRACKER to manipulate.
--***************************************************************************/
VOID
UlpTerminateScavenger(
VOID
)
{
KIRQL oldIrql;
UlTrace(URI_CACHE, (
"Http!UlpTerminateScavenger\n"
));
if (g_UriScavengerInitialized)
{
//
// Clear the "initialized" flag. If the scavenger runs soon,
// it will see this flag, set the termination event, and exit
// quickly.
//
UlAcquireSpinLock(
&g_UriScavengerSpinLock,
&oldIrql
);
g_UriScavengerInitialized = FALSE;
UlReleaseSpinLock(
&g_UriScavengerSpinLock,
oldIrql
);
//
// Cancel the scavenger timer. If the cancel fails, then the
// scavenger is either running or scheduled to run soon. In
// either case, wait for it to terminate.
//
if ( !KeCancelTimer( &g_UriScavengerTimer ) )
{
KeWaitForSingleObject(
(PVOID)&g_UriScavengerTerminationEvent,
UserRequest,
KernelMode,
FALSE,
NULL
);
}
}
//
// clear out anything remaining
//
UlpClearZombieList();
//
// The EndpointDrain should have closed all connections and released
// all references to cache entries
//
ASSERT( g_UriCacheStats.ZombieCount == 0 );
ASSERT( IsListEmpty(&g_ZombieListHead) );
} // UlpTerminateScavenger
/***************************************************************************++
Routine Description:
Figures out the scavenger interval in 100 ns ticks, and sets the timer.
--***************************************************************************/
VOID
UlpSetScavengerTimer(
VOID
)
{
LARGE_INTEGER ScavengerInterval;
//
// convert seconds to 100 nanosecond intervals (x * 10^7)
// negative numbers mean relative time
//
ScavengerInterval.QuadPart= g_UriCacheConfig.ScavengerPeriod
* -C_NS_TICKS_PER_SEC;
UlTrace(URI_CACHE, (
"Http!UlpSetScavengerTimer: %d seconds = %I64d 100ns ticks\n",
g_UriCacheConfig.ScavengerPeriod,
ScavengerInterval.QuadPart
));
KeSetTimer(
&g_UriScavengerTimer,
ScavengerInterval,
&g_UriScavengerDpc
);
} // UlpSetScavengerTimer
/***************************************************************************++
Routine Description:
Executes every UriScavengerPeriodSeconds, or when our timer gets
cancelled (on shutdown). If we're not shutting down, we run the
UlpScavenger (at passive level).
Arguments:
I ignore all of these.
--***************************************************************************/
VOID
UlpScavengerDpcRoutine(
IN PKDPC Dpc,
IN PVOID DeferredContext,
IN PVOID SystemArgument1,
IN PVOID SystemArgument2
)
{
// UlTrace(URI_CACHE, (
// "Http!UlpScavengerDpcRoutine(...)\n"
// ));
UlAcquireSpinLockAtDpcLevel(
&g_UriScavengerSpinLock
);
if( !g_UriScavengerInitialized ) {
//
// We're shutting down, so signal the termination event.
//
KeSetEvent(
&g_UriScavengerTerminationEvent,
0,
FALSE
);
} else {
//
// Do that scavenger thang.
//
if (FALSE == InterlockedExchange(&g_UriScavengerRunning, TRUE)) {
UL_QUEUE_WORK_ITEM(
&g_UriScavengerWorkItem,
&UlpScavenger
);
}
}
UlReleaseSpinLockFromDpcLevel(
&g_UriScavengerSpinLock
);
} // UlpScavengerDpcRoutine
/***************************************************************************++
Routine Description:
Looks through the cache for expired entries to put on the zombie list,
and then empties out the list.
Arguments:
pWorkItem - ignored
--***************************************************************************/
VOID
UlpScavenger(
IN PUL_WORK_ITEM pWorkItem
)
{
KIRQL oldIrql;
UlTrace(URI_CACHE, (
"Http!UlpScavenger()\n"
));
ASSERT( TRUE == g_UriScavengerRunning );
ASSERT( &g_UriScavengerWorkItem == pWorkItem );
if (g_UriScavengerInitialized)
{
UlpFilteredFlushUriCache(UlpFlushFilterScavenger, NULL);
//
// allow other instances of the scavenger to run
//
InterlockedExchange(&g_UriScavengerRunning, FALSE);
}
UlAcquireSpinLock(&g_UriScavengerSpinLock, &oldIrql);
if (g_UriScavengerInitialized)
{
//
// restart the timer
//
UlpSetScavengerTimer();
}
else
{
KeSetEvent(
&g_UriScavengerTerminationEvent,
0,
FALSE
);
}
UlReleaseSpinLock(&g_UriScavengerSpinLock, oldIrql);
} // UlpScavenger
/***************************************************************************++
Routine Description:
A filter for UlpScavenger. Called by UlpFilteredFlushUriCache.
Arguments:
pUriCacheEntry - the entry to check
pContext - ignored
--***************************************************************************/
UL_CACHE_PREDICATE
UlpFlushFilterScavenger(
IN PUL_URI_CACHE_ENTRY pUriCacheEntry,
IN PVOID pContext
)
{
PURI_FILTER_CONTEXT pUriFilterContext = (PURI_FILTER_CONTEXT) pContext;
//
// Sanity check
//
PAGED_CODE();
ASSERT( IS_VALID_URI_CACHE_ENTRY(pUriCacheEntry) );
ASSERT( pUriFilterContext != NULL
&& URI_FILTER_CONTEXT_POOL_TAG == pUriFilterContext->Signature
&& pUriFilterContext->pCallerContext == NULL );
pUriCacheEntry->ScavengerTicks++;
//
// CODEWORK: need to check for expiration time as well
//
return UlpZombifyEntry(
(pUriCacheEntry->ScavengerTicks > CACHE_ENTRY_AGE_THRESHOLD),
pUriCacheEntry,
pUriFilterContext
);
} // UlpFlushFilterScavenger
/***************************************************************************++
Routine Description:
Determine if the Translate header is present AND has a value of 'F' or 'f'.
Arguments:
pRequest - Supplies the request to query.
Return Value:
BOOLEAN - TRUE if "Translate: F", FALSE otherwise
--***************************************************************************/
BOOLEAN
UlpQueryTranslateHeader(
IN PUL_INTERNAL_REQUEST pRequest
)
{
BOOLEAN ret = FALSE;
if ( pRequest->HeaderValid[HttpHeaderTranslate] )
{
PUCHAR pValue = pRequest->Headers[HttpHeaderTranslate].pHeader;
ASSERT(NULL != pValue);
if ('f' == pValue[0] || 'F' == pValue[0])
{
ret = TRUE;
}
}
return ret;
} // UlpQueryTranslateHeader
/***************************************************************************++
Routine Description:
Add a reference on a cache entry
Arguments:
pUriCacheEntry - the entry to addref
--***************************************************************************/
LONG
UlAddRefUriCacheEntry(
IN PUL_URI_CACHE_ENTRY pUriCacheEntry,
IN REFTRACE_ACTION Action
REFERENCE_DEBUG_FORMAL_PARAMS
)
{
ASSERT( IS_VALID_URI_CACHE_ENTRY(pUriCacheEntry) );
LONG RefCount = InterlockedIncrement(&pUriCacheEntry->ReferenceCount);
WRITE_REF_TRACE_LOG(
g_UriTraceLog,
Action,
RefCount,
pUriCacheEntry,
pFileName,
LineNumber
);
UlTrace(URI_CACHE, (
"Http!UlAddRefUriCacheEntry: Entry %p, refcount=%d.\n",
pUriCacheEntry, RefCount
));
ASSERT(RefCount > 0);
return RefCount;
} // UlAddRefUriCacheEntry
/***************************************************************************++
Routine Description:
Release a reference on a cache entry
Arguments:
pUriCacheEntry - the entry to release
--***************************************************************************/
LONG
UlReleaseUriCacheEntry(
IN PUL_URI_CACHE_ENTRY pUriCacheEntry,
IN REFTRACE_ACTION Action
REFERENCE_DEBUG_FORMAL_PARAMS
)
{
ASSERT( IS_VALID_URI_CACHE_ENTRY(pUriCacheEntry) );
LONG RefCount = InterlockedDecrement(&pUriCacheEntry->ReferenceCount);
WRITE_REF_TRACE_LOG(
g_UriTraceLog,
Action,
RefCount,
pUriCacheEntry,
pFileName,
LineNumber
);
UlTrace(URI_CACHE, (
"Http!UlReleaseUriCacheEntry: (pUriCacheEntry %p '%ls')"
"refcount = %d\n",
pUriCacheEntry, pUriCacheEntry->UriKey.pUri,
RefCount
));
ASSERT(RefCount >= 0);
if (RefCount == 0)
{
if (pUriCacheEntry->Cached)
UlpRemoveEntryStats(pUriCacheEntry);
UlpDestroyUriCacheEntry(pUriCacheEntry);
}
return RefCount;
} // UlReleaseUriCacheEntry
/***************************************************************************++
Routine Description:
UL_URI_CACHE_ENTRY pseudo-constructor. Primarily used for
AddRef and tracelogging.
Arguments:
pUriCacheEntry - the entry to initialize
Hash - Hash code of pUrl
Length - Length (in bytes) of pUrl
pUrl - Unicode URL to copy
--***************************************************************************/
VOID
UlInitCacheEntry(
PUL_URI_CACHE_ENTRY pUriCacheEntry,
ULONG Hash,
ULONG Length,
PCWSTR pUrl
)
{
pUriCacheEntry->Signature = UL_URI_CACHE_ENTRY_POOL_TAG;
pUriCacheEntry->ReferenceCount = 0;
pUriCacheEntry->HitCount = 1;
pUriCacheEntry->Zombie = FALSE;
pUriCacheEntry->ZombieAddReffed = FALSE;
pUriCacheEntry->ZombieListEntry.Flink = NULL;
pUriCacheEntry->ZombieListEntry.Blink = NULL;
pUriCacheEntry->Cached = FALSE;
pUriCacheEntry->ScavengerTicks = 0;
pUriCacheEntry->UriKey.Hash = Hash;
pUriCacheEntry->UriKey.Length = Length;
pUriCacheEntry->UriKey.pUri = (PWSTR) ((PCHAR)pUriCacheEntry +
ALIGN_UP(sizeof(UL_URI_CACHE_ENTRY), PVOID));
RtlCopyMemory(
pUriCacheEntry->UriKey.pUri,
pUrl,
pUriCacheEntry->UriKey.Length + sizeof(WCHAR)
);
REFERENCE_URI_CACHE_ENTRY(pUriCacheEntry, CREATE);
UlTrace(URI_CACHE, (
"Http!UlInitCacheEntry (%p = '%ls')\n",
pUriCacheEntry, pUriCacheEntry->UriKey.pUri
));
} // UlInitCacheEntry