mirror of
https://github.com/reactos/reactos.git
synced 2025-01-01 03:54:02 +00:00
9393fc320e
Excluded: 3rd-party code (incl. wine) and most of the win32ss.
318 lines
7.8 KiB
C
318 lines
7.8 KiB
C
/*
|
|
* COPYRIGHT: See COPYING in the top level directory
|
|
* PROJECT: ReactOS system libraries
|
|
* FILE: base/services/dnsrslvr/cache.c
|
|
* PURPOSE: DNS cache functions
|
|
* PROGRAMMER: Peter Hater
|
|
*/
|
|
|
|
#include "precomp.h"
|
|
|
|
#define NDEBUG
|
|
#include <debug.h>
|
|
|
|
static RESOLVER_CACHE DnsCache;
|
|
static BOOL DnsCacheInitialized = FALSE;
|
|
|
|
#define DnsCacheLock() do { EnterCriticalSection(&DnsCache.Lock); } while (0)
|
|
#define DnsCacheUnlock() do { LeaveCriticalSection(&DnsCache.Lock); } while (0)
|
|
|
|
VOID
|
|
DnsIntCacheInitialize(VOID)
|
|
{
|
|
DPRINT("DnsIntCacheInitialize()\n");
|
|
|
|
/* Check if we're initialized */
|
|
if (DnsCacheInitialized)
|
|
return;
|
|
|
|
/* Initialize the cache lock and namespace list */
|
|
InitializeCriticalSection((LPCRITICAL_SECTION)&DnsCache.Lock);
|
|
InitializeListHead(&DnsCache.RecordList);
|
|
DnsCacheInitialized = TRUE;
|
|
}
|
|
|
|
VOID
|
|
DnsIntCacheFree(VOID)
|
|
{
|
|
DPRINT("DnsIntCacheFree()\n");
|
|
|
|
/* Check if we're initialized */
|
|
if (!DnsCacheInitialized)
|
|
return;
|
|
|
|
if (!DnsCache.RecordList.Flink)
|
|
return;
|
|
|
|
DnsIntCacheFlush(CACHE_FLUSH_ALL);
|
|
|
|
DeleteCriticalSection(&DnsCache.Lock);
|
|
DnsCacheInitialized = FALSE;
|
|
}
|
|
|
|
VOID
|
|
DnsIntCacheRemoveEntryItem(PRESOLVER_CACHE_ENTRY CacheEntry)
|
|
{
|
|
DPRINT("DnsIntCacheRemoveEntryItem(%p)\n", CacheEntry);
|
|
|
|
/* Remove the entry from the list */
|
|
RemoveEntryList(&CacheEntry->CacheLink);
|
|
|
|
/* Free record */
|
|
DnsRecordListFree(CacheEntry->Record, DnsFreeRecordList);
|
|
|
|
/* Delete us */
|
|
HeapFree(GetProcessHeap(), 0, CacheEntry);
|
|
}
|
|
|
|
DNS_STATUS
|
|
DnsIntCacheFlush(
|
|
_In_ ULONG ulFlags)
|
|
{
|
|
PLIST_ENTRY Entry, NextEntry;
|
|
PRESOLVER_CACHE_ENTRY CacheEntry;
|
|
|
|
DPRINT("DnsIntCacheFlush(%lu)\n", ulFlags);
|
|
|
|
/* Lock the cache */
|
|
DnsCacheLock();
|
|
|
|
/* Loop every entry */
|
|
Entry = DnsCache.RecordList.Flink;
|
|
while (Entry != &DnsCache.RecordList)
|
|
{
|
|
NextEntry = Entry->Flink;
|
|
|
|
/* Get this entry */
|
|
CacheEntry = CONTAINING_RECORD(Entry, RESOLVER_CACHE_ENTRY, CacheLink);
|
|
|
|
/* Remove it from list */
|
|
if (((ulFlags & CACHE_FLUSH_HOSTS_FILE_ENTRIES) && (CacheEntry->bHostsFileEntry != FALSE)) ||
|
|
((ulFlags & CACHE_FLUSH_NON_HOSTS_FILE_ENTRIES) && (CacheEntry->bHostsFileEntry == FALSE)))
|
|
DnsIntCacheRemoveEntryItem(CacheEntry);
|
|
|
|
/* Move to the next entry */
|
|
Entry = NextEntry;
|
|
}
|
|
|
|
/* Unlock the cache */
|
|
DnsCacheUnlock();
|
|
|
|
return ERROR_SUCCESS;
|
|
}
|
|
|
|
|
|
DNS_STATUS
|
|
DnsIntFlushCacheEntry(
|
|
_In_ LPCWSTR pszName,
|
|
_In_ WORD wType)
|
|
{
|
|
PLIST_ENTRY Entry, NextEntry;
|
|
PRESOLVER_CACHE_ENTRY CacheEntry;
|
|
|
|
DPRINT("DnsIntFlushCacheEntry(%S %x)\n", pszName, wType);
|
|
|
|
/* Lock the cache */
|
|
DnsCacheLock();
|
|
|
|
/* Loop every entry */
|
|
Entry = DnsCache.RecordList.Flink;
|
|
while (Entry != &DnsCache.RecordList)
|
|
{
|
|
NextEntry = Entry->Flink;
|
|
|
|
/* Get this entry */
|
|
CacheEntry = CONTAINING_RECORD(Entry, RESOLVER_CACHE_ENTRY, CacheLink);
|
|
|
|
/* Remove it from the list */
|
|
if ((_wcsicmp(CacheEntry->Record->pName, pszName) == 0) &&
|
|
(CacheEntry->bHostsFileEntry == FALSE))
|
|
{
|
|
if ((wType == DNS_TYPE_ANY) ||
|
|
(CacheEntry->Record->wType == wType))
|
|
{
|
|
DnsIntCacheRemoveEntryItem(CacheEntry);
|
|
}
|
|
}
|
|
|
|
/* Move to the next entry */
|
|
Entry = NextEntry;
|
|
}
|
|
|
|
/* Unlock the cache */
|
|
DnsCacheUnlock();
|
|
|
|
return ERROR_SUCCESS;
|
|
}
|
|
|
|
|
|
DNS_STATUS
|
|
DnsIntCacheGetEntryByName(
|
|
LPCWSTR Name,
|
|
WORD wType,
|
|
DWORD dwFlags,
|
|
PDNS_RECORDW *Record)
|
|
{
|
|
DNS_STATUS Status = DNS_INFO_NO_RECORDS;
|
|
PRESOLVER_CACHE_ENTRY CacheEntry;
|
|
PLIST_ENTRY NextEntry;
|
|
|
|
DPRINT("DnsIntCacheGetEntryByName(%S %hu 0x%lx %p)\n",
|
|
Name, wType, dwFlags, Record);
|
|
|
|
/* Assume failure */
|
|
*Record = NULL;
|
|
|
|
/* Lock the cache */
|
|
DnsCacheLock();
|
|
|
|
/* Match the Id with all the entries in the List */
|
|
NextEntry = DnsCache.RecordList.Flink;
|
|
while (NextEntry != &DnsCache.RecordList)
|
|
{
|
|
/* Get the Current Entry */
|
|
CacheEntry = CONTAINING_RECORD(NextEntry, RESOLVER_CACHE_ENTRY, CacheLink);
|
|
|
|
/* Check if this is the Catalog Entry ID we want */
|
|
if (_wcsicmp(CacheEntry->Record->pName, Name) == 0)
|
|
{
|
|
/* Copy the entry and return it */
|
|
*Record = DnsRecordSetCopyEx(CacheEntry->Record, DnsCharSetUnicode, DnsCharSetUnicode);
|
|
Status = ERROR_SUCCESS;
|
|
break;
|
|
}
|
|
|
|
NextEntry = NextEntry->Flink;
|
|
}
|
|
|
|
/* Release the cache */
|
|
DnsCacheUnlock();
|
|
|
|
return Status;
|
|
}
|
|
|
|
BOOL
|
|
DnsIntCacheRemoveEntryByName(LPCWSTR Name)
|
|
{
|
|
BOOL Ret = FALSE;
|
|
PRESOLVER_CACHE_ENTRY CacheEntry;
|
|
PLIST_ENTRY NextEntry;
|
|
|
|
DPRINT("DnsIntCacheRemoveEntryByName(%S)\n", Name);
|
|
|
|
/* Lock the cache */
|
|
DnsCacheLock();
|
|
|
|
/* Match the Id with all the entries in the List */
|
|
NextEntry = DnsCache.RecordList.Flink;
|
|
while (NextEntry != &DnsCache.RecordList)
|
|
{
|
|
/* Get the Current Entry */
|
|
CacheEntry = CONTAINING_RECORD(NextEntry, RESOLVER_CACHE_ENTRY, CacheLink);
|
|
|
|
/* Check if this is the Catalog Entry ID we want */
|
|
if (_wcsicmp(CacheEntry->Record->pName, Name) == 0)
|
|
{
|
|
/* Remove the entry */
|
|
DnsIntCacheRemoveEntryItem(CacheEntry);
|
|
Ret = TRUE;
|
|
break;
|
|
}
|
|
|
|
NextEntry = NextEntry->Flink;
|
|
}
|
|
|
|
/* Release the cache */
|
|
DnsCacheUnlock();
|
|
|
|
/* Return */
|
|
return Ret;
|
|
}
|
|
|
|
VOID
|
|
DnsIntCacheAddEntry(
|
|
_In_ PDNS_RECORDW Record,
|
|
_In_ BOOL bHostsFileEntry)
|
|
{
|
|
PRESOLVER_CACHE_ENTRY Entry;
|
|
|
|
DPRINT("DnsIntCacheAddEntry(%p %u)\n",
|
|
Record, bHostsFileEntry);
|
|
|
|
DPRINT("Name: %S\n", Record->pName);
|
|
DPRINT("TTL: %lu\n", Record->dwTtl);
|
|
|
|
/* Lock the cache */
|
|
DnsCacheLock();
|
|
|
|
/* Match the Id with all the entries in the List */
|
|
Entry = (PRESOLVER_CACHE_ENTRY)HeapAlloc(GetProcessHeap(), 0, sizeof(*Entry));
|
|
if (!Entry)
|
|
return;
|
|
|
|
Entry->bHostsFileEntry = bHostsFileEntry;
|
|
Entry->Record = DnsRecordSetCopyEx(Record, DnsCharSetUnicode, DnsCharSetUnicode);
|
|
|
|
/* Insert it to our List */
|
|
InsertTailList(&DnsCache.RecordList, &Entry->CacheLink);
|
|
|
|
/* Release the cache */
|
|
DnsCacheUnlock();
|
|
}
|
|
|
|
DNS_STATUS
|
|
DnsIntCacheGetEntries(
|
|
_Out_ DNS_CACHE_ENTRY **ppCacheEntries)
|
|
{
|
|
PRESOLVER_CACHE_ENTRY CacheEntry;
|
|
PLIST_ENTRY NextEntry;
|
|
PDNS_CACHE_ENTRY pLastEntry = NULL, pNewEntry;
|
|
|
|
/* Lock the cache */
|
|
DnsCacheLock();
|
|
|
|
*ppCacheEntries = NULL;
|
|
|
|
NextEntry = DnsCache.RecordList.Flink;
|
|
while (NextEntry != &DnsCache.RecordList)
|
|
{
|
|
/* Get the Current Entry */
|
|
CacheEntry = CONTAINING_RECORD(NextEntry, RESOLVER_CACHE_ENTRY, CacheLink);
|
|
|
|
DPRINT("1 %S %lu\n", CacheEntry->Record->pName, CacheEntry->Record->wType);
|
|
if (CacheEntry->Record->pNext)
|
|
{
|
|
DPRINT("2 %S %lu\n", CacheEntry->Record->pNext->pName, CacheEntry->Record->pNext->wType);
|
|
}
|
|
|
|
pNewEntry = midl_user_allocate(sizeof(DNS_CACHE_ENTRY));
|
|
if (pNewEntry == NULL)
|
|
{
|
|
return ERROR_OUTOFMEMORY;
|
|
}
|
|
|
|
pNewEntry->pszName = midl_user_allocate((wcslen(CacheEntry->Record->pName) + 1) * sizeof(WCHAR));
|
|
if (pNewEntry->pszName == NULL)
|
|
{
|
|
return ERROR_OUTOFMEMORY;
|
|
}
|
|
|
|
wcscpy(pNewEntry->pszName, CacheEntry->Record->pName);
|
|
pNewEntry->wType1 = CacheEntry->Record->wType;
|
|
pNewEntry->wType2 = 0;
|
|
pNewEntry->wFlags = 0;
|
|
|
|
if (pLastEntry == NULL)
|
|
*ppCacheEntries = pNewEntry;
|
|
else
|
|
pLastEntry->pNext = pNewEntry;
|
|
pLastEntry = pNewEntry;
|
|
|
|
NextEntry = NextEntry->Flink;
|
|
}
|
|
|
|
/* Release the cache */
|
|
DnsCacheUnlock();
|
|
|
|
return ERROR_SUCCESS;
|
|
}
|