You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
3215 lines
68 KiB
3215 lines
68 KiB
/*++
|
|
|
|
Copyright (c) 2000-2001 Microsoft Corporation
|
|
|
|
Module Name:
|
|
|
|
ncache.c
|
|
|
|
Abstract:
|
|
|
|
DNS Resolver Service
|
|
|
|
Cache routines
|
|
|
|
Author:
|
|
|
|
Jim Gilroy (jamesg) April 2001
|
|
|
|
Revision History:
|
|
|
|
--*/
|
|
|
|
|
|
#include "local.h"
|
|
|
|
|
|
//
|
|
// Heap corruption tracking
|
|
//
|
|
|
|
#define HEAPPROB 1
|
|
|
|
#define BAD_PTR (PVOID)(-1)
|
|
|
|
|
|
//
|
|
// Cache entry definitions
|
|
//
|
|
// Starting cache record count
|
|
//
|
|
|
|
#define CACHE_DEFAULT_SET_COUNT 3
|
|
|
|
#if 0
|
|
// Should be private but is exposed in remote
|
|
// cache enum routines.
|
|
|
|
typedef struct _CacheEntry
|
|
{
|
|
struct _CacheEntry * pNext;
|
|
PWSTR pName;
|
|
DWORD Reserved;
|
|
DWORD MaxCount;
|
|
PDNS_RECORD Records[ 1 ];
|
|
}
|
|
CACHE_ENTRY, *PCACHE_ENTRY;
|
|
#endif
|
|
|
|
//
|
|
// Cache heap
|
|
//
|
|
|
|
HANDLE g_CacheHeap = NULL;
|
|
|
|
//
|
|
// Cache hash table
|
|
//
|
|
|
|
PCACHE_ENTRY * g_HashTable = NULL;
|
|
|
|
#define INITIAL_CACHE_HEAP_SIZE (16*1024)
|
|
|
|
|
|
//
|
|
// Runtime globals
|
|
//
|
|
|
|
DWORD g_CurrentCacheTime;
|
|
|
|
DWORD g_RecordSetCount;
|
|
DWORD g_RecordSetCountLimit;
|
|
DWORD g_RecordSetCountThreshold;
|
|
|
|
DWORD g_RecordSetCache;
|
|
DWORD g_RecordSetFree;
|
|
|
|
DWORD g_EntryCount;
|
|
DWORD g_EntryAlloc;
|
|
DWORD g_EntryFree;
|
|
|
|
BOOL g_fLoadingHostsFile;
|
|
|
|
//
|
|
// Garbage collection
|
|
//
|
|
|
|
BOOL g_GarbageCollectFlag = FALSE;
|
|
|
|
DWORD g_NextGarbageIndex = 0;
|
|
DWORD g_NextGarbageTime = 0;
|
|
|
|
#define GARBAGE_LOCKOUT_INTERVAL (600) // no more then every ten minutes
|
|
|
|
|
|
//
|
|
// Wakeup flag
|
|
//
|
|
|
|
BOOL g_WakeFlag = FALSE;
|
|
|
|
//
|
|
// Cache limits
|
|
// - min count of records to hold
|
|
// - size of band in which garbage collection occurs
|
|
//
|
|
|
|
#if DBG
|
|
#define MIN_DYNAMIC_RECORD_COUNT (20)
|
|
#define CLEANUP_RECORD_COUNT_BAND (5)
|
|
#else
|
|
#define MIN_DYNAMIC_RECORD_COUNT (50)
|
|
#define CLEANUP_RECORD_COUNT_BAND (30)
|
|
#endif
|
|
|
|
|
|
//
|
|
// Static records (hosts file)
|
|
//
|
|
|
|
#define IS_STATIC_RR(prr) (IS_HOSTS_FILE_RR(prr) || IS_CLUSTER_RR(prr))
|
|
|
|
|
|
|
|
//
|
|
// Compute a hash table index value for a string
|
|
//
|
|
|
|
#define EOS (L'\0')
|
|
|
|
#define COMPUTE_STRING_HASH_1( _String, _ulHashTableSize, _lpulHash ) \
|
|
{ \
|
|
PWCHAR p; \
|
|
ULOND h = 0, g; \
|
|
\
|
|
for ( p = _String; *p != EOS; p = p + 1 ) \
|
|
{ \
|
|
h = ( h << 4 ) + (DWORD) (*p); \
|
|
if ( g = h&0xf0000000 ) \
|
|
{ \
|
|
h = h ^ ( g >> 24 ); \
|
|
h = h ^ g; \
|
|
} \
|
|
} \
|
|
*_lpulHash = h % _ulHashTableSize; \
|
|
}
|
|
|
|
|
|
//
|
|
// Compute a hash table index value for a string
|
|
// which is invairant to case
|
|
//
|
|
#define COMPUTE_STRING_HASH_2( _String, _ulHashTableSize, _lpulHash ) \
|
|
{ \
|
|
PWCHAR _p = _String; \
|
|
PWCHAR _ep = _p + wcslen( _String ); \
|
|
ULONG h = 0; \
|
|
\
|
|
while( _p < _ep ) \
|
|
{ \
|
|
h <<= 1; \
|
|
h ^= *_p++; \
|
|
} \
|
|
\
|
|
*_lpulHash = h % _ulHashTableSize; \
|
|
}
|
|
|
|
|
|
//
|
|
// Private prototypes
|
|
//
|
|
|
|
BOOL
|
|
Cache_FlushEntryRecords(
|
|
IN OUT PCACHE_ENTRY pEntry,
|
|
IN DWORD Level,
|
|
IN WORD wType
|
|
);
|
|
|
|
VOID
|
|
Cache_FlushBucket(
|
|
IN ULONG Index,
|
|
IN WORD FlushLevel
|
|
);
|
|
|
|
//
|
|
// Cache Implementation
|
|
//
|
|
// Cache is implemented as a hash on name, with chaining in the individual
|
|
// buckets. Individual name entries are blocks with name pointer and array
|
|
// of up to 3 RR set pointers. The new names\entries are put at the front of
|
|
// the bucket chain, so the oldest are at the rear.
|
|
//
|
|
//
|
|
// Cleanup:
|
|
//
|
|
// The cleanup strategy is to time out all RR sets and cleanup everything
|
|
// possible as a result. Then entries beyond a max bucket size (a resizable
|
|
// global) are deleted, the oldest queries deleted first.
|
|
//
|
|
// Ideally, we'd like to keep the most useful entries in the cache while
|
|
// being able to limit the overall cache size.
|
|
//
|
|
// A few observations:
|
|
//
|
|
// 1) max bucket size is worthless; if sufficient for pruning, it would be
|
|
// too small to allow non-uniform distributions
|
|
//
|
|
// 2) LRU should be required; on busy cache shouldn't prune something queried
|
|
// "a while" ago that is being used all the time; that adds much more traffic
|
|
// than something recently queried but then unused;
|
|
//
|
|
// 3) if necessary an LRU index could be kept; but probably some time bucket
|
|
// counting to know how "deep" pruning must be is adequate
|
|
//
|
|
//
|
|
// Memory:
|
|
//
|
|
// Currently hash itself and hash entries come from private resolver heap.
|
|
// However, RR sets are built by record parsing of messages received in dnsapi.dll
|
|
// and hence are built by the default dnsapi.dll allocator. We must match it.
|
|
//
|
|
// The downside of this is twofold:
|
|
// 1) By being in process heap, we are exposed (debug wise) to any poor code
|
|
// in services.exe. Hopefully, there are getting better, but anything that
|
|
// trashes memory is likely to cause us to have to debug, because we are the highest
|
|
// use service.
|
|
// 2) Flush \ cleanup is easy. Just kill the heap.
|
|
//
|
|
// There are several choices:
|
|
//
|
|
// 0) Copy the records. We are still susceptible to memory corruption ... but the
|
|
// interval is shorter, since we don't keep anything in process heap.
|
|
//
|
|
// 1) Query can directly call dnslib.lib query routines. Since dnslib.lib is
|
|
// explicitly compiled in, it's global for holding the allocators is modules, rather
|
|
// than process specific.
|
|
//
|
|
// 2) Add some parameter to query routines that allows pass of allocator down to
|
|
// lowest level. At high level this is straightforward. At lower level it maybe
|
|
// problematic. There may be a way to do it with a flag where the allocator is
|
|
// "optional" and used only when a flag is set.
|
|
//
|
|
|
|
|
|
|
|
|
|
//
|
|
// Cache functions
|
|
//
|
|
|
|
DNS_STATUS
|
|
Cache_Lock(
|
|
IN BOOL fNoStart
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Lock the cache
|
|
|
|
Arguments:
|
|
|
|
None.
|
|
|
|
Return Value:
|
|
|
|
NO_ERROR if successful -- cache is locked.
|
|
ErrorCode on init if cache init failed.
|
|
|
|
--*/
|
|
{
|
|
DNSDBG( LOCK, ( "Enter Cache_Lock() ..." ));
|
|
|
|
EnterCriticalSection( &CacheCS );
|
|
|
|
DNSDBG( LOCK, (
|
|
"through lock (r=%d)\n",
|
|
CacheCS.RecursionCount ));
|
|
|
|
// update global time (for TTL set and timeout)
|
|
//
|
|
// this allows us to eliminate multiple time calls
|
|
// within cache
|
|
|
|
g_CurrentCacheTime = Dns_GetCurrentTimeInSeconds();
|
|
|
|
//
|
|
// if cache not loaded -- load
|
|
// this allows us to avoid load on every PnP until we
|
|
// are actually queried
|
|
//
|
|
|
|
if ( !fNoStart && !g_HashTable )
|
|
{
|
|
DNS_STATUS status;
|
|
|
|
DNSDBG( ANY, (
|
|
"No hash table when took lock -- initializing!\n" ));
|
|
|
|
status = Cache_Initialize();
|
|
if ( status != NO_ERROR )
|
|
{
|
|
Cache_Unlock();
|
|
return status;
|
|
}
|
|
}
|
|
|
|
return NO_ERROR;
|
|
}
|
|
|
|
|
|
VOID
|
|
Cache_Unlock(
|
|
VOID
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Unlock the cache
|
|
|
|
Arguments:
|
|
|
|
None.
|
|
|
|
Return Value:
|
|
|
|
None.
|
|
|
|
--*/
|
|
{
|
|
DNSDBG( LOCK, (
|
|
"Cache_Unlock() r=%d\n",
|
|
CacheCS.RecursionCount ));
|
|
|
|
LeaveCriticalSection( &CacheCS );
|
|
}
|
|
|
|
|
|
|
|
DNS_STATUS
|
|
Cache_Initialize(
|
|
VOID
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Initialize the cache.
|
|
Create events and locks and setup basic hash.
|
|
|
|
Arguments:
|
|
|
|
None.
|
|
|
|
Return Value:
|
|
|
|
ERROR_SUCCESS if successful.
|
|
ErrorCode on failure.
|
|
|
|
--*/
|
|
{
|
|
DNS_STATUS status;
|
|
DWORD carryCount;
|
|
|
|
DNSDBG( INIT, ( "Cache_Initialize()\n" ));
|
|
|
|
//
|
|
// lock -- with "no-start" set to avoid recursion
|
|
//
|
|
|
|
LOCK_CACHE_NO_START();
|
|
|
|
//
|
|
// create cache heap
|
|
//
|
|
// want to have own heap
|
|
// 1) to simplify flush\shutdown
|
|
// 2) keep us from "entanglements" with poor services
|
|
//
|
|
|
|
g_CacheHeap = HeapCreate( 0, INITIAL_CACHE_HEAP_SIZE, 0 );
|
|
if ( !g_CacheHeap )
|
|
{
|
|
status = ERROR_NOT_ENOUGH_MEMORY;
|
|
g_HashTable = NULL;
|
|
goto Done;
|
|
}
|
|
|
|
g_HashTable = CACHE_HEAP_ALLOC_ZERO(
|
|
sizeof(PCACHE_ENTRY) * g_HashTableSize );
|
|
if ( !g_HashTable )
|
|
{
|
|
status = ERROR_NOT_ENOUGH_MEMORY;
|
|
HeapDestroy( g_CacheHeap );
|
|
g_CacheHeap = NULL;
|
|
goto Done;
|
|
}
|
|
|
|
g_WakeFlag = FALSE;
|
|
|
|
g_EntryCount = 0;
|
|
g_EntryAlloc = 0;
|
|
g_EntryFree = 0;
|
|
|
|
g_RecordSetCount = 0;
|
|
g_RecordSetCache = 0;
|
|
g_RecordSetFree = 0;
|
|
|
|
// eliminate cache size checks during hosts file load
|
|
|
|
g_RecordSetCountLimit = MAXDWORD;
|
|
g_RecordSetCountThreshold = MAXDWORD;
|
|
|
|
//
|
|
// load hosts file into cache
|
|
//
|
|
|
|
g_fLoadingHostsFile = TRUE;
|
|
InitCacheWithHostFile();
|
|
g_fLoadingHostsFile = FALSE;
|
|
|
|
//
|
|
// set cache size limit
|
|
// - above what loaded from hosts file
|
|
// - always allow some dynamic space regardless of
|
|
// g_MaxCacheSize
|
|
// - create slightly higher threshold value for kicking
|
|
// off cleanup so cleanup not running all the time
|
|
//
|
|
|
|
carryCount = g_MaxCacheSize;
|
|
if ( carryCount < MIN_DYNAMIC_RECORD_COUNT )
|
|
{
|
|
carryCount = MIN_DYNAMIC_RECORD_COUNT;
|
|
}
|
|
g_RecordSetCountLimit = g_RecordSetCount + carryCount;
|
|
g_RecordSetCountThreshold = g_RecordSetCountLimit + CLEANUP_RECORD_COUNT_BAND;
|
|
|
|
status = NO_ERROR;
|
|
|
|
Done:
|
|
|
|
UNLOCK_CACHE();
|
|
|
|
return status;
|
|
}
|
|
|
|
|
|
|
|
DNS_STATUS
|
|
Cache_Shutdown(
|
|
VOID
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Shutdown the cache.
|
|
|
|
Arguments:
|
|
|
|
None.
|
|
|
|
Return Value:
|
|
|
|
ERROR_SUCCESS if successful.
|
|
ErrorCode on failure.
|
|
|
|
--*/
|
|
{
|
|
DNSDBG( INIT, ( "Cache_Shutdown()\n" ));
|
|
|
|
//
|
|
// clean out cache and delete cache heap
|
|
// - currently Cache_Flush() does just this
|
|
//
|
|
|
|
return Cache_Flush();
|
|
}
|
|
|
|
|
|
|
|
DNS_STATUS
|
|
Cache_Flush(
|
|
VOID
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Flush the cache.
|
|
|
|
This flushes all the cache data and rereads host file but does NOT
|
|
shut down and restart cache threads (host file monitor or multicast).
|
|
|
|
Arguments:
|
|
|
|
None
|
|
|
|
Return Value:
|
|
|
|
ERROR_SUCCESS if successful.
|
|
ErrorCode on rebuild failure.
|
|
|
|
--*/
|
|
{
|
|
DWORD status = ERROR_SUCCESS;
|
|
WORD ihash;
|
|
WORD RecordIter;
|
|
|
|
DNSDBG( ANY, ( "\nCache_Flush()\n" ));
|
|
|
|
//
|
|
// wake\stop garbage collection
|
|
//
|
|
|
|
g_WakeFlag = TRUE;
|
|
|
|
//
|
|
// lock with "no start" flag
|
|
// - avoids creating cache structs if they don't exist
|
|
//
|
|
|
|
LOCK_CACHE_NO_START();
|
|
|
|
DNSLOG_F1( "Flushing DNS Cache" );
|
|
DNSLOG_F3(
|
|
" Before Cache_Flush(): entries %d, record %d",
|
|
g_EntryCount,
|
|
g_RecordSetCount );
|
|
|
|
//
|
|
// clear entries in each hash bucket
|
|
//
|
|
|
|
if ( g_HashTable )
|
|
{
|
|
for ( ihash = 0;
|
|
ihash < g_HashTableSize;
|
|
ihash++ )
|
|
{
|
|
Cache_FlushBucket(
|
|
ihash,
|
|
FLUSH_LEVEL_CLEANUP );
|
|
}
|
|
}
|
|
|
|
DNSDBG( CACHE, (
|
|
"After flushing cache:\n"
|
|
"\trecord count = %d\n"
|
|
"\tentry count = %d\n",
|
|
g_RecordSetCount,
|
|
g_EntryCount ));
|
|
|
|
DNSLOG_F3(
|
|
" After Cache_Flush() flush: entries %d, record %d",
|
|
g_EntryCount,
|
|
g_RecordSetCount );
|
|
|
|
//DNS_ASSERT( g_RecordSetCount == 0 );
|
|
//DNS_ASSERT( g_EntryCount == 0 );
|
|
|
|
g_RecordSetCount = 0;
|
|
g_EntryCount = 0;
|
|
|
|
//
|
|
// Note: can NOT delete the cache without stopping mcast
|
|
// thread which currently uses cache heap
|
|
|
|
//
|
|
// DCR: have all data in cache in single heap
|
|
// - protected
|
|
// - single destroy cleans up
|
|
|
|
// once cleaned up, delete heap
|
|
|
|
if ( g_CacheHeap )
|
|
{
|
|
HeapDestroy( g_CacheHeap );
|
|
g_CacheHeap = NULL;
|
|
}
|
|
g_HashTable = NULL;
|
|
|
|
//
|
|
// dump local IP list
|
|
// - not dumping on shutdown as the IP cleanup happens
|
|
// first and takes away the CS;
|
|
//
|
|
// note to reviewer:
|
|
// this is equivalent to the previous behavior where
|
|
// Cache_Flush() FALSE was shutdown and
|
|
// everything else used TRUE (for restart) which did a
|
|
// RefreshLocalAddrArray() to rebuild IP list
|
|
// now we simply dump the IP list rather than rebuilding
|
|
//
|
|
|
|
if ( !g_StopFlag )
|
|
{
|
|
// FIX6: no longer keep separate addr array separate from netinfo
|
|
//ClearLocalAddrArray();
|
|
}
|
|
|
|
DNSDBG( ANY, ( "Leave Cache_Flush()\n\n" ));
|
|
|
|
UNLOCK_CACHE();
|
|
|
|
return( status );
|
|
}
|
|
|
|
|
|
|
|
//
|
|
// Cache utilities
|
|
//
|
|
|
|
BOOL
|
|
Cache_IsRecordTtlValid(
|
|
IN PDNS_RECORD pRecord
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Check if TTL is still valid (or has timed out).
|
|
|
|
Arguments:
|
|
|
|
pRecord -- record to check
|
|
|
|
Return Value:
|
|
|
|
TRUE -- if TTL is still valid
|
|
FALSE -- if TTL has timed out
|
|
|
|
--*/
|
|
{
|
|
//
|
|
// static or TTL not timed out => valid
|
|
//
|
|
// note: currently flushing all records on PnP, but this is
|
|
// not strickly necessary; if stop this then MUST change
|
|
// this to whack negative cache entries that are older
|
|
// than last PnP time
|
|
//
|
|
|
|
if ( IS_STATIC_RR(pRecord) )
|
|
{
|
|
return( TRUE );
|
|
}
|
|
else
|
|
{
|
|
return( (LONG)(pRecord->dwTtl - g_CurrentCacheTime) > 0 );
|
|
}
|
|
}
|
|
|
|
|
|
|
|
//
|
|
// Cache entry routines
|
|
//
|
|
|
|
DWORD
|
|
getHashIndex(
|
|
IN PWSTR pName,
|
|
IN DWORD NameLength OPTIONAL
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Create cannonical cache form of name.
|
|
|
|
Note: no test for adequacy of buffer is done.
|
|
|
|
Arguments:
|
|
|
|
pName -- name
|
|
|
|
NameLength -- NameLength, OPTIONAL
|
|
|
|
Return Value:
|
|
|
|
None
|
|
|
|
--*/
|
|
{
|
|
register PWCHAR pstring;
|
|
register WCHAR wch;
|
|
register DWORD hash = 0;
|
|
|
|
//
|
|
// build hash by XORing characters
|
|
//
|
|
|
|
pstring = pName;
|
|
|
|
while ( wch = *pstring++ )
|
|
{
|
|
hash <<= 1;
|
|
hash ^= wch;
|
|
}
|
|
|
|
//
|
|
// mod over hash table size
|
|
//
|
|
|
|
return( hash % g_HashTableSize );
|
|
}
|
|
|
|
|
|
|
|
BOOL
|
|
makeCannonicalCacheName(
|
|
OUT PWCHAR pNameBuffer,
|
|
IN DWORD BufferLength,
|
|
IN PWSTR pName,
|
|
IN DWORD NameLength OPTIONAL
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Create cannonical cache form of name.
|
|
|
|
Arguments:
|
|
|
|
pNameBuffer -- buffer to hold cache name
|
|
|
|
BufferLength -- length of buffer
|
|
|
|
pName -- ptr to name string
|
|
|
|
NameLength -- optional, saves wsclen() call if known
|
|
|
|
Return Value:
|
|
|
|
TRUE if successful.
|
|
FALSE on bogus name.
|
|
|
|
--*/
|
|
{
|
|
INT count;
|
|
|
|
DNSDBG( TRACE, (
|
|
"makeCannonicalCacheName( %S )\n",
|
|
pName ));
|
|
|
|
//
|
|
// get length if not specified
|
|
//
|
|
|
|
if ( NameLength == 0 )
|
|
{
|
|
NameLength = wcslen( pName );
|
|
}
|
|
|
|
//
|
|
// copy and downcase string
|
|
// - "empty" buffer for prefix happiness
|
|
//
|
|
|
|
*pNameBuffer = (WCHAR) 0;
|
|
|
|
count = Dns_MakeCanonicalNameW(
|
|
pNameBuffer,
|
|
BufferLength,
|
|
pName,
|
|
NameLength+1 // convert null terminator
|
|
);
|
|
if ( count == 0 )
|
|
{
|
|
ASSERT( GetLastError() == ERROR_INSUFFICIENT_BUFFER );
|
|
return( FALSE );
|
|
}
|
|
|
|
ASSERT( count == (INT)NameLength+1 );
|
|
|
|
//
|
|
// whack any trailing dot
|
|
// - except for root node
|
|
//
|
|
|
|
count--; // account for null terminator
|
|
DNS_ASSERT( count == NameLength );
|
|
|
|
if ( count > 1 &&
|
|
pNameBuffer[count - 1] == L'.' )
|
|
{
|
|
pNameBuffer[count - 1] = 0;
|
|
}
|
|
|
|
return( TRUE );
|
|
}
|
|
|
|
|
|
|
|
PCACHE_ENTRY
|
|
Cache_CreateEntry(
|
|
IN PWSTR pName,
|
|
IN BOOL fCanonical
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Create cache entry, including allocation.
|
|
|
|
Arguments:
|
|
|
|
pName -- name
|
|
|
|
fCanonical -- TRUE if name already in cannonical form
|
|
|
|
Return Value:
|
|
|
|
Ptr to newly allocated cache entry.
|
|
NULL on error.
|
|
|
|
--*/
|
|
{
|
|
ULONG index = 0;
|
|
PCACHE_ENTRY pentry = NULL;
|
|
DWORD nameLength;
|
|
DWORD fixedLength;
|
|
PWCHAR pnameCache = NULL;
|
|
|
|
DNSDBG( TRACE, (
|
|
"Cache_CreateEntry( %S )\n",
|
|
pName ));
|
|
|
|
if ( !pName || !g_HashTable )
|
|
{
|
|
return NULL;
|
|
}
|
|
|
|
//
|
|
// alloc
|
|
//
|
|
|
|
nameLength = wcslen( pName );
|
|
|
|
fixedLength = sizeof(CACHE_ENTRY) +
|
|
(sizeof(PDNS_RECORD) * (CACHE_DEFAULT_SET_COUNT-1));
|
|
|
|
pentry = (PCACHE_ENTRY) CACHE_HEAP_ALLOC_ZERO(
|
|
fixedLength +
|
|
sizeof(WCHAR) * (nameLength+1) );
|
|
if ( !pentry )
|
|
{
|
|
goto Fail;
|
|
}
|
|
pentry->MaxCount = CACHE_DEFAULT_SET_COUNT;
|
|
|
|
pnameCache = (PWSTR) ((PBYTE)pentry + fixedLength);
|
|
|
|
//
|
|
// build the name
|
|
//
|
|
|
|
if ( fCanonical )
|
|
{
|
|
wcscpy( pnameCache, pName );
|
|
}
|
|
else
|
|
{
|
|
if ( !makeCannonicalCacheName(
|
|
pnameCache,
|
|
nameLength+1,
|
|
pName,
|
|
nameLength ) )
|
|
{
|
|
goto Fail;
|
|
}
|
|
}
|
|
pentry->pName = pnameCache;
|
|
|
|
//
|
|
// insert cache entry into cache -- first entry in bucket
|
|
//
|
|
|
|
index = getHashIndex( pnameCache, nameLength );
|
|
pentry->pNext = g_HashTable[ index ];
|
|
g_HashTable[ index ] = pentry;
|
|
g_EntryCount++;
|
|
g_EntryAlloc++;
|
|
|
|
//
|
|
// DCR: need overload detection
|
|
//
|
|
|
|
return pentry;
|
|
|
|
Fail:
|
|
|
|
// dump entry
|
|
|
|
if ( pentry )
|
|
{
|
|
CACHE_HEAP_FREE( pentry );
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
|
|
|
|
VOID
|
|
Cache_FreeEntry(
|
|
IN OUT PCACHE_ENTRY pEntry
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Free cache entry.
|
|
|
|
Arguments:
|
|
|
|
pEntry -- cache entry to free
|
|
|
|
Globals:
|
|
|
|
g_EntryCount -- decremented appropriately
|
|
|
|
g_NumberOfRecordsInCache -- decremented appropriately
|
|
|
|
Return Value:
|
|
|
|
None
|
|
|
|
--*/
|
|
{
|
|
INT iter;
|
|
|
|
DNSDBG( TRACE, (
|
|
"Cache_FreeEntry( %p )\n",
|
|
pEntry ));
|
|
|
|
//
|
|
// free entry
|
|
// - records
|
|
// - name
|
|
// - entry itself
|
|
//
|
|
|
|
if ( pEntry )
|
|
{
|
|
Cache_FlushEntryRecords(
|
|
pEntry,
|
|
FLUSH_LEVEL_CLEANUP,
|
|
0 );
|
|
|
|
#if 0
|
|
if ( pEntry->pNext )
|
|
{
|
|
DNSLOG_F1( "Cache_FreeEntry is deleting an entry that still points to other entries!" );
|
|
}
|
|
#endif
|
|
#if HEAPPROB
|
|
pEntry->pNext = DNS_BAD_PTR;
|
|
#endif
|
|
CACHE_HEAP_FREE( pEntry );
|
|
g_EntryFree--;
|
|
g_EntryCount--;
|
|
}
|
|
}
|
|
|
|
|
|
|
|
PCACHE_ENTRY
|
|
Cache_FindEntry(
|
|
IN PWSTR pName,
|
|
IN BOOL fCreate
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Find or create entry for name in cache.
|
|
|
|
Arguments:
|
|
|
|
pName -- name to find
|
|
|
|
fCreate -- TRUE to create if not found
|
|
|
|
Return Value:
|
|
|
|
Ptr to cache entry -- if successful.
|
|
NULL on failure.
|
|
|
|
--*/
|
|
{
|
|
ULONG index;
|
|
PCACHE_ENTRY pentry;
|
|
PCACHE_ENTRY pprevEntry = NULL;
|
|
WCHAR hashName[ DNS_MAX_NAME_BUFFER_LENGTH+4 ];
|
|
|
|
if ( !g_HashTable )
|
|
{
|
|
return NULL;
|
|
}
|
|
if ( !pName )
|
|
{
|
|
DNS_ASSERT( FALSE );
|
|
return NULL;
|
|
}
|
|
|
|
DNSDBG( TRACE, (
|
|
"Cache_FindEntry( %S, create=%d )\n",
|
|
pName,
|
|
fCreate ));
|
|
|
|
//
|
|
// build cache name
|
|
// - if invalid (too long) bail
|
|
//
|
|
|
|
if ( !makeCannonicalCacheName(
|
|
hashName,
|
|
DNS_MAX_NAME_BUFFER_LENGTH,
|
|
pName,
|
|
0 ) )
|
|
{
|
|
return NULL;
|
|
}
|
|
|
|
//
|
|
// find entry in cache
|
|
//
|
|
|
|
if ( LOCK_CACHE() != NO_ERROR )
|
|
{
|
|
return NULL;
|
|
}
|
|
|
|
index = getHashIndex( hashName, 0 );
|
|
|
|
pentry = g_HashTable[ index ];
|
|
|
|
DNSDBG( OFF, (
|
|
"in Cache_FindEntry\n"
|
|
"\tname = %S\n"
|
|
"\tindex = %d\n"
|
|
"\tpentry = %p\n",
|
|
hashName,
|
|
index,
|
|
pentry ));
|
|
|
|
while( pentry )
|
|
{
|
|
if ( DnsNameCompare_W( hashName, pentry->pName ) )
|
|
{
|
|
//
|
|
// found entry
|
|
// - move to front, if not already there
|
|
|
|
if ( pprevEntry )
|
|
{
|
|
pprevEntry->pNext = pentry->pNext;
|
|
pentry->pNext = g_HashTable[ index ];
|
|
g_HashTable[ index ] = pentry;
|
|
}
|
|
break;
|
|
}
|
|
ELSE
|
|
{
|
|
DNSDBG( OFF, (
|
|
"in Cache_FindEntry -- failed name compare\n"
|
|
"\tout name = %S\n"
|
|
"\tpentry = %p\n"
|
|
"\tname = %S\n",
|
|
hashName,
|
|
pentry,
|
|
pentry->pName ));
|
|
}
|
|
|
|
pprevEntry = pentry;
|
|
pentry = pentry->pNext;
|
|
}
|
|
|
|
//
|
|
// if not found -- create?
|
|
//
|
|
// DCR: optimize for create
|
|
//
|
|
|
|
if ( !pentry && fCreate )
|
|
{
|
|
pentry = Cache_CreateEntry(
|
|
hashName,
|
|
TRUE // name already canonical
|
|
);
|
|
}
|
|
|
|
DNS_ASSERT( !pentry || g_HashTable[ index ] == pentry );
|
|
UNLOCK_CACHE();
|
|
|
|
DNSDBG( TRACE, (
|
|
"Leave Cache_FindEntry\n"
|
|
"\tname = %S\n"
|
|
"\tindex = %d\n"
|
|
"\tpentry = %p\n",
|
|
hashName,
|
|
index,
|
|
pentry ));
|
|
|
|
return pentry;
|
|
}
|
|
|
|
|
|
|
|
PDNS_RECORD
|
|
Cache_FindEntryRecords(
|
|
OUT PDNS_RECORD ** pppRRList,
|
|
IN PCACHE_ENTRY pEntry,
|
|
IN WORD wType
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Find entry in cache.
|
|
|
|
Arguments:
|
|
|
|
pppRRList -- addr to recv addr of entry's ptr to RR list
|
|
|
|
pEntry -- cache entry to check
|
|
|
|
Type -- record type to find
|
|
|
|
Return Value:
|
|
|
|
Ptr to record set of desired type -- if found.
|
|
NULL if not found.
|
|
|
|
--*/
|
|
{
|
|
WORD iter;
|
|
PDNS_RECORD prr;
|
|
PDNS_RECORD * prrAddr = NULL;
|
|
|
|
DNSDBG( TRACE, (
|
|
"Cache_FindEntryRecords( %p, e=%p, type=%d )\n",
|
|
pppRRList,
|
|
pEntry,
|
|
wType ));
|
|
|
|
//
|
|
// check all the records at the cache entry
|
|
//
|
|
|
|
for ( iter = 0;
|
|
iter < pEntry->MaxCount;
|
|
iter++ )
|
|
{
|
|
prrAddr = &pEntry->Records[iter];
|
|
prr = *prrAddr;
|
|
|
|
if ( !prr )
|
|
{
|
|
continue;
|
|
}
|
|
if ( !Cache_IsRecordTtlValid( prr ) )
|
|
{
|
|
DNSDBG( TRACE, (
|
|
"Whacking timed out record %p at cache entry %p\n",
|
|
prr,
|
|
pEntry ));
|
|
Dns_RecordListFree( prr );
|
|
pEntry->Records[iter] = NULL;
|
|
g_RecordSetCount--;
|
|
g_RecordSetFree--;
|
|
continue;
|
|
}
|
|
|
|
//
|
|
// find matching type
|
|
// - direct type match
|
|
// - NAME_ERROR
|
|
//
|
|
|
|
if ( prr->wType == wType ||
|
|
( prr->wType == DNS_TYPE_ANY &&
|
|
prr->wDataLength == 0 ) )
|
|
{
|
|
goto Done;
|
|
}
|
|
|
|
//
|
|
// CNAME match
|
|
// - walk list and determine if for matching type
|
|
|
|
if ( prr->wType == DNS_TYPE_CNAME &&
|
|
wType != DNS_TYPE_CNAME )
|
|
{
|
|
PDNS_RECORD prrChain = prr->pNext;
|
|
|
|
while ( prrChain )
|
|
{
|
|
if ( prrChain->wType == wType )
|
|
{
|
|
// chain to desired type -- take RR set
|
|
goto Done;
|
|
}
|
|
prrChain = prrChain->pNext;
|
|
}
|
|
}
|
|
|
|
// records for another type -- continue
|
|
}
|
|
|
|
// type not found
|
|
|
|
prr = NULL;
|
|
|
|
Done:
|
|
|
|
if ( pppRRList )
|
|
{
|
|
*pppRRList = prrAddr;
|
|
}
|
|
|
|
DNSDBG( TRACE, (
|
|
"Leave Cache_FindEntryRecords => %p\n",
|
|
prr ));
|
|
|
|
return prr;
|
|
}
|
|
|
|
|
|
|
|
BOOL
|
|
Cache_FlushEntryRecords(
|
|
IN OUT PCACHE_ENTRY pEntry,
|
|
IN DWORD Level,
|
|
IN WORD wType
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Free cache entry.
|
|
|
|
Arguments:
|
|
|
|
pEntry -- cache entry to flush
|
|
|
|
FlushLevel -- flush level
|
|
FLUSH_LEVEL_NORMAL -- flush matching type, invalid, NAME_ERROR
|
|
FLUSH_LEVEL_WIRE -- to flush all wire data, but leave hosts and cluster
|
|
FLUSH_LEVEL_INVALID -- flush only invalid records
|
|
FLUSH_LEVEL_STRONG -- to flush all but hosts file
|
|
FLUSH_LEVEL_CLEANUP -- to flush all records for full cache flush
|
|
|
|
wType -- flush type for levels with type
|
|
DNS type -- to flush specifically this type
|
|
|
|
Globals:
|
|
|
|
g_EntryCount -- decremented appropriately
|
|
|
|
g_NumberOfRecordsInCache -- decremented appropriately
|
|
|
|
Return Value:
|
|
|
|
TRUE if entry flushed completely.
|
|
FALSE if records left.
|
|
|
|
--*/
|
|
{
|
|
INT iter;
|
|
BOOL recordsLeft = FALSE;
|
|
|
|
DNSDBG( TRACE, (
|
|
"Cache_FlushEntryRecords( %p, %08x, %d )\n",
|
|
pEntry,
|
|
Level,
|
|
wType ));
|
|
|
|
//
|
|
// loop through records sets -- flush where appropriate
|
|
//
|
|
// CLEANUP flush
|
|
// - everything
|
|
//
|
|
// STRONG (user initiated) flush
|
|
// - all cached records, including cluster
|
|
// but hostsfile saved
|
|
//
|
|
// WIRE flush
|
|
// - all wire cached records
|
|
// hosts file AND cluster saved
|
|
//
|
|
// INVALID flush
|
|
// - timedout only
|
|
//
|
|
// NORMAL flush (regular flush done on caching)
|
|
// - timed out records
|
|
// - records of desired type
|
|
// - NAME_ERROR
|
|
//
|
|
|
|
for ( iter = 0;
|
|
iter < (INT)pEntry->MaxCount;
|
|
iter++ )
|
|
{
|
|
PDNS_RECORD prr = pEntry->Records[iter];
|
|
BOOL flush;
|
|
|
|
if ( !prr )
|
|
{
|
|
continue;
|
|
}
|
|
|
|
//
|
|
// switch on flush type
|
|
// yes there are optimizations, but this is simple
|
|
//
|
|
|
|
if ( Level == FLUSH_LEVEL_NORMAL )
|
|
{
|
|
flush = ( !IS_STATIC_RR(prr)
|
|
&&
|
|
( prr->wType == wType ||
|
|
( prr->wType == DNS_TYPE_ANY &&
|
|
prr->wDataLength == 0 ) ) );
|
|
}
|
|
else if ( Level == FLUSH_LEVEL_WIRE )
|
|
{
|
|
flush = !IS_STATIC_RR(prr);
|
|
}
|
|
else if ( Level == FLUSH_LEVEL_INVALID )
|
|
{
|
|
flush = !Cache_IsRecordTtlValid(prr);
|
|
}
|
|
else if ( Level == FLUSH_LEVEL_CLEANUP )
|
|
{
|
|
flush = TRUE;
|
|
}
|
|
else
|
|
{
|
|
DNS_ASSERT( Level == FLUSH_LEVEL_STRONG );
|
|
flush = !IS_HOSTS_FILE_RR(prr);
|
|
}
|
|
|
|
if ( flush )
|
|
{
|
|
pEntry->Records[iter] = NULL;
|
|
Dns_RecordListFree( prr );
|
|
g_RecordSetCount--;
|
|
g_RecordSetFree--;
|
|
}
|
|
else
|
|
{
|
|
recordsLeft = TRUE;
|
|
}
|
|
}
|
|
|
|
return !recordsLeft;
|
|
}
|
|
|
|
|
|
|
|
VOID
|
|
Cache_FlushBucket(
|
|
IN ULONG Index,
|
|
IN WORD FlushLevel
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Cleanup cache bucket.
|
|
|
|
Arguments:
|
|
|
|
Index -- Index of hash bucket to trim.
|
|
|
|
FlushLevel -- level of flush desired
|
|
see Cache_FlushEntryRecords() for description of
|
|
flush levels
|
|
|
|
Return Value:
|
|
|
|
None
|
|
|
|
--*/
|
|
{
|
|
PCACHE_ENTRY pentry;
|
|
PCACHE_ENTRY pprev;
|
|
INT countCompleted;
|
|
|
|
DNSDBG( CACHE, (
|
|
"Cache_FlushBucket( %d, %08x )\n",
|
|
Index,
|
|
FlushLevel ));
|
|
|
|
//
|
|
// flush entries in this bucket
|
|
//
|
|
// note: using hack here that hash table pointer can
|
|
// be treated as cache entry for purposes of accessing
|
|
// it's next pointer (since it's the first field in
|
|
// a CACHE_ENTRY)
|
|
// if this changes, must explicitly fix up "first entry"
|
|
// case or move to double-linked list that can free
|
|
// empty penty without regard to it's location
|
|
//
|
|
|
|
if ( !g_HashTable )
|
|
{
|
|
return;
|
|
}
|
|
|
|
//
|
|
// flush entries
|
|
//
|
|
// avoid holding lock too long by handling no more then
|
|
// fifty entries at a time
|
|
// note: generally 50 entries will cover entire bucket but
|
|
// can still be completed in reasonable time;
|
|
//
|
|
// DCR: smarter flush -- avoid lock\unlock
|
|
// peer into CS and don't unlock when no one waiting
|
|
// if waiting unlock and give up timeslice
|
|
// DCR: some LRU flush for garbage collection
|
|
//
|
|
|
|
countCompleted = 0;
|
|
|
|
while ( 1 )
|
|
{
|
|
INT count = 0;
|
|
INT countStop = countCompleted + 50;
|
|
|
|
LOCK_CACHE_NO_START();
|
|
if ( !g_HashTable )
|
|
{
|
|
UNLOCK_CACHE();
|
|
break;
|
|
}
|
|
|
|
DNSDBG( CACHE, (
|
|
"locked for bucket flush -- completed=%d, stop=%d\n",
|
|
count,
|
|
countStop ));
|
|
|
|
pprev = (PCACHE_ENTRY) &g_HashTable[ Index ];
|
|
|
|
while ( pentry = pprev->pNext )
|
|
{
|
|
// bypass any previously checked entries
|
|
|
|
if ( count++ < countCompleted )
|
|
{
|
|
pprev = pentry;
|
|
continue;
|
|
}
|
|
if ( count > countStop )
|
|
{
|
|
break;
|
|
}
|
|
|
|
// flush -- if successful cut from list and
|
|
// drop counts so countCompleted used in bypass
|
|
// will be correct and won't skip anyone
|
|
|
|
if ( Cache_FlushEntryRecords(
|
|
pentry,
|
|
FlushLevel,
|
|
0 ) )
|
|
{
|
|
pprev->pNext = pentry->pNext;
|
|
Cache_FreeEntry( pentry );
|
|
count--;
|
|
countStop--;
|
|
continue;
|
|
}
|
|
pprev = pentry;
|
|
}
|
|
|
|
UNLOCK_CACHE();
|
|
countCompleted = count;
|
|
|
|
// stop when
|
|
// - cleared all the entries in the bucket
|
|
// - shutdown, except exempt the shutdown flush itself
|
|
|
|
if ( !pentry ||
|
|
(g_StopFlag && FlushLevel != FLUSH_LEVEL_CLEANUP) )
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
|
|
DNSDBG( CACHE, (
|
|
"Leave Cache_FlushBucket( %d, %08x )\n"
|
|
"\trecord count = %d\n"
|
|
"\tentry count = %d\n",
|
|
Index,
|
|
FlushLevel,
|
|
g_RecordSetCount,
|
|
g_EntryCount ));
|
|
}
|
|
|
|
|
|
|
|
//
|
|
// Cache interface routines
|
|
//
|
|
|
|
VOID
|
|
Cache_PrepareRecordList(
|
|
IN OUT PDNS_RECORD pRecordList
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Prepare record list for cache.
|
|
|
|
Arguments:
|
|
|
|
pRecordList - record list to put in cache
|
|
|
|
Return Value:
|
|
|
|
Ptr to screened, prepared record list.
|
|
|
|
--*/
|
|
{
|
|
PDNS_RECORD prr = pRecordList;
|
|
PDNS_RECORD pnext;
|
|
DWORD ttl;
|
|
DWORD maxTtl;
|
|
|
|
DNSDBG( TRACE, (
|
|
"Cache_PrepareRecordList( rr=%p )\n",
|
|
prr ));
|
|
|
|
if ( !prr )
|
|
{
|
|
return;
|
|
}
|
|
|
|
//
|
|
// static (currently host file) TTL records
|
|
//
|
|
// currently no action required -- records come one
|
|
// at a time and no capability to even to the pName=NULL
|
|
// step
|
|
//
|
|
|
|
if ( IS_STATIC_RR(prr) )
|
|
{
|
|
return;
|
|
}
|
|
|
|
//
|
|
// wire records get relative TTL
|
|
// - compute minimum TTL for set
|
|
// - save TTL as timeout (offset by TTL from current time)
|
|
//
|
|
// DCR: TTL still not per set
|
|
// - but this is at least better than Win2K where
|
|
// multiple sets and did NOT find minimum
|
|
//
|
|
|
|
maxTtl = g_MaxCacheTtl;
|
|
if ( prr->wType == DNS_TYPE_SOA )
|
|
{
|
|
maxTtl = g_MaxSOACacheEntryTtlLimit;
|
|
}
|
|
|
|
//
|
|
// get caching TTL
|
|
// - minimum TTL in set
|
|
// - offset from current time
|
|
|
|
ttl = Dns_RecordListGetMinimumTtl( prr );
|
|
if ( ttl > maxTtl )
|
|
{
|
|
ttl = maxTtl;
|
|
}
|
|
|
|
ttl += g_CurrentCacheTime;
|
|
|
|
#if 0
|
|
// screening done at higher level now
|
|
//
|
|
// screen records
|
|
// - no non-RPCable types
|
|
// - no Authority records
|
|
//
|
|
|
|
if ( prr->wType != 0 )
|
|
{
|
|
prr = Dns_RecordListScreen(
|
|
prr,
|
|
SCREEN_OUT_AUTHORITY | SCREEN_OUT_NON_RPC );
|
|
|
|
DNS_ASSERT( prr );
|
|
}
|
|
#endif
|
|
|
|
//
|
|
// set timeout on all records in set
|
|
//
|
|
// note: FreeOwner handling depends on leading record
|
|
// in having owner name set, otherwise this produces
|
|
// bogus name owner fields
|
|
//
|
|
// DCR: set record list TTL function in dnslib
|
|
//
|
|
|
|
pnext = prr;
|
|
|
|
while ( pnext )
|
|
{
|
|
pnext->dwTtl = ttl;
|
|
|
|
if ( !FLAG_FreeOwner( pnext ) )
|
|
{
|
|
pnext->pName = NULL;
|
|
}
|
|
pnext = pnext->pNext;
|
|
}
|
|
}
|
|
|
|
|
|
|
|
VOID
|
|
Cache_RestoreRecordListForRpc(
|
|
IN OUT PDNS_RECORD pRecordList
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Restore cache record list for RPC.
|
|
|
|
Arguments:
|
|
|
|
pRecordList - record list to put in cache
|
|
|
|
Return Value:
|
|
|
|
None
|
|
|
|
--*/
|
|
{
|
|
PDNS_RECORD prr = pRecordList;
|
|
DWORD currentTime;
|
|
|
|
DNSDBG( TRACE, (
|
|
"Cache_RestoreRecordListForRpc( rr=%p )\n",
|
|
prr ));
|
|
|
|
if ( !prr )
|
|
{
|
|
DNS_ASSERT( FALSE );
|
|
return;
|
|
}
|
|
|
|
//
|
|
// static TTL records need no action
|
|
//
|
|
|
|
if ( IS_STATIC_RR(prr) )
|
|
{
|
|
return;
|
|
}
|
|
|
|
//
|
|
// turn timeouts back into TTLs
|
|
//
|
|
|
|
currentTime = g_CurrentCacheTime;
|
|
|
|
while ( prr )
|
|
{
|
|
DWORD ttl = prr->dwTtl - currentTime;
|
|
|
|
if ( (LONG)ttl < 0 )
|
|
{
|
|
ttl = 0;
|
|
}
|
|
prr->dwTtl = ttl;
|
|
prr = prr->pNext;
|
|
}
|
|
}
|
|
|
|
|
|
|
|
VOID
|
|
Cache_RecordSetAtomic(
|
|
IN PWSTR pwsName,
|
|
IN WORD wType,
|
|
IN PDNS_RECORD pRecordSet
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Cache record set atomically at entry.
|
|
|
|
Cache_RecordList() handles breakup of record list
|
|
and appropriate placing of records. This does caching
|
|
of single blob at particular location.
|
|
|
|
Arguments:
|
|
|
|
pRecordSet -- record list to add
|
|
|
|
Globals:
|
|
|
|
g_EntryCount -- decremented appropriately
|
|
|
|
g_NumberOfRecordsInCache -- decremented appropriately
|
|
|
|
Return Value:
|
|
|
|
None
|
|
|
|
--*/
|
|
{
|
|
INT iter;
|
|
WORD wtype;
|
|
PWSTR pname;
|
|
BOOL fstatic;
|
|
PCACHE_ENTRY pentry;
|
|
BOOL fretry;
|
|
WORD flushLevel;
|
|
|
|
|
|
DNSDBG( TRACE, (
|
|
"Cache_RecordSetAtomic( %S, type=%d, rr=%p )\n",
|
|
pwsName,
|
|
wType,
|
|
pRecordSet ));
|
|
|
|
if ( !pRecordSet )
|
|
{
|
|
return;
|
|
}
|
|
fstatic = IS_STATIC_RR(pRecordSet);
|
|
|
|
DNS_ASSERT( !fstatic ||
|
|
pRecordSet->pNext == NULL ||
|
|
(pRecordSet->wType==DNS_TYPE_CNAME) )
|
|
|
|
//
|
|
// determine caching type
|
|
// - specified OR from records
|
|
// CNAMEs will be at the head of a lookup from another type
|
|
//
|
|
|
|
wtype = wType;
|
|
if ( !wtype )
|
|
{
|
|
wtype = pRecordSet->wType;
|
|
}
|
|
|
|
//
|
|
// if name specified use it, otherwise use from records
|
|
//
|
|
|
|
pname = pwsName;
|
|
if ( !pname )
|
|
{
|
|
pname = pRecordSet->pName;
|
|
}
|
|
|
|
//
|
|
// prepare RR set for cache
|
|
//
|
|
|
|
Cache_PrepareRecordList( pRecordSet );
|
|
|
|
//
|
|
// find\create cache entry and cache
|
|
//
|
|
|
|
if ( LOCK_CACHE() != NO_ERROR )
|
|
{
|
|
LOCK_CACHE_NO_START();
|
|
goto Failed;
|
|
}
|
|
|
|
pentry = Cache_FindEntry(
|
|
pname,
|
|
TRUE // create
|
|
);
|
|
if ( !pentry )
|
|
{
|
|
goto Failed;
|
|
}
|
|
|
|
//
|
|
// clean up existing records at node
|
|
// - remove stale records
|
|
// - remove records of same type
|
|
// - if NAME_ERROR caching remove everything
|
|
// from wire
|
|
//
|
|
|
|
flushLevel = FLUSH_LEVEL_NORMAL;
|
|
|
|
if ( wtype == DNS_TYPE_ALL &&
|
|
pRecordSet->wDataLength == 0 )
|
|
{
|
|
flushLevel = FLUSH_LEVEL_WIRE;
|
|
}
|
|
|
|
Cache_FlushEntryRecords(
|
|
pentry,
|
|
flushLevel,
|
|
wtype );
|
|
|
|
//
|
|
// check for matching record type still there
|
|
//
|
|
|
|
for ( iter = 0;
|
|
iter < (INT)pentry->MaxCount;
|
|
iter++ )
|
|
{
|
|
PDNS_RECORD prrExist = pentry->Records[iter];
|
|
|
|
if ( !prrExist ||
|
|
prrExist->wType != wtype )
|
|
{
|
|
continue;
|
|
}
|
|
|
|
// matching type still there after flush
|
|
// - if trying to cache wire set at hostfile entry, fail
|
|
|
|
DNS_ASSERT( IS_STATIC_RR(prrExist) );
|
|
|
|
if ( !fstatic )
|
|
{
|
|
DNSDBG( ANY, (
|
|
"ERROR: attempted caching at static (hosts file) record data!\n"
|
|
"\tpRecord = %p\n"
|
|
"\tName = %S\n"
|
|
"\tType = %d\n"
|
|
"\t-- Dumping new cache record list.\n",
|
|
pRecordSet,
|
|
pRecordSet->pName,
|
|
pRecordSet->wType ));
|
|
goto Failed;
|
|
}
|
|
|
|
//
|
|
// append host file records
|
|
// - start at "record" which is addr of record ptr entry
|
|
// making pNext field the actual pointer
|
|
// - delete duplicates
|
|
// - tack new RR on end
|
|
// - blow away new RR name if existing record
|
|
//
|
|
// DCR: should have simple "make cache RR set" function that
|
|
// handles name and TTL issues
|
|
//
|
|
// DCR: broken if non-flush load hits wire data; wire data
|
|
// may have multiple RR sets
|
|
//
|
|
|
|
else
|
|
{
|
|
PDNS_RECORD prr;
|
|
PDNS_RECORD prrPrev = (PDNS_RECORD) &pentry->Records[iter];
|
|
|
|
while ( prr = prrPrev->pNext )
|
|
{
|
|
// matches existing record?
|
|
// - cut existing record from list and free
|
|
|
|
if ( Dns_RecordCompare( prr, pRecordSet ) )
|
|
{
|
|
prrPrev->pNext = prr->pNext;
|
|
Dns_RecordFree( prr );
|
|
}
|
|
else
|
|
{
|
|
prrPrev = prr;
|
|
}
|
|
}
|
|
|
|
//
|
|
// tack entry on to end
|
|
// - if existing records of type delete name
|
|
//
|
|
|
|
if ( prrPrev != (PDNS_RECORD)&pentry->Records[iter] )
|
|
{
|
|
if ( IS_FREE_OWNER(pRecordSet) )
|
|
{
|
|
RECORD_HEAP_FREE( pRecordSet->pName );
|
|
pRecordSet->pName = NULL;
|
|
}
|
|
}
|
|
prrPrev->pNext = pRecordSet;
|
|
goto Done;
|
|
}
|
|
}
|
|
|
|
//
|
|
// put record into cache entry
|
|
//
|
|
// if no slot is available, switch to a harder scrub
|
|
//
|
|
// DCR: realloc if out of slots
|
|
//
|
|
|
|
fretry = FALSE;
|
|
|
|
while ( 1 )
|
|
{
|
|
for ( iter = 0;
|
|
iter < (INT)pentry->MaxCount;
|
|
iter++ )
|
|
{
|
|
if ( pentry->Records[iter] == NULL )
|
|
{
|
|
pentry->Records[iter] = pRecordSet;
|
|
g_RecordSetCount++;
|
|
g_RecordSetCache++;
|
|
goto Done;
|
|
}
|
|
}
|
|
|
|
if ( !fretry )
|
|
{
|
|
DNSDBG( QUERY, (
|
|
"No slots caching RR set %p at entry %p\n"
|
|
"\tdoing strong flush to free slot.\n",
|
|
pRecordSet,
|
|
pentry ));
|
|
|
|
Cache_FlushEntryRecords(
|
|
pentry,
|
|
FLUSH_LEVEL_WIRE,
|
|
0 );
|
|
|
|
fretry = TRUE;
|
|
continue;
|
|
}
|
|
|
|
DNSDBG( ANY, (
|
|
"ERROR: Failed to cache set %p at entry %p\n",
|
|
pRecordSet,
|
|
pentry ));
|
|
goto Failed;
|
|
}
|
|
|
|
Failed:
|
|
|
|
DNSDBG( TRACE, ( "Cache_RecordSetAtomic() => failed\n" ));
|
|
Dns_RecordListFree( pRecordSet );
|
|
|
|
Done:
|
|
|
|
UNLOCK_CACHE();
|
|
DNSDBG( TRACE, ( "Leave Cache_RecordSetAtomic()\n" ));
|
|
return;
|
|
}
|
|
|
|
|
|
|
|
VOID
|
|
Cache_RecordList(
|
|
IN OUT PDNS_RECORD pRecordList
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Cache record list.
|
|
|
|
This is cache routine for "oddball" records -- not caching under
|
|
queried name.
|
|
- hostfile
|
|
- answer records at CNAME
|
|
- additional data at additional name
|
|
|
|
Arguments:
|
|
|
|
pRecordList -- record list to cache
|
|
|
|
Return Value:
|
|
|
|
None
|
|
|
|
--*/
|
|
{
|
|
BOOL fcnameAnswer = FALSE;
|
|
PDNS_RECORD pnextRR = pRecordList;
|
|
PDNS_RECORD prr;
|
|
BOOL fstatic;
|
|
|
|
|
|
DNSDBG( TRACE, (
|
|
"Cache_RecordList( rr=%p )\n",
|
|
pRecordList ));
|
|
|
|
if ( !pRecordList )
|
|
{
|
|
return;
|
|
}
|
|
fstatic = IS_STATIC_RR(pRecordList);
|
|
|
|
//
|
|
// cache records:
|
|
// - cache additional records in query
|
|
// - cache CNAME data from query
|
|
// - cache host file data
|
|
//
|
|
// background: Glenn's caching paradigm was to cache all answer
|
|
// data at the queried name in the API call (name might be short).
|
|
// However, not caching the CNAME data can cause problems, so this
|
|
// was tacked on.
|
|
//
|
|
// For CNAME caching we throw away the CNAMEs themselves and just
|
|
// cache the actually data (address) records at the CNAME node.
|
|
//
|
|
|
|
//
|
|
// cache additional records
|
|
//
|
|
|
|
while ( prr = pnextRR )
|
|
{
|
|
BOOL fcacheSet = FALSE;
|
|
|
|
pnextRR = Dns_RecordSetDetach( prr );
|
|
|
|
//
|
|
// host file data -- always cache
|
|
//
|
|
// for CNAME want CNAME AND associated answer data
|
|
// - detach to get new next set
|
|
// - append answer data back on to CNAME for caching
|
|
// - next RR set (if exists) will be another CNAME
|
|
// to the same address data
|
|
//
|
|
// DCR: follow CNAMEs in cache
|
|
// then could pull this hack
|
|
// and avoid double building of answer data in dnsapi
|
|
//
|
|
|
|
if ( fstatic )
|
|
{
|
|
fcacheSet = TRUE;
|
|
|
|
if ( prr->wType == DNS_TYPE_CNAME &&
|
|
pnextRR &&
|
|
pnextRR->wType != DNS_TYPE_CNAME )
|
|
{
|
|
PDNS_RECORD panswer = pnextRR;
|
|
|
|
pnextRR = Dns_RecordSetDetach( panswer );
|
|
|
|
Dns_RecordListAppend( prr, panswer );
|
|
}
|
|
}
|
|
|
|
//
|
|
// wire data -- do NOT cache:
|
|
// - answer records for queried name (not CNAME)
|
|
// - CNAME records when doing caching of answer data under CNAME
|
|
// - authority section records (NS, SOA, etc)
|
|
// - OPT records
|
|
//
|
|
|
|
else if ( prr->Flags.S.Section == DNSREC_ANSWER )
|
|
{
|
|
if ( prr->wType == DNS_TYPE_CNAME )
|
|
{
|
|
fcnameAnswer = TRUE;
|
|
}
|
|
else if ( fcnameAnswer )
|
|
{
|
|
fcacheSet = TRUE;
|
|
}
|
|
}
|
|
else if ( prr->Flags.S.Section == DNSREC_ADDITIONAL )
|
|
{
|
|
if ( prr->wType != DNS_TYPE_OPT )
|
|
{
|
|
fcacheSet = TRUE;
|
|
}
|
|
}
|
|
|
|
if ( !fcacheSet )
|
|
{
|
|
Dns_RecordListFree( prr );
|
|
continue;
|
|
}
|
|
|
|
//
|
|
// cache the set
|
|
//
|
|
// flip the section field to "Answer" section
|
|
//
|
|
// DCR: section caching?
|
|
//
|
|
// note: section fields in cache indicate whether
|
|
// answer data (or additional) once out of
|
|
// cache;
|
|
// this is necessary since we cache everything
|
|
// at node and return it in one RR list; we'd
|
|
// to change must
|
|
// - return in different lists with some indication
|
|
// in cache of what's what
|
|
// OR
|
|
// - another indication of what's what
|
|
//
|
|
|
|
//if ( !fstatic )
|
|
// currently HostFile entries get answer too
|
|
{
|
|
PDNS_RECORD ptemp = prr;
|
|
while ( ptemp )
|
|
{
|
|
ptemp->Flags.S.Section = DNSREC_ANSWER;
|
|
ptemp = ptemp->pNext;
|
|
}
|
|
}
|
|
|
|
Cache_RecordSetAtomic(
|
|
NULL,
|
|
0,
|
|
prr );
|
|
}
|
|
|
|
DNSDBG( TRACE, ( "Leave Cache_RecordList()\n" ));
|
|
}
|
|
|
|
|
|
|
|
VOID
|
|
Cache_FlushRecords(
|
|
IN PWSTR pName,
|
|
IN DWORD Level,
|
|
IN WORD Type
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Flush cached records corresponding to a name and type.
|
|
|
|
Arguments:
|
|
|
|
pName -- name of records to delete
|
|
|
|
Level -- flush level
|
|
|
|
Type -- type of records to delete;
|
|
0 to flush all records at name
|
|
|
|
Return Value:
|
|
|
|
ERROR_SUCCESS if successful.
|
|
ErrorCode on failure.
|
|
|
|
--*/
|
|
{
|
|
WORD iter;
|
|
PCACHE_ENTRY pentry = NULL;
|
|
PCACHE_ENTRY pprevEntry = NULL;
|
|
|
|
|
|
DNSDBG( TRACE, (
|
|
"Cache_FlushRecords( %S, %d )\n",
|
|
pName,
|
|
Type ));
|
|
|
|
//
|
|
// lock with no-start
|
|
// - bail if no cache
|
|
//
|
|
// need this as PnP release notifications will attempt to
|
|
// flush local cache entries; this avoids rebuilding when
|
|
// already down
|
|
//
|
|
|
|
LOCK_CACHE_NO_START();
|
|
if ( !g_HashTable )
|
|
{
|
|
goto Done;
|
|
}
|
|
|
|
//
|
|
// find entry in cache
|
|
//
|
|
pentry = Cache_FindEntry(
|
|
pName,
|
|
FALSE // no create
|
|
);
|
|
if ( !pentry )
|
|
{
|
|
goto Done;
|
|
}
|
|
|
|
//
|
|
// flush records of type
|
|
// - zero type will flush all
|
|
//
|
|
// note: Cache_FindEntry() always moves the found entry
|
|
// to the front of the hash bucket list; this allows
|
|
// us to directly whack the entry
|
|
//
|
|
|
|
if ( Cache_FlushEntryRecords(
|
|
pentry,
|
|
Level,
|
|
Type ) )
|
|
{
|
|
DWORD index = getHashIndex(
|
|
pentry->pName,
|
|
0 );
|
|
|
|
DNS_ASSERT( pentry == g_HashTable[index] );
|
|
if ( pentry == g_HashTable[index] )
|
|
{
|
|
g_HashTable[ index ] = pentry->pNext;
|
|
Cache_FreeEntry( pentry );
|
|
}
|
|
}
|
|
|
|
Done:
|
|
|
|
UNLOCK_CACHE();
|
|
}
|
|
|
|
|
|
|
|
#if 0
|
|
BOOL
|
|
ReadCachedResults(
|
|
OUT PDNS_RESULTS pResults,
|
|
IN PWSTR pwsName,
|
|
IN WORD wType
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Find records of given name and type in cache.
|
|
|
|
Arguments:
|
|
|
|
pResults -- addr to receive results
|
|
|
|
pwsName -- name
|
|
|
|
wType -- record type to find
|
|
|
|
Return Value:
|
|
|
|
TRUE if results found.
|
|
FALSE if no cached data for name and type.
|
|
|
|
--*/
|
|
{
|
|
PDNS_RECORD prr;
|
|
DNS_STATUS status;
|
|
BOOL found = FALSE;
|
|
|
|
//
|
|
// clear results
|
|
//
|
|
|
|
RtlZeroMemory( pResults, sizeof(*pResults) );
|
|
|
|
// get cache results
|
|
|
|
|
|
// break out into results buffer
|
|
|
|
if ( found )
|
|
{
|
|
BreakRecordsIntoBlob(
|
|
pResults,
|
|
prr,
|
|
wType );
|
|
|
|
pResults->Status = status;
|
|
}
|
|
|
|
return( found );
|
|
}
|
|
#endif
|
|
|
|
|
|
|
|
//
|
|
// Cache utilities for remote routines
|
|
//
|
|
|
|
PDNS_RECORD
|
|
Cache_FindRecordsPrivate(
|
|
IN PWSTR pwsName,
|
|
IN WORD wType
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Find records of given name and type in cache.
|
|
|
|
Arguments:
|
|
|
|
pwsName -- name
|
|
|
|
Type -- record type to find
|
|
|
|
Return Value:
|
|
|
|
Ptr to record set of desired type -- if found.
|
|
NULL if not found.
|
|
|
|
--*/
|
|
{
|
|
PCACHE_ENTRY pentry;
|
|
PDNS_RECORD prr = NULL;
|
|
|
|
DNSDBG( TRACE, (
|
|
"Cache_FindRecordsPrivate( %S, type=%d )\n",
|
|
pwsName,
|
|
wType ));
|
|
|
|
if ( LOCK_CACHE() != NO_ERROR )
|
|
{
|
|
goto Done;
|
|
}
|
|
|
|
pentry = Cache_FindEntry(
|
|
pwsName,
|
|
FALSE );
|
|
if ( pentry )
|
|
{
|
|
prr = Cache_FindEntryRecords(
|
|
NULL, // don't need RR list ptr
|
|
pentry,
|
|
wType );
|
|
}
|
|
|
|
UNLOCK_CACHE();
|
|
|
|
Done:
|
|
|
|
DNSDBG( TRACE, (
|
|
"Leave Cache_FindRecordsPrivate( %S, type=%d ) => %p\n",
|
|
pwsName,
|
|
wType,
|
|
prr ));
|
|
|
|
return prr;
|
|
}
|
|
|
|
|
|
|
|
BOOL
|
|
Cache_GetRecordsForRpc(
|
|
OUT PDNS_RECORD * ppRecordList,
|
|
OUT PDNS_STATUS pStatus,
|
|
IN PWSTR pwsName,
|
|
IN WORD wType,
|
|
IN DWORD Flags
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Find records of given name and type in cache.
|
|
|
|
Arguments:
|
|
|
|
ppRecordList -- addr to receive pointer to record list
|
|
|
|
pStatus -- addr to get status return
|
|
|
|
pwsName -- name
|
|
|
|
Type -- record type to find
|
|
|
|
Flags -- query flags
|
|
|
|
Return Value:
|
|
|
|
TRUE if cache hit. OUT params are valid.
|
|
FALSE if cache miss. OUT params are unset.
|
|
|
|
--*/
|
|
{
|
|
PDNS_RECORD prr;
|
|
PDNS_RECORD prrResult = NULL;
|
|
DNS_STATUS status = NO_ERROR;
|
|
|
|
DNSDBG( RPC, (
|
|
"Cache_GetRecordsForRpc( %S, t=%d )\n",
|
|
pwsName,
|
|
wType ));
|
|
|
|
if ( (Flags & DNS_QUERY_BYPASS_CACHE) &&
|
|
(Flags & DNS_QUERY_NO_HOSTS_FILE) )
|
|
{
|
|
return FALSE;
|
|
}
|
|
if ( LOCK_CACHE() != NO_ERROR )
|
|
{
|
|
return FALSE;
|
|
}
|
|
|
|
//
|
|
// check cache for name and type
|
|
// - if name or type missing, jump to wire lookup
|
|
//
|
|
|
|
prr = Cache_FindRecordsPrivate(
|
|
pwsName,
|
|
wType );
|
|
if ( !prr )
|
|
{
|
|
goto Failed;
|
|
}
|
|
|
|
//
|
|
// cache hit
|
|
//
|
|
// if only interested in host file data ignore
|
|
//
|
|
|
|
if ( IS_HOSTS_FILE_RR(prr) )
|
|
{
|
|
if ( Flags & DNS_QUERY_NO_HOSTS_FILE )
|
|
{
|
|
goto Failed;
|
|
}
|
|
}
|
|
else // cache data
|
|
{
|
|
if ( Flags & DNS_QUERY_BYPASS_CACHE )
|
|
{
|
|
goto Failed;
|
|
}
|
|
}
|
|
|
|
//
|
|
// build response from cache data
|
|
// - cached NAME_ERROR or empty
|
|
// - cached records
|
|
//
|
|
|
|
if ( prr->wDataLength == 0 )
|
|
{
|
|
status = (prr->wType == DNS_TYPE_ANY)
|
|
? DNS_ERROR_RCODE_NAME_ERROR
|
|
: DNS_INFO_NO_RECORDS;
|
|
}
|
|
else
|
|
{
|
|
// for CNAME query, get only the CNAME record itself
|
|
// not the data at the CNAME
|
|
//
|
|
// DCR: CNAME handling should be optional -- not given
|
|
// for cache display purposes
|
|
//
|
|
|
|
if ( wType == DNS_TYPE_CNAME &&
|
|
prr->wType == DNS_TYPE_CNAME &&
|
|
prr->Flags.S.Section == DNSREC_ANSWER )
|
|
{
|
|
prrResult = Dns_RecordCopyEx(
|
|
prr,
|
|
DnsCharSetUnicode,
|
|
DnsCharSetUnicode );
|
|
}
|
|
else
|
|
{
|
|
prrResult = Dns_RecordSetCopyEx(
|
|
prr,
|
|
DnsCharSetUnicode,
|
|
DnsCharSetUnicode );
|
|
}
|
|
|
|
if ( prrResult )
|
|
{
|
|
Cache_RestoreRecordListForRpc( prrResult );
|
|
status = ERROR_SUCCESS;
|
|
}
|
|
else
|
|
{
|
|
status = ERROR_NOT_ENOUGH_MEMORY;
|
|
}
|
|
}
|
|
|
|
UNLOCK_CACHE();
|
|
|
|
// set return values
|
|
|
|
*ppRecordList = prrResult;
|
|
*pStatus = status;
|
|
|
|
return TRUE;
|
|
|
|
|
|
Failed:
|
|
|
|
UNLOCK_CACHE();
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
|
|
VOID
|
|
Cache_DeleteMatchingRecords(
|
|
IN PDNS_RECORD pRecords
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Delete particular records from the cache.
|
|
|
|
This is used to delete cluster records.
|
|
|
|
Arguments:
|
|
|
|
pRecords -- records to remove from cache
|
|
|
|
Return Value:
|
|
|
|
None
|
|
|
|
--*/
|
|
{
|
|
PCACHE_ENTRY pentry = NULL;
|
|
PDNS_RECORD * prrListAddr;
|
|
PDNS_RECORD prr;
|
|
PDNS_RECORD pnextRR;
|
|
|
|
|
|
DNSDBG( TRACE, (
|
|
"Cache_DeleteMatchingRecords( %p )\n",
|
|
pRecords ));
|
|
|
|
|
|
//
|
|
// lock with no-start
|
|
// - bail if no cache
|
|
//
|
|
// need this as PnP release notifications will attempt to
|
|
// flush local cache entries; this avoids rebuilding when
|
|
// already down
|
|
//
|
|
|
|
LOCK_CACHE_NO_START();
|
|
if ( !g_HashTable )
|
|
{
|
|
goto Done;
|
|
}
|
|
|
|
//
|
|
// check all records
|
|
//
|
|
|
|
pnextRR = pRecords;
|
|
|
|
while ( prr = pnextRR )
|
|
{
|
|
pnextRR = prr->pNext;
|
|
|
|
//
|
|
// find entry in cache
|
|
//
|
|
|
|
pentry = Cache_FindEntry(
|
|
prr->pName,
|
|
FALSE // no create
|
|
);
|
|
if ( !pentry )
|
|
{
|
|
DNSDBG( TRACE, (
|
|
"No cache entry for record %p (n=%S)\n",
|
|
prr,
|
|
prr->pName ));
|
|
continue;
|
|
}
|
|
|
|
//
|
|
// find matching records for type
|
|
//
|
|
|
|
prrListAddr = NULL;
|
|
|
|
Cache_FindEntryRecords(
|
|
&prrListAddr,
|
|
pentry,
|
|
prr->wType );
|
|
|
|
if ( !prrListAddr )
|
|
{
|
|
DNSDBG( TRACE, (
|
|
"No cache record matching type for record %p (n=%S)\n",
|
|
prr,
|
|
prr->pName ));
|
|
continue;
|
|
}
|
|
|
|
//
|
|
// delete matching record from list
|
|
//
|
|
|
|
Dns_DeleteRecordFromList(
|
|
prrListAddr, // addr of list
|
|
prr // record to delete
|
|
);
|
|
}
|
|
|
|
Done:
|
|
|
|
UNLOCK_CACHE();
|
|
}
|
|
|
|
|
|
|
|
//
|
|
// Garbage collection
|
|
//
|
|
|
|
VOID
|
|
Cache_SizeCheck(
|
|
VOID
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Check cache size.
|
|
|
|
Arguments:
|
|
|
|
Flag -- flag, currently unused
|
|
|
|
Return Value:
|
|
|
|
None
|
|
|
|
--*/
|
|
{
|
|
//
|
|
// ok -- don't signal for garbage collect
|
|
//
|
|
// - below threshold
|
|
// - already in garbage collection
|
|
// - collected recently
|
|
//
|
|
|
|
if ( g_RecordSetCount < g_RecordSetCountThreshold ||
|
|
g_GarbageCollectFlag ||
|
|
g_NextGarbageTime > GetCurrentTimeInSeconds() )
|
|
{
|
|
return;
|
|
}
|
|
|
|
DNSDBG( CACHE, (
|
|
"Cache_SizeCheck() over threshold!\n"
|
|
"\tRecordSetCount = %d\n"
|
|
"\tRecordSetCountLimit = %d\n"
|
|
"\tStarting garbage collection ...\n",
|
|
g_RecordSetCount,
|
|
g_RecordSetCountThreshold ));
|
|
|
|
//
|
|
// signal within lock, so that service thread
|
|
// can do signal within lock and avoid race on StopFlag check
|
|
// obviously better to simply not overload lock
|
|
//
|
|
|
|
LOCK_CACHE_NO_START();
|
|
if ( !g_StopFlag )
|
|
{
|
|
g_GarbageCollectFlag = TRUE;
|
|
SetEvent( g_hStopEvent );
|
|
}
|
|
UNLOCK_CACHE();
|
|
}
|
|
|
|
|
|
|
|
VOID
|
|
Cache_GarbageCollect(
|
|
IN DWORD Flag
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Garbage collect cache.
|
|
|
|
Arguments:
|
|
|
|
Flag -- flag, currently unused
|
|
|
|
Return Value:
|
|
|
|
None
|
|
|
|
--*/
|
|
{
|
|
DWORD iter;
|
|
DWORD index;
|
|
WORD flushLevel;
|
|
DWORD passCount;
|
|
|
|
DNSDBG( CACHE, (
|
|
"Cache_GarbageCollect()\n"
|
|
"\tNextIndex = %d\n"
|
|
"\tRecordSetCount = %d\n"
|
|
"\tRecordSetLimit = %d\n"
|
|
"\tRecordSetThreshold = %d\n",
|
|
g_NextGarbageIndex,
|
|
g_RecordSetCount,
|
|
g_RecordSetCountLimit,
|
|
g_RecordSetCountThreshold
|
|
));
|
|
|
|
if ( !g_HashTable )
|
|
{
|
|
return;
|
|
}
|
|
|
|
//
|
|
// collect timed out data in cache
|
|
//
|
|
// DCR: smart garbage detect
|
|
// - cleans until below limit
|
|
// - first pass invalid
|
|
// - then the hard stuff
|
|
// use restartable index so get through the cach
|
|
//
|
|
|
|
passCount = 0;
|
|
while ( 1 )
|
|
{
|
|
if ( passCount == 0 )
|
|
{
|
|
flushLevel = FLUSH_LEVEL_INVALID;
|
|
}
|
|
else if ( passCount == 1 )
|
|
{
|
|
flushLevel = FLUSH_LEVEL_GARBAGE;
|
|
}
|
|
else
|
|
{
|
|
break;
|
|
}
|
|
passCount++;
|
|
|
|
//
|
|
// flush all hash bins at current flush level
|
|
// until
|
|
// - service stop
|
|
// - push cache size below limit
|
|
//
|
|
|
|
for ( iter = 0;
|
|
iter < g_HashTableSize;
|
|
iter++ )
|
|
{
|
|
index = (iter + g_NextGarbageIndex) % g_HashTableSize;
|
|
|
|
if ( g_StopFlag ||
|
|
g_WakeFlag ||
|
|
g_RecordSetCount < g_RecordSetCountLimit )
|
|
{
|
|
passCount = MAXDWORD;
|
|
break;
|
|
}
|
|
|
|
Cache_FlushBucket(
|
|
index,
|
|
flushLevel );
|
|
}
|
|
|
|
index++;
|
|
if ( index >= g_HashTableSize )
|
|
{
|
|
index = 0;
|
|
}
|
|
g_NextGarbageIndex = index;
|
|
}
|
|
|
|
//
|
|
// reset garbage globals
|
|
// - lockout for interval
|
|
// - clear signal flag
|
|
// - reset event (if not shuttting down)
|
|
//
|
|
// note: reset signal within lock, so that service thread
|
|
// can do signal within lock and avoid race on StopFlag check
|
|
// obviously better to simply not overload lock
|
|
//
|
|
|
|
g_NextGarbageTime = GetCurrentTimeInSeconds() + GARBAGE_LOCKOUT_INTERVAL;
|
|
|
|
LOCK_CACHE_NO_START();
|
|
if ( !g_StopFlag )
|
|
{
|
|
g_GarbageCollectFlag = FALSE;
|
|
ResetEvent( g_hStopEvent );
|
|
}
|
|
UNLOCK_CACHE();
|
|
|
|
DNSDBG( CACHE, (
|
|
"Leave Cache_GarbageCollect()\n"
|
|
"\tNextIndex = %d\n"
|
|
"\tNextTime = %d\n"
|
|
"\tRecordSetCount = %d\n"
|
|
"\tRecordSetLimit = %d\n"
|
|
"\tRecordSetThreshold = %d\n",
|
|
g_NextGarbageIndex,
|
|
g_NextGarbageTime,
|
|
g_RecordSetCount,
|
|
g_RecordSetCountLimit,
|
|
g_RecordSetCountThreshold
|
|
));
|
|
}
|
|
|
|
|
|
|
|
//
|
|
// Hostfile load stuff
|
|
//
|
|
|
|
VOID
|
|
LoadHostFileIntoCache(
|
|
IN PSTR pszFileName
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Read hosts file into cache.
|
|
|
|
Arguments:
|
|
|
|
pFileName -- file name to load
|
|
|
|
Return Value:
|
|
|
|
None.
|
|
|
|
--*/
|
|
{
|
|
HOST_FILE_INFO hostInfo;
|
|
|
|
DNSDBG( INIT, ( "Enter LoadHostFileIntoCache\n" ));
|
|
|
|
//
|
|
// read entries from host file until exhausted
|
|
// - cache A record for each name and alias
|
|
// - cache PTR to name
|
|
//
|
|
|
|
RtlZeroMemory(
|
|
&hostInfo,
|
|
sizeof(hostInfo) );
|
|
|
|
hostInfo.pszFileName = pszFileName;
|
|
|
|
if ( !HostsFile_Open( &hostInfo ) )
|
|
{
|
|
return;
|
|
}
|
|
hostInfo.fBuildRecords = TRUE;
|
|
|
|
while ( HostsFile_ReadLine( &hostInfo ) )
|
|
{
|
|
// cache all the records we sucked out
|
|
|
|
Cache_RecordList( hostInfo.pForwardRR );
|
|
Cache_RecordList( hostInfo.pReverseRR );
|
|
Cache_RecordList( hostInfo.pAliasRR );
|
|
}
|
|
|
|
HostsFile_Close( &hostInfo );
|
|
|
|
DNSDBG( INIT, ( "Leave LoadHostFileIntoCache\n" ));
|
|
}
|
|
|
|
|
|
|
|
VOID
|
|
InitCacheWithHostFile(
|
|
VOID
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Initialize cache with host(s) file.
|
|
|
|
This handles regular cache file and ICS file if it
|
|
exists.
|
|
|
|
Arguments:
|
|
|
|
None
|
|
|
|
Return Value:
|
|
|
|
None.
|
|
|
|
--*/
|
|
{
|
|
DNSDBG( INIT, ( "Enter InitCacheWithHostFile\n" ));
|
|
|
|
//
|
|
// load host file into cache
|
|
//
|
|
|
|
LoadHostFileIntoCache( NULL );
|
|
|
|
//
|
|
// if running ICS, load it's file also
|
|
//
|
|
|
|
LoadHostFileIntoCache( "hosts.ics" );
|
|
|
|
DNSDBG( INIT, ( "Leave InitCacheWithHostFile\n\n\n" ));
|
|
}
|
|
|
|
|
|
|
|
DNS_STATUS
|
|
Cache_QueryResponse(
|
|
IN OUT PQUERY_BLOB pBlob
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Find records of given name and type in cache.
|
|
|
|
Arguments:
|
|
|
|
pBlob -- query blob
|
|
|
|
Uses:
|
|
pwsName
|
|
wType
|
|
Status
|
|
pRecords
|
|
fCacheNegativeResponse
|
|
|
|
Sets:
|
|
pRecords - may be reset to exclude non-RPCable records
|
|
|
|
Return Value:
|
|
|
|
ErrorStatus -- same as query status, unless processing error during caching
|
|
|
|
--*/
|
|
{
|
|
DNS_STATUS status = pBlob->Status;
|
|
PWSTR pname = pBlob->pNameOrig;
|
|
WORD wtype = pBlob->wType;
|
|
PDNS_RECORD presultRR = pBlob->pRecords;
|
|
|
|
|
|
DNSDBG( RPC, (
|
|
"\nCache_QueryResponse( %S, type %d )\n",
|
|
pname,
|
|
wtype ));
|
|
|
|
//
|
|
// successful response
|
|
// - make copy of records to return to caller
|
|
// - cache actual query record set
|
|
// - make copy to cache any additional data
|
|
//
|
|
|
|
if ( status == ERROR_SUCCESS && presultRR )
|
|
{
|
|
DWORD copyFlag;
|
|
PDNS_RECORD prrCache;
|
|
|
|
// cleanup for RPC and caching
|
|
|
|
prrCache = Dns_RecordListScreen(
|
|
presultRR,
|
|
SCREEN_OUT_AUTHORITY | SCREEN_OUT_NON_RPC );
|
|
|
|
//
|
|
// make copy for return
|
|
// - don't include authority records
|
|
//
|
|
// NOTE: IMPORTANT
|
|
// we return (RPC) a COPY of the wire set and cache the
|
|
// wire set; this is because the wire set has imbedded data
|
|
// (the data pointers are not actual heap allocations) and
|
|
// and hence can not be RPC'd (without changing the RPC
|
|
// definition to flat data)
|
|
//
|
|
// if we later want to return authority data on first query,
|
|
// then
|
|
// - clean non-RPC only
|
|
// - including owner name fixups
|
|
// - copy for result set
|
|
// - clean original for authority -- cache
|
|
// - clean any additional -- cache
|
|
//
|
|
// note: do name pointer fixup by making round trip into cache format
|
|
//
|
|
// DCR: shouldn't have external name pointers anywhere
|
|
// DCR: do RPC-able cleanup on original set before copy
|
|
// OR
|
|
// DCR: have "cache state" on record
|
|
// then could move original results to cache state and caching
|
|
// routines could detect state and avoid double TTLing
|
|
|
|
presultRR = Dns_RecordListCopyEx(
|
|
prrCache,
|
|
0,
|
|
// SCREEN_OUT_AUTHORITY
|
|
DnsCharSetUnicode,
|
|
DnsCharSetUnicode );
|
|
|
|
pBlob->pRecords = presultRR;
|
|
if ( !presultRR )
|
|
{
|
|
Dns_RecordListFree( prrCache );
|
|
status = DNS_ERROR_NO_MEMORY;
|
|
goto Done;
|
|
}
|
|
|
|
// name pointer fixup
|
|
|
|
Cache_PrepareRecordList( presultRR );
|
|
Cache_RestoreRecordListForRpc( presultRR );
|
|
|
|
//
|
|
// do NOT cache local records
|
|
//
|
|
// note: we went through this function only to get
|
|
// PTR records and CNAME records in RPC format
|
|
// (no imbedded pointers)
|
|
//
|
|
|
|
if ( pBlob->pLocalRecords )
|
|
{
|
|
Dns_RecordListFree( prrCache );
|
|
goto Done;
|
|
}
|
|
|
|
//
|
|
// cache original data
|
|
//
|
|
|
|
if ( prrCache )
|
|
{
|
|
Cache_RecordSetAtomic(
|
|
pname,
|
|
wtype,
|
|
prrCache );
|
|
}
|
|
|
|
//
|
|
// extra records
|
|
// - additional data
|
|
// - CNAME answer data to cache at CNAME itself
|
|
// in CNAME case must include ANSWER data, but
|
|
// skip the CNAME itself
|
|
//
|
|
// Cache_RecordList() breaks records into RR sets before caching
|
|
//
|
|
|
|
prrCache = presultRR;
|
|
copyFlag = SCREEN_OUT_ANSWER | SCREEN_OUT_AUTHORITY;
|
|
|
|
if ( prrCache->wType == DNS_TYPE_CNAME )
|
|
{
|
|
prrCache = prrCache->pNext;
|
|
copyFlag = SCREEN_OUT_AUTHORITY;
|
|
}
|
|
|
|
prrCache = Dns_RecordListCopyEx(
|
|
prrCache,
|
|
copyFlag,
|
|
DnsCharSetUnicode,
|
|
DnsCharSetUnicode );
|
|
if ( prrCache )
|
|
{
|
|
Cache_RecordList( prrCache );
|
|
}
|
|
}
|
|
|
|
//
|
|
// negative response
|
|
//
|
|
|
|
else if ( status == DNS_ERROR_RCODE_NAME_ERROR ||
|
|
status == DNS_INFO_NO_RECORDS )
|
|
{
|
|
DWORD ttl;
|
|
PDNS_RECORD prr;
|
|
|
|
if ( !pBlob->fCacheNegative )
|
|
{
|
|
DNSDBG( QUERY, (
|
|
"No negative caching for %S, type=%d\n",
|
|
pname, wtype ));
|
|
goto Done;
|
|
}
|
|
|
|
//
|
|
// create negative cache entry
|
|
//
|
|
// DCR: should use TTL returned in SOA
|
|
//
|
|
|
|
prr = Dns_AllocateRecord( 0 );
|
|
if ( !prr )
|
|
{
|
|
status = ERROR_NOT_ENOUGH_MEMORY;
|
|
goto Done;
|
|
}
|
|
|
|
prr->pName = (PWSTR) Dns_StringCopyAllocate(
|
|
(PCHAR) pname,
|
|
0, // NULL terminated
|
|
DnsCharSetUnicode,
|
|
DnsCharSetUnicode );
|
|
if ( prr->pName )
|
|
{
|
|
SET_FREE_OWNER( prr );
|
|
}
|
|
|
|
prr->wDataLength = 0;
|
|
ttl = g_MaxNegativeCacheTtl;
|
|
|
|
if ( wtype == DNS_TYPE_SOA
|
|
&&
|
|
ttl > g_NegativeSOACacheTime )
|
|
{
|
|
ttl = g_NegativeSOACacheTime;
|
|
}
|
|
prr->dwTtl = ttl;
|
|
prr->Flags.S.CharSet = DnsCharSetUnicode;
|
|
prr->Flags.S.Section = DNSREC_ANSWER;
|
|
prr->Flags.DW |= DNSREC_NOEXIST;
|
|
|
|
if ( status == DNS_ERROR_RCODE_NAME_ERROR )
|
|
{
|
|
prr->wType = DNS_TYPE_ANY;
|
|
}
|
|
else
|
|
{
|
|
prr->wType = wtype;
|
|
}
|
|
|
|
Cache_RecordSetAtomic(
|
|
NULL, // default name
|
|
0, // default type
|
|
prr );
|
|
}
|
|
|
|
// failure return from query
|
|
// - nothing to cache
|
|
|
|
else
|
|
{
|
|
DNSDBG( QUERY, (
|
|
"Uncacheable error code %d -- no caching for %S, type=%d\n",
|
|
status,
|
|
pname,
|
|
wtype ));
|
|
}
|
|
|
|
Done:
|
|
|
|
//
|
|
// check cache size to see if garbage collect necessary
|
|
//
|
|
// note we do this only on query caching; this avoids
|
|
// - jamming ourselves in hosts file load
|
|
// - wakeup and grabbing lock between separate sets of query response
|
|
//
|
|
|
|
Cache_SizeCheck();
|
|
|
|
return status;
|
|
}
|
|
|
|
//
|
|
// End ncache.c
|
|
//
|