2020-09-30 16:53:55 +02:00

522 lines
18 KiB
C++

/*++
Copyright (c) 1996 Microsoft Corp.
Module Name: hashutil.cxx
Abstract:
Implementation of linked list of hash tables for cache index lookup.
Author:
Rajeev Dujari (rajeevd) 22-Oct-96
--*/
#include <wininetp.h>
#include <cache.hxx>
#define SIG_HASH ('H'|('A'<<8)|('S'<<16)|('H'<<24))
typedef LIST_FILEMAP_ENTRY HASH_FILEMAP_ENTRY;
// hash table parameters
#define BYTES_PER_PAGE 4096
#define ITEMS_PER_BUCKET ((BYTES_PER_PAGE - sizeof(HASH_FILEMAP_ENTRY))\
/ (SLOT_COUNT * sizeof(HASH_ITEM)))
#define BYTES_PER_TABLE (sizeof(HASH_FILEMAP_ENTRY) \
+ SLOT_COUNT * ITEMS_PER_BUCKET * sizeof(HASH_ITEM))
//
// Hash Function: Pearson's method
//
PRIVATE DWORD HashKey (LPCSTR lpsz, DWORD dwAddedHash)
{
union
{
DWORD dw;
BYTE c[4];
}
Hash, Hash2;
const static BYTE bTranslate[256] =
{
1, 14,110, 25, 97,174,132,119,138,170,125,118, 27,233,140, 51,
87,197,177,107,234,169, 56, 68, 30, 7,173, 73,188, 40, 36, 65,
49,213,104,190, 57,211,148,223, 48,115, 15, 2, 67,186,210, 28,
12,181,103, 70, 22, 58, 75, 78,183,167,238,157,124,147,172,144,
176,161,141, 86, 60, 66,128, 83,156,241, 79, 46,168,198, 41,254,
178, 85,253,237,250,154,133, 88, 35,206, 95,116,252,192, 54,221,
102,218,255,240, 82,106,158,201, 61, 3, 89, 9, 42,155,159, 93,
166, 80, 50, 34,175,195,100, 99, 26,150, 16,145, 4, 33, 8,189,
121, 64, 77, 72,208,245,130,122,143, 55,105,134, 29,164,185,194,
193,239,101,242, 5,171,126, 11, 74, 59,137,228,108,191,232,139,
6, 24, 81, 20,127, 17, 91, 92,251,151,225,207, 21, 98,113,112,
84,226, 18,214,199,187, 13, 32, 94,220,224,212,247,204,196, 43,
249,236, 45,244,111,182,153,136,129, 90,217,202, 19,165,231, 71,
230,142, 96,227, 62,179,246,114,162, 53,160,215,205,180, 47,109,
44, 38, 31,149,135, 0,216, 52, 63, 23, 37, 69, 39,117,146,184,
163,200,222,235,248,243,219, 10,152,131,123,229,203, 76,120,209
};
// Seed the hash values based on the first character.
Hash.c[0] = bTranslate[ *lpsz];
Hash.c[1] = bTranslate[(*lpsz+1) & 255];
Hash.c[2] = bTranslate[(*lpsz+2) & 255];
Hash.c[3] = bTranslate[(*lpsz+3) & 255];
Hash.dw += dwAddedHash;
while (*++lpsz)
{
// Allow URLs differing only by trailing slash to collide.
if (lpsz[0] == '/' && lpsz[1] == 0)
break;
Hash2.c[0] = Hash.c[0] ^ *lpsz;
Hash2.c[1] = Hash.c[1] ^ *lpsz;
Hash2.c[2] = Hash.c[2] ^ *lpsz;
Hash2.c[3] = Hash.c[3] ^ *lpsz;
Hash.c[0] = bTranslate[Hash2.c[0]];
Hash.c[1] = bTranslate[Hash2.c[1]];
Hash.c[2] = bTranslate[Hash2.c[2]];
Hash.c[3] = bTranslate[Hash2.c[3]];
}
return Hash.dw;
}
//
// HashLookupItem support functions specific to urlcache:
// AllocTable
// IsMatch
//
PRIVATE HASH_FILEMAP_ENTRY* AllocTable
(LPVOID pAllocObj, LPBYTE* ppBase, LPDWORD* ppdwOffset)
{
// Save the offset to the table offset.
DWORD_PTR dpOffsetToTableOffset = (LPBYTE)*ppdwOffset - *ppBase; // 64BIT
// Ask for BYTES_PER_PAGE instead of BYTES_PER_TABLE
// so the allocator knows to align on a page boundary.
INET_ASSERT (BYTES_PER_PAGE >= BYTES_PER_TABLE);
MEMMAP_FILE* pmmf = (MEMMAP_FILE*) pAllocObj;
HASH_FILEMAP_ENTRY* pTable =
(HASH_FILEMAP_ENTRY *) pmmf->AllocateEntry (BYTES_PER_PAGE);
if (!pTable)
return NULL;
INET_ASSERT (! (((LPBYTE) pTable - *pmmf->GetHeapStart()) & (BYTES_PER_PAGE-1)) );
// Chain new table to previous table.
*ppBase = *pmmf->GetHeapStart();
*ppdwOffset = (DWORD*) (*ppBase + dpOffsetToTableOffset);
**ppdwOffset = (DWORD) ((LPBYTE)pTable - *ppBase); // 64BIT
// Initialize the header.
pTable->dwSig = SIG_HASH;
pTable->dwNext = 0;
// Fill the rest of the entry with HASH_END
DWORD* pdw = (DWORD *) (pTable + 1);
DWORD cdw = SLOT_COUNT * ITEMS_PER_BUCKET * (sizeof(HASH_ITEM)/sizeof(DWORD));
INET_ASSERT (!(sizeof(HASH_ITEM) % sizeof(DWORD)));
while (cdw--)
*pdw++ = HASH_END;
// Return the new table.
return pTable;
}
//
// IsMatch: determine if hash table item with a matching 32-bit hash value
// is an actual match or return NULL if a collision.
//
PRIVATE HASH_ITEM* URL_CONTAINER::IsMatch
(HASH_ITEM *pItem, LPCSTR pszKey, DWORD dwFlags)
{
MEMMAP_FILE* pmmf = _UrlObjStorage;
dwFlags &= (LOOKUP_BIT_REDIR | LOOKUP_BIT_CREATE);
if (pmmf->IsBadOffset (pItem->dwOffset))
{
// Fix up a bad hash table item. This could happen if a thread
// died between allocating a hash table item and setting the offset.
pItem->MarkFree();
return NULL;
}
FILEMAP_ENTRY* pEntry = (FILEMAP_ENTRY*)
(*pmmf->GetHeapStart() + pItem->dwOffset);
switch (pEntry->dwSig)
{
case SIG_URL:
{
// Fail if lookup flags are inconsistent with url entry type.
INET_ASSERT (!(pItem->dwHash & HASH_BIT_NOTURL));
// Get pointer to URL.
URL_FILEMAP_ENTRY *pUrlEntry = (URL_FILEMAP_ENTRY *) pEntry;
LPSTR pszUrl = ((LPSTR) pUrlEntry) + pUrlEntry->UrlNameOffset;
LPCSTR pszKey2 = pszKey, pszUrl2 = pszUrl;
while ( *pszKey2 && *pszUrl2 && *pszKey2 == *pszUrl2 )
{
pszKey2++;
pszUrl2++;
}
if (!*pszKey2 && ! *pszUrl2)
{
// Found exact match.
if (dwFlags == LOOKUP_REDIR_CREATE)
{
// We are have a cache entry for a URL which is now
// redirecting. Delete the cache entry.
DeleteUrlEntry (pUrlEntry, pItem, SIG_DELETE);
return NULL;
}
return pItem;
}
// If redirects allowed, check for trailing slash match.
if ((dwFlags == LOOKUP_URL_TRANSLATE)
&& (pItem->dwHash & HASH_BIT_REDIR))
{
DWORD cbUrl = strlen (pszUrl);
DWORD cbKey = strlen (pszKey);
INET_ASSERT (cbUrl && pszUrl[cbUrl - 1] == '/');
if (cbUrl == (cbKey + 1) && !memcmp (pszUrl, pszKey, cbKey))
return pItem;
}
return NULL;
}
case SIG_REDIR:
{
// When online, filter out offline redirect entries.
if (dwFlags == LOOKUP_URL_NOCREATE)
return NULL;
// Check that redirect URL matches exactly.
REDIR_FILEMAP_ENTRY* pRedir = (REDIR_FILEMAP_ENTRY *) pEntry;
if (lstrcmp (pszKey, pRedir->szUrl))
return NULL;
switch (dwFlags)
{
case LOOKUP_URL_CREATE:
// We are creating a new entry for a URL that once
// redirected. Delete the stale redirect entry.
pmmf->FreeEntry (pRedir);
pItem->MarkFree();
return NULL;
case LOOKUP_REDIR_CREATE:
// Return the redirect item if we're looking for it.
return pItem;
case LOOKUP_URL_TRANSLATE:
// Otherwise, translate through the redirect item.
pItem = (HASH_ITEM *)
(*pmmf->GetHeapStart() + pRedir->dwItemOffset);
// Perform some consistency checks.
if (pItem->dwHash & HASH_BIT_NOTURL)
return NULL; // not an URL entry
if ((pItem->dwHash & ~SLOT_MASK) != pRedir->dwHashValue)
return NULL; // not a matching URL entry
return pItem;
default:
INET_ASSERT (FALSE);
}
}
default:
{
// Fix up a bad hash table entry. This can happen if a thread
// died between allocating a hash table item and setting the offset.
pItem->MarkFree();
return NULL;
}
}
}
//
// HashFindItem: finds a matching entry or else the first free slot
//
BOOL URL_CONTAINER::HashFindItem
(LPCSTR pszKey, DWORD dwFlags, HASH_ITEM** ppItem)
{
INET_ASSERT(!((dwFlags & LOOKUP_URL_DONT_FOLLOW) && (dwFlags & LOOKUP_BIT_CREATE)));
DWORD dwFind = 0;
BOOL fLookAgain = !(dwFlags & LOOKUP_URL_DONT_FOLLOW) && GlobalIdentity;
again:
LPVOID pAllocObj = (LPVOID) _UrlObjStorage;
LPBYTE pBase = *_UrlObjStorage->GetHeapStart();
LPDWORD pdwTableOffset = _UrlObjStorage->GetPtrToHashTableOffset();
// Scan flags.
BOOL fCreate = dwFlags & LOOKUP_BIT_CREATE;
HASH_ITEM* pFree = NULL;
DWORD nBlock = 0;
// Hash the URL and calculate the slot.
DWORD dwHash = HashKey(pszKey, dwFind);
DWORD iSlot = dwHash & SLOT_MASK;
dwHash &= ~SLOT_MASK;
// Walk through the list of hash tables.
while (*pdwTableOffset && !_UrlObjStorage->IsBadOffset(*pdwTableOffset))
{
// Calculate offset to next hash table and validate signature.
HASH_FILEMAP_ENTRY* pTable =
(HASH_FILEMAP_ENTRY*) (pBase + *pdwTableOffset);
if (pTable->dwSig != SIG_HASH || pTable->nBlock != nBlock++)
break;
// Calculate offset to bucket in this table.
HASH_ITEM* pItem = ((HASH_ITEM*) (pTable + 1)) + iSlot * ITEMS_PER_BUCKET;
// Scan the bucket.
for (DWORD iSeat=0; iSeat<ITEMS_PER_BUCKET; iSeat++, pItem++)
{
// No reserved bits should ever be set on an item.
INET_ASSERT (!(pItem->dwHash & HASH_BIT_RESERVED));
switch (pItem->dwHash)
{
case HASH_FREE: // free item but more items might follow
{
INET_ASSERT (!(pItem->dwHash & ~SLOT_MASK));
// If caller wants a free item, record the first one we find.
if (!pFree && fCreate)
pFree = pItem;
}
continue;
case HASH_END: // first previously unused free item; no more to follow
{
INET_ASSERT (!(pItem->dwHash & ~SLOT_MASK));
if (!fCreate)
*ppItem = NULL;
else
{
// Hand out the first free slot.
if (pFree)
{
// Invalidate offset in case caller neglects to set it.
pFree->dwOffset = HASH_END;
*ppItem = pFree;
}
else
{
// The first free slot has never been used before.
INET_ASSERT (pItem->dwOffset == HASH_END);
*ppItem = pItem;
}
(*ppItem)->dwHash = dwHash;
}
}
return FALSE;
default:
{
// Check if the key matches.
if (dwHash == (pItem->dwHash & ~SLOT_MASK))
{
if (dwFlags & INTERNET_CACHE_FLAG_ALLOW_COLLISIONS)
{
*ppItem = pItem;
return TRUE;
}
HASH_ITEM* pItem2 = IsMatch(pItem, pszKey, dwFlags);
if (pItem2)
{
LPURL_FILEMAP_ENTRY pEntry =
(URL_FILEMAP_ENTRY*)(*(((MEMMAP_FILE*)_UrlObjStorage)->GetHeapStart())
+ pItem2->dwOffset);
// This will check for a ~U: header or IDENTITY_CACHE_ENTRY
// We use the first for compatibility with Wininet5
// We use IDENTITY_CACHE_ENTRY for entries we wish to hide from Wininet5
if ((pEntry->dwSig==SIG_URL)
&& (!(dwFlags & LOOKUP_URL_DONT_FOLLOW)
&& ((pEntry->CacheEntryType & IDENTITY_CACHE_ENTRY)
|| IsPerUserEntry(pEntry))))
{
// We'll search again for an entry corresponding to GlobalIdentity
if (fLookAgain)
{
fLookAgain = FALSE;
dwFind = GlobalIdentity;
goto again;
}
// Guarantee that this is what we want
if (pEntry->GetIdentity()!=GlobalIdentity)
{
continue;
}
// If we're looking for an identity-0 cache entry
// and there is no filename, we need to do the following:
// 1. if we're trying to create an entry, then return this
// 2. otherwise, say No, the entry is not present
if (!pEntry->InternalFileNameOffset && !fCreate)
{
*ppItem = NULL;
return FALSE;
}
}
*ppItem = pItem2;
return TRUE;
}
}
}
continue;
} // end switch
} // end for loop to scan seats in bucket
// Follow the link to the next table.
pdwTableOffset = &pTable->dwNext;
} // end while (*pdwTableOffset)
// If we've encountered a corrupt table, we'll have to recover
if (*pdwTableOffset)
{
INET_ASSERT(FALSE);
*pdwTableOffset = 0;
}
// We are out a buckets, so an item hasn't been found.
if (fCreate && !pFree)
{
// Caller wanted a free item but we didn't find one.
HASH_FILEMAP_ENTRY* pTable = AllocTable
(pAllocObj, &pBase, &pdwTableOffset);
//////////////////////////////////////////////////////////////////////
// WARNING: the file might have grown and remapped, so any pointers //
// hereafter must be recalculated by offsets from the new base. //
//////////////////////////////////////////////////////////////////////
if (pTable)
{
pTable->nBlock = nBlock;
// Calculate next free slot.
pFree = ((HASH_ITEM*) (pTable + 1)) + iSlot * ITEMS_PER_BUCKET;
INET_ASSERT (pFree->dwHash == HASH_END);
INET_ASSERT (pFree->dwOffset == HASH_END);
}
}
// Return free item if desired and indicate no item found.
if (pFree)
{
INET_ASSERT (fCreate);
pFree->dwHash = dwHash;
pFree->dwOffset = HASH_END; // invalid in case caller neglects to set it
}
*ppItem = pFree;
return FALSE;
}
//
// HashFindNextItem: scans the table for the next valid URL item
//
PUBLIC
HASH_ITEM*
HashGetNextItem
(
IN LPVOID pAllocObj, // allocator object
IN LPBYTE pBase, // base for all offsets
IN OUT LPDWORD pdwItemOffset, // current item offset
IN DWORD dwFlags // include redirects?
)
{
INET_ASSERT (!(dwFlags & ~LOOKUP_BIT_REDIR));
// Check if there if the hash table is empty (or we are at the end already.)
if (!*pdwItemOffset)
return NULL;
HASH_ITEM* pItem = (HASH_ITEM*) (pBase + *pdwItemOffset);
// Calculate current table offset, assuming it's the previous page boundary.
INET_ASSERT (BYTES_PER_TABLE <= BYTES_PER_PAGE);
HASH_FILEMAP_ENTRY* pTable =
(HASH_FILEMAP_ENTRY*) (((DWORD_PTR)pItem) & ~(BYTES_PER_PAGE - 1));
// Advance item pointer to next location.
if (pItem == (HASH_ITEM*) pTable)
pItem = (HASH_ITEM*) (pTable + 1); // first location in table
else
pItem++; // next location in table
do // Scan the list of tables.
{
if (pTable->dwSig != SIG_HASH)
break;
// Scan the current table.
for (; (LPBYTE) pItem < ((LPBYTE) pTable) + BYTES_PER_TABLE; pItem++)
{
// No reserved bits should be set.
INET_ASSERT (!(pItem->dwHash & HASH_BIT_RESERVED));
if (!(pItem->dwHash & HASH_BIT_NOTURL)
|| (dwFlags /* & LOOKUP_BIT_REDIR */)
&& ((pItem->dwHash & HASH_FLAG_MASK) == HASH_REDIR))
{
// Found a valid entry.
*pdwItemOffset = (DWORD) ((LPBYTE)pItem - pBase); // 64BIT
return pItem;
}
}
// Follow the link to the next table.
if (!pTable->dwNext)
pTable = NULL;
else
{
// Validate the table signature and sequence number.
DWORD nBlock = pTable->nBlock;
pTable = (HASH_FILEMAP_ENTRY*) (pBase + pTable->dwNext);
if (pTable->dwSig != SIG_HASH || pTable->nBlock != nBlock + 1)
pTable = NULL;
// Set pointer to first location in table.
pItem = (HASH_ITEM*) (pTable + 1);
}
}
while (pTable);
// We reached the end of the last table.
*pdwItemOffset = 0;
return NULL;
}