1665 lines
45 KiB
C++
1665 lines
45 KiB
C++
/*++
|
|
Copyright (c) 1994 Microsoft Corporation
|
|
|
|
Module Name:
|
|
filemap.cxx
|
|
|
|
Abstract:
|
|
contains implementation of MEMMAP_FILE class.
|
|
|
|
Author:
|
|
Madan Appiah (madana) 28-April-1995
|
|
|
|
Environment:
|
|
User Mode - Win32
|
|
|
|
Revision History:
|
|
|
|
Shishir Pardikar (shishirp) added: (as of (7/6/96)
|
|
|
|
1) Fix crossproces problems on win95 in checksizegrowandremap
|
|
2) Exception handling to deal with badsector being memorymapped
|
|
3) More robust validation at init time
|
|
4) Reinitialization code to really clear the cache
|
|
5) Bug fixes in GrowMap while growing partially filled dword
|
|
--*/
|
|
|
|
#include <cache.hxx>
|
|
|
|
|
|
#define FILE_SIZE_MAX_DIGITS 16
|
|
|
|
|
|
DWORD ValidateAndCreatePath(LPTSTR PathName)
|
|
{
|
|
DWORD Error, len;
|
|
DWORD FileAttribute;
|
|
LPTSTR PathDelimit;
|
|
|
|
// check to see the path specified is there.
|
|
FileAttribute = GetFileAttributes(PathName);
|
|
if (FileAttribute != 0xFFFFFFFF) {
|
|
// check to see the attribute says it is a dir.
|
|
if (!(FileAttribute & FILE_ATTRIBUTE_DIRECTORY)) {
|
|
return(ERROR_INVALID_PARAMETER);
|
|
}
|
|
|
|
// We found the file and it is a dir.
|
|
// Set the system attribute just in case
|
|
// it has been unset.
|
|
SetFileAttributes(PathName, FILE_ATTRIBUTE_SYSTEM);
|
|
return(ERROR_SUCCESS);
|
|
}
|
|
|
|
Error = GetLastError();
|
|
|
|
if ((Error != ERROR_FILE_NOT_FOUND) && (Error != ERROR_PATH_NOT_FOUND)) {
|
|
return(Error);
|
|
}
|
|
|
|
// we did not find the path, so create it.
|
|
if (CreateDirectory(PathName, NULL)) {
|
|
// done.
|
|
SetFileAttributes(PathName, FILE_ATTRIBUTE_SYSTEM);
|
|
return(ERROR_SUCCESS);
|
|
}
|
|
|
|
Error = GetLastError();
|
|
if (Error != ERROR_PATH_NOT_FOUND) {
|
|
return(Error);
|
|
}
|
|
|
|
// sub-path is not found, create it first.
|
|
len = lstrlen(PathName);
|
|
if (len < 5) {
|
|
SetLastError(ERROR_INVALID_NAME);
|
|
return (ERROR_INVALID_NAME);
|
|
}
|
|
|
|
PathDelimit = PathName + len - 1;
|
|
|
|
// step back from the trailing backslash
|
|
if (*PathDelimit == PATH_CONNECT_CHAR) {
|
|
PathDelimit--;
|
|
}
|
|
|
|
// find the last path delimiter.
|
|
while (PathDelimit > PathName) {
|
|
if (*PathDelimit == PATH_CONNECT_CHAR) {
|
|
break;
|
|
}
|
|
|
|
PathDelimit--;
|
|
}
|
|
|
|
if (PathDelimit == PathName) {
|
|
return(ERROR_INVALID_PARAMETER);
|
|
}
|
|
|
|
*PathDelimit = TEXT('\0');
|
|
|
|
// validate sub-path now.
|
|
Error = ValidateAndCreatePath(PathName);
|
|
|
|
// replace the connect char anyway.
|
|
*PathDelimit = PATH_CONNECT_CHAR;
|
|
|
|
if (Error != ERROR_SUCCESS) {
|
|
return(Error);
|
|
}
|
|
|
|
// try to create one more time.
|
|
if (CreateDirectory(PathName, NULL)) {
|
|
// done.
|
|
return(ERROR_SUCCESS);
|
|
}
|
|
|
|
Error = GetLastError();
|
|
return(Error);
|
|
}
|
|
|
|
|
|
DWORD MEMMAP_FILE::CheckSizeGrowAndRemapAddress(VOID)
|
|
{
|
|
DWORD dwNewFileSize;
|
|
|
|
#ifdef WIN95_BUG
|
|
if (_FileSize == (dwNewFileSize = _HeaderInfo->FileSize)) {
|
|
return(ERROR_SUCCESS);
|
|
}
|
|
#endif //WIN95_BUG
|
|
|
|
// ideally we would have liked to do as in the above two lines
|
|
// this works right on NT but doesn't on win95.
|
|
|
|
// This is because the filesize is a part of the mapname
|
|
// In the initial state the index file size is 8192. So the
|
|
// memorymap name is c:_windows_temporaray internet files_8192.
|
|
// Both the processes have this map in their address space.
|
|
// Process B starts pumping in the data, and at some point the index
|
|
// file needs to be grown. Process B, increases the index file to 16384,
|
|
// updates the filesize in the header "while it is still mapped in the
|
|
// map corresponding to the old filesize" and then remaps to the new map
|
|
// with the name c:_windows_temporaray internet files_16384.
|
|
// Any subsequent growth is now recorded in this map.
|
|
// The old map c:_windows_temporaray internet files_8192 still has only the first transition.
|
|
|
|
// the work around is to actually get the filesize from the filesystem
|
|
// This works correctly on win95 and NT both. Optimally, we would
|
|
// check for a transition and then get the real size, but we will do that after IE30 ships.
|
|
|
|
//NB!!!!!!! The check below is the basis of our cross process
|
|
// cache. All APIs finally make this call before touching the memory
|
|
// mapped file. If there is a chneg, they remap it to the new size
|
|
// with the sizename as part of the mapping, so they get the latest
|
|
// stuff.
|
|
// When anyone gets here, they are protected by a crossprocess mutex
|
|
|
|
if (_FileSize == (dwNewFileSize = GetFileSize(_FileHandle, NULL))) {
|
|
return(ERROR_SUCCESS);
|
|
}
|
|
|
|
// so other user of the memmap file has increased the file size,
|
|
// let's remap our address space so that the new portion is visible to us too.
|
|
|
|
DWORD Error;
|
|
DWORD OldFileSize;
|
|
DWORD OldNumBitMapDWords;
|
|
|
|
// set our internal file size and num bit map entries.
|
|
|
|
OldFileSize = _FileSize;
|
|
OldNumBitMapDWords = _NumBitMapDWords;
|
|
|
|
_FileSize = dwNewFileSize;
|
|
|
|
Error = RemapAddress();
|
|
if (Error != ERROR_SUCCESS) {
|
|
// reset the file size.
|
|
_FileSize = OldFileSize;
|
|
_NumBitMapDWords = OldNumBitMapDWords;
|
|
}
|
|
else {
|
|
#if INET_DEBUG
|
|
if ((GetFileSize(_FileHandle, NULL)) != (_HeaderInfo->FileSize)) {
|
|
TcpsvcsDbgPrint((DEBUG_FILE_VALIDATE, "GetFileSize!= (_FileSize)\n"));
|
|
TcpsvcsDbgAssert(FALSE);
|
|
}
|
|
#endif //INET_DEBG
|
|
|
|
_NumBitMapDWords = (_HeaderInfo->NumUrlInternalEntries + (NUM_BITS_IN_DWORD - 1)) / NUM_BITS_IN_DWORD; // cell
|
|
}
|
|
|
|
return(Error);
|
|
}
|
|
|
|
|
|
BOOL MEMMAP_FILE::ValidateCache(VOID)
|
|
/*++
|
|
This private member function validates the cache file content.
|
|
Arguments:
|
|
NONE.
|
|
Return Value:
|
|
TRUE - if the cache is valid.
|
|
FALSE - otherwise.
|
|
--*/
|
|
{
|
|
BOOL ReturnCode = FALSE;
|
|
int i, k;
|
|
DWORD BitPosition, TotalAlloced, MaxAllocedPosition, RunningCounter;
|
|
|
|
__try {
|
|
// validate signatue.
|
|
if (memcmp(
|
|
_HeaderInfo->FileSignature,
|
|
CACHE_SIGNATURE,
|
|
MAX_SIG_SIZE * sizeof(TCHAR)) != 0) {
|
|
|
|
TcpsvcsDbgPrint((DEBUG_FILE_VALIDATE, "File signature does not match.\n"));
|
|
goto Cleanup;
|
|
}
|
|
|
|
// Also check the index does not contain entries with a higher
|
|
// version than the current machine can handle. This can happen
|
|
// due to Windows kludgy concept of roaming, which replicates parts of the file system and registry hkcu.
|
|
|
|
LPSTR pszHighVer = (LPSTR)(_HeaderInfo->dwHeaderData + CACHE_HEADER_DATA_HIGH_VERSION_STRING);
|
|
if (pszHighVer[0] != 'V' || pszHighVer[3] != 0)
|
|
memset(pszHighVer, 0, sizeof(DWORD));
|
|
else if (!g_szFixup[0] || strcmp(g_szFixup, pszHighVer) < 0)
|
|
{
|
|
TcpsvcsDbgPrint((DEBUG_FILE_VALIDATE, "Cannot handle uplevel index file.\n"));
|
|
goto Cleanup;
|
|
}
|
|
|
|
// check the hash table root offset is valid
|
|
if (_HeaderInfo->dwHashTableOffset != 0) {
|
|
if (_HeaderInfo->dwHashTableOffset > _FileSize) {
|
|
TcpsvcsDbgPrint((DEBUG_FILE_VALIDATE, "invalid b-tree root offset.\n"));
|
|
goto Cleanup;
|
|
}
|
|
}
|
|
|
|
// check file size.
|
|
if (_HeaderInfo->FileSize != _FileSize) {
|
|
TcpsvcsDbgPrint((DEBUG_FILE_VALIDATE, "invalid file size.\n"));
|
|
goto Cleanup;
|
|
}
|
|
|
|
// one more file size check.
|
|
DWORD ExpectedFileSize;
|
|
ExpectedFileSize = HEADER_ENTRY_SIZE + _HeaderInfo->NumUrlInternalEntries * _EntrySize;
|
|
|
|
// cell the size to GlobalMapFileGrowSize.
|
|
if (ExpectedFileSize % GlobalMapFileGrowSize) {
|
|
ExpectedFileSize = ((ExpectedFileSize / GlobalMapFileGrowSize) + 1) * GlobalMapFileGrowSize;
|
|
}
|
|
|
|
if (_FileSize != ExpectedFileSize) {
|
|
// it is ok if the file size is one block bigger.
|
|
TcpsvcsDbgPrint((DEBUG_FILE_VALIDATE, "Invalid file size.\n"));
|
|
goto Cleanup;
|
|
}
|
|
|
|
if (_HeaderInfo->NumUrlInternalEntries < _HeaderInfo->NumUrlEntriesAlloced) {
|
|
TcpsvcsDbgPrint((DEBUG_FILE_VALIDATE, "Invalid alloc entires.\n"));
|
|
goto Cleanup;
|
|
}
|
|
|
|
TotalAlloced = 0;
|
|
MaxAllocedPosition = 0;
|
|
RunningCounter = 0;
|
|
|
|
// scan the enire bitmap and do some consistency check for allocated bits
|
|
for (i = 0; i < BIT_MAP_ARRAY_SIZE; ++i) {
|
|
// k goes from 0 to 31
|
|
// BitPosition goes from 0x00000001 to 0x80000000
|
|
for (BitPosition = 1, k = 0; k < NUM_BITS_IN_DWORD; ++k, BitPosition <<= 1) {
|
|
++RunningCounter;
|
|
if (_HeaderInfo->AllocationBitMap[i] & BitPosition) {
|
|
++TotalAlloced;
|
|
MaxAllocedPosition = RunningCounter;
|
|
}
|
|
}
|
|
}
|
|
|
|
// if the max allocated bit is greter than the number of possible entries for this filesize,
|
|
// or the total allocated bits are greater (the above condition subsumes this one, but it is OK to be paranoid)
|
|
// or totalbits alloced don't match the count there this header is not OK
|
|
|
|
if ((MaxAllocedPosition > _HeaderInfo->NumUrlInternalEntries)
|
|
|| (TotalAlloced > _HeaderInfo->NumUrlInternalEntries)
|
|
|| (TotalAlloced != _HeaderInfo->NumUrlEntriesAlloced)) {
|
|
TcpsvcsDbgPrint((DEBUG_FILE_VALIDATE, "Invalid alloc bitmap\n"));
|
|
goto Cleanup;
|
|
}
|
|
|
|
// every thing is fine.
|
|
ReturnCode = TRUE;
|
|
}
|
|
__except (EXCEPTION_EXECUTE_HANDLER) {
|
|
ReturnCode = FALSE;
|
|
_Status = ERROR_WRITE_FAULT;
|
|
}
|
|
ENDEXCEPT
|
|
|
|
Cleanup :
|
|
if (ReturnCode == FALSE) {
|
|
TcpsvcsDbgPrint((DEBUG_FILE_VALIDATE, "Invalid Cache, or bad disk\n"));
|
|
}
|
|
|
|
return(ReturnCode);
|
|
}
|
|
|
|
void MEMMAP_FILE::CloseMapping(void)
|
|
{
|
|
if (_BaseAddr) // view
|
|
{
|
|
UnmapViewOfFile(_BaseAddr);
|
|
_BaseAddr = NULL;
|
|
}
|
|
if (_FileMappingHandle) // mapping
|
|
{
|
|
CloseHandle(_FileMappingHandle);
|
|
_FileMappingHandle = NULL;
|
|
}
|
|
if (_FileHandle) // file
|
|
{
|
|
CloseHandle(_FileHandle);
|
|
_FileHandle = NULL;
|
|
}
|
|
}
|
|
|
|
|
|
DWORD MEMMAP_FILE::RemapAddress(VOID)
|
|
/*++
|
|
This private member function remaps the memory mapped file just after the file size has been modified.
|
|
Container must be locked when this function is called.
|
|
Return Value:
|
|
Windows Error Code.
|
|
--*/
|
|
{
|
|
DWORD Error = ERROR_SUCCESS;
|
|
PVOID OldBaseAddr;
|
|
DWORD OldViewSize;
|
|
PVOID VirtualBase;
|
|
BOOL BoolError;
|
|
LPTSTR MapName = NULL;
|
|
|
|
CloseMapping();
|
|
|
|
// Create/Open memory mapped file.
|
|
_FileHandle = CreateFile(
|
|
_FileName,
|
|
GENERIC_READ | GENERIC_WRITE,
|
|
FILE_SHARE_READ | FILE_SHARE_WRITE,
|
|
// share this file with others while it is being used.
|
|
CreateAllAccessSecurityAttributes(NULL, NULL, NULL),
|
|
OPEN_ALWAYS,
|
|
FILE_FLAG_RANDOM_ACCESS,
|
|
NULL);
|
|
if (_FileHandle == INVALID_HANDLE_VALUE) {
|
|
Error = _Status = GetLastError();
|
|
TcpsvcsDbgPrint((DEBUG_ERRORS, "Reinitialize:File open failed, %ld.\n", _Status));
|
|
TcpsvcsDbgAssert(FALSE);
|
|
_FileHandle = NULL;
|
|
goto Cleanup;
|
|
}
|
|
|
|
#ifndef unix
|
|
/**
|
|
* UNIX:
|
|
* Mainwin does not support MapName in CreateFileMapping API
|
|
* Let us leave the MapName as NULL till this functionality
|
|
* is available.
|
|
*/
|
|
|
|
// make a map name.
|
|
DWORD MapNameSize;
|
|
MapNameSize = (lstrlen(_FullPathName) + lstrlen(_FileName) + 1 + FILE_SIZE_MAX_DIGITS) * sizeof(TCHAR);
|
|
MapName = (LPTSTR)CacheHeap->Alloc(MapNameSize);
|
|
if (MapName == NULL) {
|
|
Error = ERROR_NOT_ENOUGH_MEMORY;
|
|
goto Cleanup;
|
|
}
|
|
|
|
memcpy(MapName, _FileName, _FullPathNameLen + sizeof(MEMMAP_FILE_NAME));
|
|
memcpy(MapName + _FullPathNameLen + sizeof(MEMMAP_FILE_NAME) - 1, DIR_SEPARATOR_STRING, sizeof(DIR_SEPARATOR_STRING));
|
|
wsprintf(MapName + lstrlen(MapName), "%u", _FileSize);
|
|
|
|
#ifndef unix
|
|
#define BACKSLASH_CHAR TEXT('\\')
|
|
#else
|
|
#define BACKSLASH_CHAR TEXT('/')
|
|
#endif /* unix */
|
|
#define UNDERSCORE_CHAR TEXT('_')
|
|
#define TERMINATING_CHAR TEXT('\0')
|
|
|
|
LPTSTR ScanMapName;
|
|
|
|
|
|
// Replace '\' with '_'.
|
|
|
|
ScanMapName = MapName;
|
|
|
|
while (*ScanMapName != TERMINATING_CHAR) {
|
|
if (*ScanMapName == BACKSLASH_CHAR) {
|
|
*ScanMapName = UNDERSCORE_CHAR;
|
|
}
|
|
|
|
ScanMapName++;
|
|
}
|
|
#endif /* !unix */
|
|
|
|
// re-create memory mapping.
|
|
_FileMappingHandle = OpenFileMapping(FILE_MAP_WRITE, FALSE, MapName);
|
|
if (_FileMappingHandle == NULL && (GetLastError() == ERROR_FILE_NOT_FOUND || GetLastError() == ERROR_INVALID_NAME))
|
|
{
|
|
_FileMappingHandle = CreateFileMapping(
|
|
_FileHandle,
|
|
CreateAllAccessSecurityAttributes(NULL, NULL, NULL),
|
|
PAGE_READWRITE,
|
|
0, // high dword of max memory mapped file size.
|
|
#if defined(UNIX) && defined(ux10)
|
|
1024 * 1024, // map entire file.
|
|
#else
|
|
0, // map entire file.
|
|
#endif
|
|
MapName);
|
|
}
|
|
|
|
if (_FileMappingHandle == NULL) {
|
|
Error = _Status = GetLastError();
|
|
goto Cleanup;
|
|
}
|
|
|
|
// remap view region.
|
|
_BaseAddr = MapViewOfFileEx(_FileMappingHandle, FILE_MAP_WRITE, 0, 0, #if defined(UNIX) && defined(ux10) 1024 * 1024, // MAP entire file.
|
|
#else
|
|
0, // MAP entire file.
|
|
#endif
|
|
NULL);
|
|
|
|
#if defined(UNIX) && defined(ux10)
|
|
DWORD FilePointer = SetFilePointer(_FileHandle, _FileSize, NULL, FILE_BEGIN);
|
|
if (FilePointer == 0xFFFFFFFF)
|
|
{
|
|
Error = _Status = GetLastError();
|
|
goto Cleanup;
|
|
}
|
|
|
|
BoolError = SetEndOfFile(_FileHandle);
|
|
if (BoolError == FALSE)
|
|
{
|
|
Error = _Status = GetLastError();
|
|
goto Cleanup;
|
|
}
|
|
#endif
|
|
|
|
if (_BaseAddr == NULL)
|
|
{
|
|
Error = _Status = GetLastError();
|
|
TcpsvcsDbgAssert(FALSE);
|
|
|
|
TcpsvcsDbgPrint((DEBUG_ERRORS, "MapViewOfFile failed to extend address space, %ld.\n", Error));
|
|
goto Cleanup;
|
|
}
|
|
|
|
// reset other pointers.
|
|
_HeaderInfo = (LPMEMMAP_HEADER)_BaseAddr;
|
|
_EntryArray = ((LPBYTE)_BaseAddr + HEADER_ENTRY_SIZE);
|
|
|
|
_Status = Error = ERROR_SUCCESS;
|
|
|
|
Cleanup:
|
|
if (MapName != NULL) {
|
|
CacheHeap->Free(MapName);
|
|
}
|
|
|
|
return(Error);
|
|
}
|
|
|
|
|
|
DWORD MEMMAP_FILE::GrowMapFile(DWORD dwMapFileGrowSize)
|
|
/*++
|
|
This private member function extends the memory mapped file and creates more free url store entries.
|
|
Arguments:
|
|
NONE.
|
|
Return Value:
|
|
Windows Error Code.
|
|
--*/
|
|
{
|
|
DWORD Error, i;
|
|
BOOL BoolError;
|
|
DWORD FilePointer;
|
|
DWORD OldNumUrlInternalEntries;
|
|
char buff[PAGE_SIZE];
|
|
|
|
// check to see that we have reached the limit.
|
|
// we can hold only MAX_URL_ENTRIES url entries.
|
|
// so the file size can grow more than
|
|
|
|
// HEADER_ENTRY_SIZE + MAX_URL_ENTRIES * _EntrySize
|
|
|
|
#if INET_DEBUG
|
|
if (GetFileSize(_FileHandle, NULL) != (_FileSize)) {
|
|
TcpsvcsDbgPrint((DEBUG_FILE_VALIDATE, "GetFileSize!= (_FileSize)\n"));
|
|
TcpsvcsDbgAssert(FALSE);
|
|
}
|
|
#endif //INET_DEBG
|
|
|
|
//BUGBUG - need to fix this
|
|
if ((_FileSize + dwMapFileGrowSize) >= (HEADER_ENTRY_SIZE + MAX_URL_ENTRIES * _EntrySize)) {
|
|
// best matching error code.
|
|
Error = ERROR_NOT_ENOUGH_MEMORY;
|
|
goto Cleanup;
|
|
}
|
|
|
|
FilePointer = SetFilePointer(_FileHandle, dwMapFileGrowSize, NULL, FILE_END);
|
|
|
|
if (FilePointer != (_FileSize + dwMapFileGrowSize))
|
|
{
|
|
TcpsvcsDbgPrint((DEBUG_FILE_VALIDATE, "FilePointer != (_FileSize + dwMapFileGrowSize)\n"));
|
|
|
|
TcpsvcsDbgAssert(FALSE);
|
|
|
|
_Status = GetLastError();
|
|
Error = _Status;
|
|
|
|
goto Cleanup;
|
|
}
|
|
|
|
if (FilePointer == 0xFFFFFFFF) {
|
|
Error = GetLastError();
|
|
TcpsvcsDbgAssert(FALSE);
|
|
goto Cleanup;
|
|
}
|
|
|
|
BoolError = SetEndOfFile(_FileHandle);
|
|
|
|
if (BoolError != TRUE) {
|
|
Error = GetLastError();
|
|
TcpsvcsDbgAssert(FALSE);
|
|
goto Cleanup;
|
|
}
|
|
|
|
#if INET_DEBUG
|
|
if (GetFileSize(_FileHandle, NULL) != (_FileSize + dwMapFileGrowSize)) {
|
|
TcpsvcsDbgPrint((DEBUG_FILE_VALIDATE, "GetFileSize!= (_FileSize + dwMapFileGrowSize)\n"));
|
|
TcpsvcsDbgAssert(FALSE);
|
|
}
|
|
#endif
|
|
|
|
// adjust internal size parameters.
|
|
_FileSize += dwMapFileGrowSize;
|
|
|
|
// also set the new file size in the memory mapped file so that
|
|
// other user will remap their address space and view the new portion.
|
|
|
|
_HeaderInfo->FileSize = _FileSize;
|
|
|
|
OldNumUrlInternalEntries = _HeaderInfo->NumUrlInternalEntries;
|
|
_HeaderInfo->NumUrlInternalEntries += dwMapFileGrowSize / _EntrySize;
|
|
_NumBitMapDWords = (_HeaderInfo->NumUrlInternalEntries + (NUM_BITS_IN_DWORD - 1)) / NUM_BITS_IN_DWORD; // cell
|
|
|
|
// remap
|
|
|
|
Error = RemapAddress();
|
|
|
|
if (Error != ERROR_SUCCESS) {
|
|
goto Cleanup;
|
|
}
|
|
|
|
memset((_EntryArray + _EntrySize * OldNumUrlInternalEntries), 0, dwMapFileGrowSize);
|
|
Error = ERROR_SUCCESS;
|
|
|
|
Cleanup:
|
|
return(Error);
|
|
}
|
|
|
|
|
|
BOOL MEMMAP_FILE::CheckNextNBits(DWORD& nArrayIndex, DWORD &dwStartMask, DWORD nBitsRequired, DWORD& nBitsFound)
|
|
{
|
|
/*++
|
|
Determines if the next N bits are unset.
|
|
|
|
Arguments:
|
|
[IN/OUT]
|
|
DWORD &nArrayIndex, DWORD &dwMask
|
|
|
|
[IN]
|
|
DWORD nBitsRequired
|
|
|
|
[OUT]
|
|
DWORD &nBitsFound
|
|
|
|
Return Value:
|
|
|
|
TRUE if the next N bits were found unset.
|
|
FALSE otherwise.
|
|
|
|
Notes:
|
|
This function assumes that the range of bits to be checked lie
|
|
within a valid area of the bit map.
|
|
--*/
|
|
DWORD i, j;
|
|
DWORD nIdx = nArrayIndex;
|
|
DWORD dwMask = dwStartMask;
|
|
BOOL fFound = FALSE;
|
|
LPDWORD BitMap = &_HeaderInfo->AllocationBitMap[nIdx];
|
|
|
|
nBitsFound = 0;
|
|
|
|
// Check if the next nBitsRequired bits are unset
|
|
for (i = 0; i < nBitsRequired; i++)
|
|
{
|
|
// Is this bit unset?
|
|
if ((*BitMap & dwMask) == 0)
|
|
{
|
|
// Have sufficient unset bits been found?
|
|
if (++nBitsFound == nBitsRequired)
|
|
{
|
|
// Found sufficient bits. Success.
|
|
fFound = TRUE;
|
|
goto exit;
|
|
}
|
|
}
|
|
|
|
// Ran into a set bit. Fail.
|
|
else
|
|
{
|
|
// Indicate the array and bit index
|
|
// of the set bit encountered.
|
|
nArrayIndex = nIdx;
|
|
dwStartMask = dwMask;
|
|
goto exit;
|
|
}
|
|
|
|
// Left rotate the bit mask.
|
|
dwMask <<= 1;
|
|
if (dwMask == 0x0)
|
|
{
|
|
dwMask = 0x1;
|
|
BitMap = &_HeaderInfo->AllocationBitMap[++nIdx];
|
|
}
|
|
|
|
} // Loop nBitsRequired times.
|
|
|
|
|
|
exit:
|
|
return fFound;
|
|
}
|
|
|
|
|
|
BOOL MEMMAP_FILE::SetNextNBits(DWORD nIdx, DWORD dwMask, DWORD nBitsRequired)
|
|
/*++
|
|
Given an array index and bit mask, sets the next N bits.
|
|
|
|
Arguments:
|
|
[IN]
|
|
DWORD nIdx, DWORD dwMask, DWORD nBitsRequired
|
|
Return Value:
|
|
TRUE if the next N bits were found unset, and successfully set.
|
|
FALSE if unable to set all the required bits.
|
|
Notes:
|
|
This function assumes that the range of bits to be set lie
|
|
within a valid area of the bit map. If the function returns
|
|
false, no bits are set.
|
|
--*/
|
|
{
|
|
DWORD i, j, nBitsSet = 0;
|
|
LPDWORD BitMap = &_HeaderInfo->AllocationBitMap[nIdx];
|
|
BitMap = &_HeaderInfo->AllocationBitMap[nIdx];
|
|
|
|
for (i = 0; i < nBitsRequired; i++)
|
|
{
|
|
// Check that this bit is not already set.
|
|
if (*BitMap & dwMask)
|
|
{
|
|
INET_ASSERT(FALSE);
|
|
|
|
// Fail. Unset the bits we just set and exit.
|
|
for (j = nBitsSet; j > 0; j--)
|
|
{
|
|
INET_ASSERT((*BitMap & dwMask) == 0);
|
|
|
|
// Right rotate the bit mask.
|
|
dwMask >>= 1;
|
|
if (dwMask == 0x0)
|
|
{
|
|
dwMask = 0x80000000;
|
|
BitMap = &_HeaderInfo->AllocationBitMap[--nIdx];
|
|
}
|
|
*BitMap &= ~dwMask;
|
|
}
|
|
return FALSE;
|
|
}
|
|
|
|
*BitMap |= dwMask;
|
|
nBitsSet++;
|
|
|
|
// Left rotate the bit mask.
|
|
dwMask <<= 1;
|
|
if (dwMask == 0x0)
|
|
{
|
|
dwMask = 0x1;
|
|
BitMap = &_HeaderInfo->AllocationBitMap[++nIdx];
|
|
}
|
|
}
|
|
|
|
// Success.
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
DWORD MEMMAP_FILE::GetAndSetNextFreeEntry(DWORD nBitsRequired)
|
|
/*++
|
|
This private member function computes the first available free entry index.
|
|
Arguments:
|
|
DWORD nBitsRequired
|
|
Return Value:
|
|
Next available free entry Index.
|
|
--*/
|
|
{
|
|
DWORD i, nReturnBit = 0xFFFFFFFF;
|
|
|
|
// Align if 4k or greater
|
|
BOOL fAlign = (nBitsRequired >= NUM_BITS_IN_DWORD ? TRUE : FALSE);
|
|
|
|
// Scan DWORDS from the beginning of the byte array.
|
|
DWORD nArrayIndex = 0;
|
|
while (nArrayIndex < _NumBitMapDWords)
|
|
{
|
|
// Process starting from this DWORD if alignment is not required
|
|
// and there are free bits, or alignment is required and all bits
|
|
// are free.
|
|
if (_HeaderInfo->AllocationBitMap[nArrayIndex] != 0xFFFFFFFF
|
|
&& (!fAlign || (fAlign && _HeaderInfo->AllocationBitMap[nArrayIndex] == 0)))
|
|
{
|
|
DWORD nBitIndex = 0;
|
|
DWORD dwMask = 0x1;
|
|
LPDWORD BitMap = &_HeaderInfo->AllocationBitMap[nArrayIndex];
|
|
|
|
// Find a candidate slot.
|
|
while (nBitIndex < NUM_BITS_IN_DWORD)
|
|
{
|
|
// Found first bit of a candidate slot.
|
|
if ((*BitMap & dwMask) == 0)
|
|
{
|
|
// Calculate leading bit value.
|
|
DWORD nLeadingBit = NUM_BITS_IN_DWORD * nArrayIndex + nBitIndex;
|
|
|
|
// Don't exceed the number of internal entries.
|
|
if (nLeadingBit + nBitsRequired > _HeaderInfo->NumUrlInternalEntries)
|
|
{
|
|
// Overstepped last internal entry
|
|
goto exit;
|
|
}
|
|
|
|
// If we just need one bit, then we're done.
|
|
if (nBitsRequired == 1)
|
|
{
|
|
*BitMap |= dwMask;
|
|
nReturnBit = nLeadingBit;
|
|
_HeaderInfo->NumUrlEntriesAlloced += 1;
|
|
goto exit;
|
|
}
|
|
|
|
// Additional bits required.
|
|
DWORD nBitsFound;
|
|
DWORD nIdx = nArrayIndex;
|
|
|
|
// Check the next nBitsRequired bits. Set them if free.
|
|
if (CheckNextNBits(nIdx, dwMask, nBitsRequired, nBitsFound))
|
|
{
|
|
if (SetNextNBits(nIdx, dwMask, nBitsRequired))
|
|
{
|
|
// Return the offset of the leading bit.
|
|
_HeaderInfo->NumUrlEntriesAlloced += nBitsRequired;
|
|
nReturnBit = nLeadingBit;
|
|
goto exit;
|
|
}
|
|
// Bad news.
|
|
else
|
|
{
|
|
// The bits are free, but we couldn't set them. Fail.
|
|
goto exit;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// This slot has insufficient contiguous free bits.
|
|
// Update the array index. We break back to looping
|
|
// over the bits in the DWORD where the interrupting
|
|
// bit was found.
|
|
nArrayIndex = nIdx;
|
|
nBitIndex = (nBitIndex + nBitsFound) % NUM_BITS_IN_DWORD;
|
|
break;
|
|
}
|
|
|
|
} // Found a free leading bit.
|
|
else
|
|
{
|
|
// Continue looking at bits in this DWORD.
|
|
nBitIndex++;
|
|
dwMask <<= 1;
|
|
}
|
|
|
|
} // Loop over bits in DWORD.
|
|
|
|
} // If we found a candidate DWORD.
|
|
|
|
nArrayIndex++;
|
|
|
|
} // Loop through all DWORDS.
|
|
exit:
|
|
return nReturnBit;
|
|
}
|
|
|
|
|
|
MemMapStatus MEMMAP_FILE::Init(LPTSTR PathName, DWORD EntrySize)
|
|
/*++
|
|
MEMMAP_FILE object constructor.
|
|
Arguments:
|
|
PathName : full path name of the memory mapped file.
|
|
EntrySize : size of the each entry in this container.
|
|
Return Value:
|
|
NONE.
|
|
--*/
|
|
{
|
|
DWORD cb;
|
|
|
|
_EntrySize = EntrySize;
|
|
_FullPathName = NULL;
|
|
_FileName = NULL;
|
|
_FileSize = 0;
|
|
_FileHandle = NULL;
|
|
_FileMappingHandle = NULL;
|
|
_BaseAddr = NULL;
|
|
_HeaderInfo = NULL;
|
|
_EntryArray = NULL;
|
|
_NumBitMapDWords = 0;
|
|
|
|
// Validate the path and create the path if it is not already there.
|
|
_Status = ValidateAndCreatePath(PathName);
|
|
if (_Status != ERROR_SUCCESS)
|
|
goto Cleanup;
|
|
|
|
// Path to memory mapped file.
|
|
cb = strlen(PathName);
|
|
_FullPathName = (LPTSTR)CacheHeap->Alloc(cb + sizeof(DIR_SEPARATOR_STRING));
|
|
|
|
if (_FullPathName == NULL)
|
|
{
|
|
_Status = ERROR_NOT_ENOUGH_MEMORY;
|
|
INET_ASSERT(FALSE);
|
|
goto Cleanup;
|
|
}
|
|
|
|
memcpy(_FullPathName, PathName, cb + 1);
|
|
AppendSlashIfNecessary(_FullPathName, cb);
|
|
|
|
_FullPathNameLen = cb;
|
|
|
|
// Construct memory mapped file name.
|
|
_FileName = (LPTSTR)CacheHeap->Alloc(cb + sizeof(MEMMAP_FILE_NAME));
|
|
if (!_FileName)
|
|
{
|
|
_Status = ERROR_NOT_ENOUGH_MEMORY;
|
|
INET_ASSERT(FALSE);
|
|
goto Cleanup;
|
|
}
|
|
memcpy(_FileName, _FullPathName, cb);
|
|
memcpy(_FileName + cb,
|
|
MEMMAP_FILE_NAME, sizeof(MEMMAP_FILE_NAME));
|
|
|
|
// Create/Open memory mapped file.
|
|
_FileHandle = CreateFile(
|
|
_FileName,
|
|
GENERIC_READ | GENERIC_WRITE,
|
|
FILE_SHARE_READ | FILE_SHARE_WRITE,
|
|
CreateAllAccessSecurityAttributes(NULL, NULL, NULL),
|
|
OPEN_ALWAYS,
|
|
FILE_FLAG_RANDOM_ACCESS,
|
|
NULL);
|
|
_Status = GetLastError();
|
|
if (_FileHandle == INVALID_HANDLE_VALUE)
|
|
{
|
|
_FileHandle = NULL;
|
|
TcpsvcsDbgAssert(FALSE);
|
|
goto Cleanup;
|
|
}
|
|
else
|
|
{
|
|
SetFileTime(_FileHandle, NULL, NULL, (LPFILETIME)&dwdwSessionStartTime);
|
|
}
|
|
|
|
// Check to this file is new.
|
|
if (_Status == ERROR_ALREADY_EXISTS)
|
|
{
|
|
// Old file.
|
|
|
|
_Status = ERROR_SUCCESS;
|
|
_NewFile = FALSE;
|
|
|
|
_FileSize = GetFileSize(_FileHandle, NULL);
|
|
|
|
if (_FileSize == 0xFFFFFFFF)
|
|
{
|
|
_Status = GetLastError();
|
|
TcpsvcsDbgAssert(FALSE);
|
|
goto Cleanup;
|
|
}
|
|
|
|
if ((_FileSize < GlobalMapFileGrowSize) || ((_FileSize %GlobalMapFileGrowSize) != 0))
|
|
{
|
|
TcpsvcsDbgAssert(FALSE);
|
|
if (!Reinitialize())
|
|
{
|
|
TcpsvcsDbgAssert(FALSE);
|
|
SetLastError(_Status);
|
|
goto Cleanup;
|
|
}
|
|
// Reinitialization results in new file.
|
|
_NewFile = TRUE;
|
|
}
|
|
}
|
|
else if (_Status == ERROR_SUCCESS)
|
|
{
|
|
BOOL BoolError;
|
|
DWORD FilePointer;
|
|
|
|
// New file.
|
|
_NewFile = TRUE;
|
|
|
|
// Set initial file size.
|
|
_FileSize = GlobalMapFileGrowSize;
|
|
FilePointer = SetFilePointer(_FileHandle, _FileSize, NULL, FILE_BEGIN);
|
|
|
|
if (FilePointer == 0xFFFFFFFF)
|
|
{
|
|
_Status = GetLastError();
|
|
goto Cleanup;
|
|
}
|
|
|
|
if (FilePointer != _FileSize)
|
|
{
|
|
TcpsvcsDbgPrint((DEBUG_FILE_VALIDATE,
|
|
"FilePointer != (_FileSize)\n"));
|
|
|
|
TcpsvcsDbgAssert(FALSE);
|
|
}
|
|
|
|
BoolError = SetEndOfFile(_FileHandle);
|
|
|
|
if (BoolError != TRUE)
|
|
{
|
|
_Status = GetLastError();
|
|
goto Cleanup;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// We should not reach here.
|
|
TcpsvcsDbgAssert(FALSE);
|
|
}
|
|
_Status = RemapAddress();
|
|
|
|
if (_Status != ERROR_SUCCESS)
|
|
{
|
|
TcpsvcsDbgAssert(FALSE);
|
|
goto Cleanup;
|
|
}
|
|
|
|
TcpsvcsDbgPrint((DEBUG_ERRORS, "Header Size, %ld.\n", HEADER_ENTRY_SIZE));
|
|
TcpsvcsDbgPrint((DEBUG_ERRORS, "Size of elements, %ld.\n", sizeof(MEMMAP_HEADER_SMALL)));
|
|
TcpsvcsDbgPrint((DEBUG_ERRORS, "Bit Array size, %ld.\n", BIT_MAP_ARRAY_SIZE));
|
|
TcpsvcsDbgPrint((DEBUG_ERRORS, "Memmap Header Size, %ld.\n", sizeof(MEMMAP_HEADER)));
|
|
|
|
TcpsvcsDbgAssert(HEADER_ENTRY_SIZE >= sizeof(MEMMAP_HEADER));
|
|
|
|
// validate the file content if the file is not new.
|
|
if (_NewFile != TRUE)
|
|
{
|
|
if (ValidateCache() == FALSE)
|
|
{
|
|
if (!Reinitialize())
|
|
{
|
|
_Status = ERROR_INTERNET_INTERNAL_ERROR;
|
|
goto Cleanup;
|
|
}
|
|
|
|
// Succeeded in re-initializing the file, we
|
|
// treat this as if we created a new file.
|
|
_NewFile = TRUE;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// It is a brand new file. Initialize file header.
|
|
if (!InitHeaderInfo())
|
|
{
|
|
// This can happen if there is an exception while
|
|
// initializing headers
|
|
_Status = ERROR_INTERNET_INTERNAL_ERROR;
|
|
goto Cleanup;
|
|
|
|
}
|
|
}
|
|
|
|
// Compute number of bitmap DWORDs used.
|
|
_NumBitMapDWords =
|
|
(_HeaderInfo->NumUrlInternalEntries + (NUM_BITS_IN_DWORD - 1)) /
|
|
NUM_BITS_IN_DWORD; //cell
|
|
|
|
// We are done.
|
|
_Status = ERROR_SUCCESS;
|
|
|
|
Cleanup:
|
|
|
|
if (_Status != ERROR_SUCCESS)
|
|
{
|
|
INET_ASSERT(FALSE);
|
|
TcpsvcsDbgPrint((DEBUG_ERRORS, "MEMMAP_FILE::Initfailed, %ld\n", _Status));
|
|
|
|
SetLastError(_Status);
|
|
}
|
|
|
|
if (_NewFile)
|
|
return MEMMAP_STATUS_REINITIALIZED;
|
|
else
|
|
return MEMMAP_STATUS_OPENED_EXISTING;
|
|
}
|
|
|
|
|
|
MEMMAP_FILE::~MEMMAP_FILE(VOID)
|
|
/*++
|
|
Routine Description:
|
|
MEMMAP_FILE object destructor.
|
|
Arguments:
|
|
None.
|
|
Return Value:
|
|
None.
|
|
--*/
|
|
{
|
|
CloseMapping();
|
|
CacheHeap->Free(_FileName);
|
|
CacheHeap->Free(_FullPathName);
|
|
}
|
|
|
|
|
|
BOOL MEMMAP_FILE::ReAllocateEntry(LPFILEMAP_ENTRY pEntry, DWORD cbBytes)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Attempts to reallocate an entry at the location given.
|
|
|
|
Arguments:
|
|
|
|
LPFILEMAP_ENTRY pEntry: Pointer to location in file map.
|
|
DWORD cbBytes : Number of bytes requested
|
|
|
|
Return Value:
|
|
|
|
Original value of pEntry if successful. pEntry->nBlocks is set to the new
|
|
value, but all other fields in the entry are unmodified. If insufficient contiguous
|
|
bits are found at the end of the original entry, NULL is returned, indicating failure.
|
|
In this case the entry remains unmodified.
|
|
|
|
Notes:
|
|
|
|
The Map file should *not* be grown if insufficient additional bits are not found.
|
|
|
|
--*/
|
|
{
|
|
// Validate cbBytes
|
|
if (cbBytes > MAX_ENTRY_SIZE)
|
|
{
|
|
INET_ASSERT(FALSE);
|
|
return FALSE;
|
|
}
|
|
|
|
// Validate pEntry.
|
|
DWORD cbEntryOffset = (DWORD)PtrDifference(pEntry, _EntryArray);
|
|
if (IsBadOffset(cbEntryOffset))
|
|
{
|
|
INET_ASSERT(FALSE);
|
|
return FALSE;
|
|
}
|
|
|
|
// Calculate number of blocks required for this entry.
|
|
DWORD nBlocksRequired = NUMBLOCKS(ROUNDUPBLOCKS(cbBytes));
|
|
|
|
// Sufficient space in current slot?
|
|
if (nBlocksRequired <= pEntry->nBlocks)
|
|
{
|
|
// We're done.
|
|
return TRUE;
|
|
}
|
|
else
|
|
{
|
|
// Determine if additional free bits are
|
|
// available at the end of this entry.
|
|
// If not, return NULL.
|
|
|
|
// Determine the array and bit indicese of the first
|
|
// free bit immediately following the last set bit of
|
|
// the entry.
|
|
DWORD nTrailingIndex = cbEntryOffset / _EntrySize + pEntry->nBlocks;
|
|
DWORD nArrayIndex = nTrailingIndex / NUM_BITS_IN_DWORD;
|
|
DWORD nBitIndex = nTrailingIndex % NUM_BITS_IN_DWORD;
|
|
DWORD dwMask = 0x1 << nBitIndex;
|
|
DWORD nAdditionalBlocksRequired = nBlocksRequired - pEntry->nBlocks;
|
|
DWORD nBlocksFound;
|
|
|
|
// Don't exceed the number of internal entries.
|
|
if (nTrailingIndex + nAdditionalBlocksRequired
|
|
> _HeaderInfo->NumUrlInternalEntries)
|
|
{
|
|
// Overstepped last internal entry. Here we should fail
|
|
// by returning NULL. Note - DO NOT attempt to grow the
|
|
// map file at this point. The caller does not expect this.
|
|
return FALSE;
|
|
}
|
|
|
|
if (CheckNextNBits(nArrayIndex, dwMask,
|
|
nAdditionalBlocksRequired, nBlocksFound))
|
|
{
|
|
// We were able to grow the entry.
|
|
SetNextNBits(nArrayIndex, dwMask, nAdditionalBlocksRequired);
|
|
pEntry->nBlocks = nBlocksRequired;
|
|
_HeaderInfo->NumUrlEntriesAlloced += nAdditionalBlocksRequired;
|
|
return TRUE;
|
|
}
|
|
else
|
|
// Couldn't grow the entry.
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
LPFILEMAP_ENTRY MEMMAP_FILE::AllocateEntry(DWORD cbBytes)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Member function that returns an free entry from the cache list. If
|
|
none is available free, it grows the map file, makes more free
|
|
entries.
|
|
|
|
Arguments:
|
|
|
|
DWORD cbBytes : Number of bytes requested
|
|
DWORD cbOffset: Offset from beginning of bit map where allocation is requested.
|
|
|
|
Return Value:
|
|
|
|
If NULL, GetStatus() will return actual error code.
|
|
|
|
--*/
|
|
{
|
|
LPFILEMAP_ENTRY NewEntry;
|
|
|
|
// Validate cbBytes
|
|
if (cbBytes > MAX_ENTRY_SIZE)
|
|
{
|
|
INET_ASSERT(FALSE);
|
|
return 0;
|
|
}
|
|
|
|
// Find and mark off a set of contiguous bits
|
|
// spanning the requested number of bytes.
|
|
DWORD nBlocksRequired = NUMBLOCKS(ROUNDUPBLOCKS(cbBytes));
|
|
DWORD FreeEntryIndex = GetAndSetNextFreeEntry(nBlocksRequired);
|
|
|
|
// Failed to find space.
|
|
if (FreeEntryIndex == 0xFFFFFFFF)
|
|
{
|
|
// Map file is full, grow it now.
|
|
_Status = GrowMapFile(cbBytes <= GlobalMapFileGrowSize ?
|
|
GlobalMapFileGrowSize : ROUNDUPTOPOWEROF2(cbBytes, ALLOC_PAGES * PAGE_SIZE));
|
|
|
|
// Failed to grow map file.
|
|
if (_Status != ERROR_SUCCESS)
|
|
{
|
|
return NULL;
|
|
}
|
|
|
|
// Retry with enlarged map file.
|
|
FreeEntryIndex = GetAndSetNextFreeEntry(nBlocksRequired);
|
|
|
|
TcpsvcsDbgAssert(FreeEntryIndex != 0xFFFFFFFF);
|
|
|
|
// Failed to allocate bytes after enlarging map file.
|
|
if (FreeEntryIndex == 0xFFFFFFFF)
|
|
{
|
|
return NULL;
|
|
}
|
|
}
|
|
|
|
INET_ASSERT((cbBytes < PAGE_SIZE)
|
|
|| ((cbBytes >= PAGE_SIZE) && !((_EntrySize * FreeEntryIndex) % PAGE_SIZE)));
|
|
|
|
// Cast the memory.
|
|
NewEntry = (LPFILEMAP_ENTRY)
|
|
(_EntryArray + _EntrySize * FreeEntryIndex);
|
|
|
|
// Mark the allocated space.
|
|
#ifdef DBG
|
|
ResetEntryData(NewEntry, SIG_ALLOC, nBlocksRequired);
|
|
#else
|
|
NewEntry->dwSig = SIG_ALLOC;
|
|
#endif // DBG
|
|
|
|
// Set the number of blocks in the entry.
|
|
NewEntry->nBlocks = nBlocksRequired;
|
|
|
|
return NewEntry;
|
|
}
|
|
|
|
|
|
BOOL MEMMAP_FILE::FreeEntry(LPFILEMAP_ENTRY Entry)
|
|
/*++
|
|
|
|
This public member function frees up a file cache entry.
|
|
|
|
Arguments:
|
|
|
|
UrlEntry : pointer to the entry that being freed.
|
|
|
|
Return Value:
|
|
|
|
TRUE - if the entry is successfully removed from the cache.
|
|
FALSE - otherwise.
|
|
|
|
--*/
|
|
{
|
|
DWORD nIndex, nArrayIndex,
|
|
nOffset, nBlocks, BitMask;
|
|
|
|
LPDWORD BitMap;
|
|
|
|
|
|
// Validate the pointer passed in.
|
|
|
|
if (((LPBYTE)Entry < _EntryArray)
|
|
|| ((LPBYTE)Entry >=
|
|
(_EntryArray + _EntrySize *
|
|
_HeaderInfo->NumUrlInternalEntries)))
|
|
{
|
|
TcpsvcsDbgAssert(FALSE);
|
|
return FALSE;
|
|
}
|
|
|
|
// Compute and check offset (number of bytes from start).
|
|
nOffset = (DWORD)PtrDifference(Entry, _EntryArray);
|
|
if (nOffset % _EntrySize)
|
|
{
|
|
// Pointer does not point to a valid entry.
|
|
TcpsvcsDbgAssert(FALSE);
|
|
return FALSE;
|
|
}
|
|
|
|
nBlocks = Entry->nBlocks;
|
|
|
|
if (nBlocks > (MAX_ENTRY_SIZE / NORMAL_ENTRY_SIZE))
|
|
{
|
|
INET_ASSERT(FALSE);
|
|
return FALSE;
|
|
}
|
|
|
|
// Compute indicese
|
|
nIndex = nOffset / _EntrySize;
|
|
nArrayIndex = nIndex / NUM_BITS_IN_DWORD;
|
|
|
|
|
|
// Unmark the index bits in the map.
|
|
|
|
|
|
BitMap = &_HeaderInfo->AllocationBitMap[nArrayIndex];
|
|
BitMask = 0x1 << (nIndex % NUM_BITS_IN_DWORD);
|
|
for (DWORD i = 0; i < nBlocks; i++)
|
|
{
|
|
// Check we don't free unset bits
|
|
if (!(*BitMap & BitMask))
|
|
{
|
|
TcpsvcsDbgPrint((DEBUG_ERRORS, "Attempted to free unset bits. Ignoring...\n"));
|
|
return FALSE;
|
|
}
|
|
|
|
*BitMap &= ~BitMask;
|
|
BitMask <<= 1;
|
|
if (BitMask == 0x0)
|
|
{
|
|
BitMask = 0x1;
|
|
BitMap = &_HeaderInfo->AllocationBitMap[++nArrayIndex];
|
|
}
|
|
}
|
|
|
|
// Mark the freed space.
|
|
ResetEntryData(Entry, SIG_FREE, nBlocks);
|
|
|
|
// Reduce the count of allocated entries.
|
|
TcpsvcsDbgAssert(_HeaderInfo->NumUrlEntriesAlloced > 0);
|
|
_HeaderInfo->NumUrlEntriesAlloced -= nBlocks;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
BOOL
|
|
MEMMAP_FILE::Reinitialize(void)
|
|
/*++
|
|
|
|
This member function reinitializes a cache index file
|
|
|
|
Arguments:
|
|
|
|
|
|
|
|
Return Value:
|
|
|
|
Windows error code
|
|
|
|
|
|
--*/
|
|
{
|
|
TcpsvcsDbgAssert(_FileHandle != NULL);
|
|
|
|
// Close view, mapping, and file.
|
|
CloseMapping();
|
|
|
|
BOOL BoolError, fReinited = FALSE;
|
|
DWORD FilePointer;
|
|
|
|
// check for exclusive access, we do this by opening the
|
|
// file in exclsive mode, if we succeed we are the only one
|
|
_FileHandle = CreateFile(
|
|
_FileName,
|
|
GENERIC_WRITE,
|
|
0, // no read/write sharing
|
|
CreateAllAccessSecurityAttributes(NULL, NULL, NULL),
|
|
OPEN_ALWAYS,
|
|
FILE_ATTRIBUTE_NORMAL,
|
|
NULL
|
|
);
|
|
|
|
if (_FileHandle == INVALID_HANDLE_VALUE)
|
|
_FileHandle = NULL;
|
|
else
|
|
{
|
|
DWORD FilePointer = SetFilePointer
|
|
(_FileHandle, GlobalMapFileGrowSize, NULL, FILE_BEGIN);
|
|
|
|
if (FilePointer != 0xFFFFFFFF)
|
|
{
|
|
if (SetEndOfFile(_FileHandle))
|
|
{
|
|
// Success!
|
|
_FileSize = GlobalMapFileGrowSize;
|
|
fReinited = TRUE;
|
|
}
|
|
else
|
|
{
|
|
TcpsvcsDbgPrint((DEBUG_ERRORS, "SetEndOfFile failed: %u\n",
|
|
GetLastError()));
|
|
}
|
|
}
|
|
|
|
// Following will be done by RemapAddress calling CloseMapping
|
|
// CloseHandle (_FileHandle);
|
|
// _FileHandle = NULL
|
|
}
|
|
|
|
// Re-attach to the file.
|
|
|
|
_Status = RemapAddress();
|
|
|
|
if (_Status != ERROR_SUCCESS)
|
|
{
|
|
TcpsvcsDbgPrint((DEBUG_ERRORS,
|
|
"Reinitialize:Remap failed, %ld.\n", _Status));
|
|
TcpsvcsDbgAssert(FALSE);
|
|
goto Cleanup;
|
|
}
|
|
|
|
if (fReinited)
|
|
{
|
|
// if there is an exception due to bad sector, this will set
|
|
// _status to something other than ERROR_SUCCESS
|
|
if (!InitHeaderInfo())
|
|
goto Cleanup;
|
|
|
|
_NumBitMapDWords =
|
|
(_HeaderInfo->NumUrlInternalEntries + (NUM_BITS_IN_DWORD - 1)) /
|
|
NUM_BITS_IN_DWORD; // cell
|
|
}
|
|
|
|
Cleanup:
|
|
|
|
return fReinited;
|
|
}
|
|
|
|
BOOL
|
|
MEMMAP_FILE::InitHeaderInfo()
|
|
/*++
|
|
|
|
This member function intializes the memorymapped headerinfo
|
|
structure
|
|
|
|
Arguments:
|
|
|
|
|
|
|
|
Return Value:
|
|
|
|
None
|
|
|
|
--*/
|
|
{
|
|
|
|
// initialize file header.
|
|
|
|
BOOL fSuccess = TRUE;
|
|
|
|
__try {
|
|
TcpsvcsDbgAssert(_HeaderInfo != NULL);
|
|
|
|
memcpy(_HeaderInfo->FileSignature, CACHE_SIGNATURE, sizeof(CACHE_SIGNATURE));
|
|
|
|
_HeaderInfo->FileSize = _FileSize; // set file size in the memmap file.
|
|
_HeaderInfo->dwHashTableOffset = 0;
|
|
_HeaderInfo->CacheSize = (LONGLONG)0;
|
|
_HeaderInfo->CacheLimit = (LONGLONG)0;
|
|
_HeaderInfo->ExemptUsage = (LONGLONG)0;
|
|
_HeaderInfo->nDirCount = 0;
|
|
|
|
for (int i = 0; i < DEFAULT_MAX_DIRS; i++)
|
|
{
|
|
_HeaderInfo->DirArray[i].nFileCount = 0;
|
|
_HeaderInfo->DirArray[i].sDirName[0] = '\0';
|
|
}
|
|
|
|
_HeaderInfo->NumUrlInternalEntries =
|
|
((_FileSize - HEADER_ENTRY_SIZE) /
|
|
_EntrySize);
|
|
|
|
_HeaderInfo->NumUrlEntriesAlloced = 0;
|
|
|
|
memset(_HeaderInfo->AllocationBitMap, 0, sizeof(_HeaderInfo->AllocationBitMap));
|
|
memset(_EntryArray, 0, (_FileSize - HEADER_ENTRY_SIZE));
|
|
memset(_HeaderInfo->dwHeaderData, 0, sizeof(DWORD) * NUM_HEADER_DATA_DWORDS);
|
|
|
|
_Status = ERROR_SUCCESS;
|
|
}
|
|
__except (EXCEPTION_EXECUTE_HANDLER) {
|
|
|
|
_Status = ERROR_WRITE_FAULT;
|
|
fSuccess = FALSE;
|
|
}
|
|
ENDEXCEPT
|
|
|
|
return (fSuccess);
|
|
}
|
|
|
|
LPFILEMAP_ENTRY MEMMAP_FILE::FindNextEntry(DWORD* pdwEnum, DWORD dwFilter, GROUPID GroupId)
|
|
{
|
|
while (1)
|
|
{
|
|
// Get the next item in the hash table.
|
|
HASH_ITEM *pItem = HashGetNextItem(this, (LPBYTE)_BaseAddr, pdwEnum, 0);
|
|
if (!pItem)
|
|
return NULL;
|
|
|
|
// continue if search entry within group but hash bit says no group
|
|
// (may avoid unnecessary page hit by pulling non-relevent pEntry)
|
|
if (GroupId && !pItem->HasGroup())
|
|
continue;
|
|
|
|
|
|
// Get the entry from the item.
|
|
URL_FILEMAP_ENTRY* pEntry = ValidateUrlOffset(pItem->dwOffset);
|
|
if (!pEntry)
|
|
{
|
|
pItem->MarkFree();
|
|
continue;
|
|
}
|
|
|
|
// No filter - continue enum until ERROR_NO_MORE_ITEMS.
|
|
if (!dwFilter)
|
|
continue;
|
|
|
|
// Temporary hack to always show 1.1 entries
|
|
// until we have a better way of dealing with them.
|
|
dwFilter |= INCLUDE_BY_DEFAULT_CACHE_ENTRY;
|
|
|
|
// Continue enum if no match on cache entry type.
|
|
if ((dwFilter & pEntry->CacheEntryType) != pEntry->CacheEntryType)
|
|
continue;
|
|
|
|
// Continue enum if no match on group.
|
|
if (GroupId)
|
|
{
|
|
if (pItem->HasMultiGroup())
|
|
{
|
|
// need to search the list
|
|
LIST_GROUP_ENTRY* pListGroup = NULL;
|
|
pListGroup = ValidateListGroupOffset(pEntry->dwGroupOffset);
|
|
if (!pListGroup)
|
|
continue;
|
|
|
|
BOOL fFoundOnList = FALSE;
|
|
while (pListGroup && pListGroup->dwGroupOffset)
|
|
{
|
|
GROUP_ENTRY* pGroup = NULL;
|
|
pGroup = ValidateGroupOffset(
|
|
pListGroup->dwGroupOffset, pItem);
|
|
if (!pGroup)
|
|
{
|
|
break;
|
|
}
|
|
|
|
if (GroupId == pGroup->gid)
|
|
{
|
|
fFoundOnList = TRUE;
|
|
break;
|
|
}
|
|
|
|
if (!pListGroup->dwNext)
|
|
{
|
|
break;
|
|
}
|
|
|
|
// next group on list
|
|
pListGroup = ValidateListGroupOffset(pListGroup->dwNext);
|
|
}
|
|
|
|
if (!fFoundOnList)
|
|
continue;
|
|
}
|
|
else if (GroupId != ((GROUP_ENTRY*)((LPBYTE)_BaseAddr + pEntry->dwGroupOffset))->gid)
|
|
{
|
|
continue;
|
|
}
|
|
|
|
}
|
|
|
|
return (LPFILEMAP_ENTRY)(((LPBYTE)_BaseAddr) + pItem->dwOffset);
|
|
}
|
|
}
|
|
|
|
BOOL MEMMAP_FILE::IsBadOffset(DWORD dwOffset)
|
|
{
|
|
|
|
ASSERT_ISPOWEROF2(_EntrySize);
|
|
return (dwOffset == 0
|
|
|| (dwOffset & (_EntrySize - 1))
|
|
|| (dwOffset >= _FileSize));
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
BOOL MEMMAP_FILE::IsBadGroupOffset(DWORD dwOffset)
|
|
{
|
|
return (dwOffset == 0 || (dwOffset >= _FileSize));
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
GROUP_ENTRY* MEMMAP_FILE::ValidateGroupOffset(DWORD dwOffset, HASH_ITEM* hItem)
|
|
{
|
|
GROUP_ENTRY *pEntry = NULL;
|
|
|
|
// if hash item is available, check the group bit first.
|
|
if (hItem && !hItem->HasGroup())
|
|
{
|
|
return NULL;
|
|
}
|
|
|
|
// check the offset
|
|
if (IsBadGroupOffset(dwOffset))
|
|
{
|
|
INET_ASSERT(FALSE);
|
|
return NULL;
|
|
}
|
|
|
|
|
|
// Validate page signature.
|
|
// since we know all the allocated page are aligned with
|
|
// 4K boundary, so from the offset, we can get the
|
|
// the offset of this page by:
|
|
// pageOffset = Offset - Offset(mod)4K
|
|
|
|
|
|
DWORD dwOffsetInPage = dwOffset & 0x00000FFF;
|
|
FILEMAP_ENTRY* pFM = (FILEMAP_ENTRY*)
|
|
((LPBYTE)_BaseAddr + dwOffset - dwOffsetInPage);
|
|
|
|
// Get the Group.
|
|
if (pFM->dwSig == SIG_ALLOC && pFM->nBlocks)
|
|
{
|
|
pEntry = (GROUP_ENTRY *)((LPBYTE)_BaseAddr + dwOffset);
|
|
}
|
|
|
|
return pEntry;
|
|
}
|
|
|
|
|
|
|
|
URL_FILEMAP_ENTRY* MEMMAP_FILE::ValidateUrlOffset(DWORD dwOffset)
|
|
{
|
|
// Validate offset.
|
|
if (IsBadOffset(dwOffset))
|
|
{
|
|
INET_ASSERT(FALSE);
|
|
return NULL;
|
|
}
|
|
|
|
// Validate signature.
|
|
URL_FILEMAP_ENTRY *pEntry =
|
|
(URL_FILEMAP_ENTRY *)((LPBYTE)_BaseAddr + dwOffset);
|
|
if (pEntry->dwSig != SIG_URL)
|
|
{
|
|
INET_ASSERT(FALSE);
|
|
return NULL;
|
|
}
|
|
|
|
// TODO: validate entry offsets, string terminations etc.
|
|
return pEntry;
|
|
}
|
|
|
|
|
|
LIST_GROUP_ENTRY* MEMMAP_FILE::ValidateListGroupOffset(DWORD dwOffset)
|
|
{
|
|
LIST_GROUP_ENTRY *pEntry = NULL;
|
|
|
|
// Validate offset.
|
|
if (IsBadGroupOffset(dwOffset))
|
|
{
|
|
INET_ASSERT(FALSE);
|
|
return NULL;
|
|
}
|
|
|
|
|
|
// Validate page signature.
|
|
// since we know all the allocated page are aligned with
|
|
// 4K boundary, so from the offset, we can get the
|
|
// the offset of this page by:
|
|
// pageOffset = Offset - Offset(mod)4K
|
|
|
|
|
|
DWORD dwOffsetInPage = dwOffset & 0x00000FFF;
|
|
FILEMAP_ENTRY* pFM = (FILEMAP_ENTRY*)
|
|
((LPBYTE)_BaseAddr + dwOffset - dwOffsetInPage);
|
|
|
|
|
|
if (pFM->dwSig == SIG_ALLOC && pFM->nBlocks)
|
|
{
|
|
pEntry = (LIST_GROUP_ENTRY*)((LPBYTE)_BaseAddr + dwOffset);
|
|
}
|
|
|
|
return pEntry;
|
|
}
|