1041 lines
27 KiB
C
1041 lines
27 KiB
C
/*++
|
|
|
|
Copyright (c) 1991 Microsoft Corporation
|
|
|
|
Module Name:
|
|
|
|
hivemap.c
|
|
|
|
Abstract:
|
|
|
|
This module implements HvpBuildMap - used to build the initial map for a hive
|
|
|
|
Author:
|
|
|
|
Bryan M. Willman (bryanwi) 28-Mar-92
|
|
|
|
Environment:
|
|
|
|
|
|
Revision History:
|
|
Dragos C. Sambotin (dragoss) 25-Jan-99
|
|
Implementation of bin-size chunk loading of hives.
|
|
--*/
|
|
|
|
#include "cmp.h"
|
|
|
|
#ifdef ALLOC_PRAGMA
|
|
#pragma alloc_text(PAGE,HvpBuildMap)
|
|
#pragma alloc_text(PAGE,HvpFreeMap)
|
|
#pragma alloc_text(PAGE,HvpAllocateMap)
|
|
#pragma alloc_text(PAGE,HvpBuildMapAndCopy)
|
|
#pragma alloc_text(PAGE,HvpEnlistFreeCells)
|
|
#pragma alloc_text(PAGE,HvpInitMap)
|
|
#pragma alloc_text(PAGE,HvpCleanMap)
|
|
#pragma alloc_text(PAGE,HvpEnlistBinInMap)
|
|
#pragma alloc_text(PAGE,HvpGetBinMemAlloc)
|
|
#endif
|
|
|
|
extern struct {
|
|
PHHIVE Hive;
|
|
ULONG Status;
|
|
ULONG Space;
|
|
HCELL_INDEX MapPoint;
|
|
PHBIN BinPoint;
|
|
} HvCheckHiveDebug;
|
|
|
|
//Dragos: Modified functions
|
|
NTSTATUS
|
|
HvpBuildMapAndCopy(
|
|
PHHIVE Hive,
|
|
PVOID Image
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Creates the map for the Stable storage of the hive, and inits
|
|
the map for the volatile storage.
|
|
|
|
Following fields in hive must already be filled in:
|
|
|
|
Allocate, Free
|
|
|
|
Will initialize Storage structure of HHIVE.
|
|
|
|
This function is called for the HINIT_MEMORY case. The hive is guaranteed
|
|
to be in paged-pool. More than that, the hive image is contiguous.
|
|
It'll then copy from that image to the new paged-pool allocations.
|
|
|
|
Arguments:
|
|
|
|
Hive - Pointer to hive control structure to build map for.
|
|
|
|
Image - pointer to flat memory image of original hive.
|
|
|
|
Return Value:
|
|
|
|
TRUE - it worked
|
|
FALSE - either hive is corrupt or no memory for map
|
|
|
|
--*/
|
|
{
|
|
PHBASE_BLOCK BaseBlock;
|
|
ULONG Length;
|
|
ULONG MapSlots;
|
|
ULONG Tables;
|
|
PHMAP_TABLE t = NULL;
|
|
PHMAP_DIRECTORY d = NULL;
|
|
PHBIN Bin;
|
|
PHBIN CurrentBin;
|
|
ULONG Offset;
|
|
ULONG_PTR Address;
|
|
PHMAP_ENTRY Me;
|
|
NTSTATUS Status;
|
|
PULONG Vector;
|
|
|
|
|
|
CmKdPrintEx((DPFLTR_CONFIG_ID,CML_HIVE,"HvpBuildMap:\n"));
|
|
CmKdPrintEx((DPFLTR_CONFIG_ID,CML_HIVE,"\tHive=%p",Hive));
|
|
|
|
|
|
//
|
|
// Compute size of data region to be mapped
|
|
//
|
|
BaseBlock = Hive->BaseBlock;
|
|
Length = BaseBlock->Length;
|
|
if ((Length % HBLOCK_SIZE) != 0 ) {
|
|
Status = STATUS_REGISTRY_CORRUPT;
|
|
goto ErrorExit1;
|
|
}
|
|
MapSlots = Length / HBLOCK_SIZE;
|
|
if( MapSlots > 0 ) {
|
|
Tables = (MapSlots-1) / HTABLE_SLOTS;
|
|
} else {
|
|
Tables = 0;
|
|
}
|
|
|
|
Hive->Storage[Stable].Length = Length;
|
|
|
|
//
|
|
// allocate dirty vector if one is not already present (from HvpRecoverData)
|
|
//
|
|
|
|
if (Hive->DirtyVector.Buffer == NULL) {
|
|
Vector = (PULONG)((Hive->Allocate)(ROUND_UP(Length/HSECTOR_SIZE/8,sizeof(ULONG)), TRUE,CM_FIND_LEAK_TAG22));
|
|
if (Vector == NULL) {
|
|
Status = STATUS_NO_MEMORY;
|
|
goto ErrorExit1;
|
|
}
|
|
RtlZeroMemory(Vector, Length / HSECTOR_SIZE / 8);
|
|
RtlInitializeBitMap(&Hive->DirtyVector, Vector, Length / HSECTOR_SIZE);
|
|
Hive->DirtyAlloc = ROUND_UP(Length/HSECTOR_SIZE/8,sizeof(ULONG));
|
|
}
|
|
|
|
//
|
|
// allocate and build structure for map
|
|
//
|
|
if (Tables == 0) {
|
|
|
|
//
|
|
// Just 1 table, no need for directory
|
|
//
|
|
t = (Hive->Allocate)(sizeof(HMAP_TABLE), FALSE,CM_FIND_LEAK_TAG23);
|
|
if (t == NULL) {
|
|
Status = STATUS_INSUFFICIENT_RESOURCES;
|
|
goto ErrorExit1;
|
|
}
|
|
RtlZeroMemory(t, sizeof(HMAP_TABLE));
|
|
Hive->Storage[Stable].Map =
|
|
(PHMAP_DIRECTORY)&(Hive->Storage[Stable].SmallDir);
|
|
Hive->Storage[Stable].SmallDir = t;
|
|
|
|
} else {
|
|
|
|
//
|
|
// Need directory and multiple tables
|
|
//
|
|
d = (PHMAP_DIRECTORY)(Hive->Allocate)(sizeof(HMAP_DIRECTORY), FALSE,CM_FIND_LEAK_TAG24);
|
|
if (d == NULL) {
|
|
Status = STATUS_INSUFFICIENT_RESOURCES;
|
|
goto ErrorExit1;
|
|
}
|
|
RtlZeroMemory(d, sizeof(HMAP_DIRECTORY));
|
|
|
|
//
|
|
// Allocate tables and fill in dir
|
|
//
|
|
if (HvpAllocateMap(Hive, d, 0, Tables) == FALSE) {
|
|
Status = STATUS_INSUFFICIENT_RESOURCES;
|
|
goto ErrorExit2;
|
|
}
|
|
Hive->Storage[Stable].Map = d;
|
|
Hive->Storage[Stable].SmallDir = 0;
|
|
}
|
|
|
|
//
|
|
// Now we have to allocate the memory for the HBINs and fill in
|
|
// the map appropriately. We are careful never to allocate less
|
|
// than a page to avoid fragmenting pool. As long as the page
|
|
// size is a multiple of HBLOCK_SIZE (a fairly good assumption as
|
|
// long as HBLOCK_SIZE is 4k) this strategy will prevent pool
|
|
// fragmentation.
|
|
//
|
|
// If we come across an HBIN that is entirely composed of a freed
|
|
// HCELL, then we do not allocate memory, but mark its HBLOCKs in
|
|
// the map as not present. HvAllocateCell will allocate memory for
|
|
// the bin when it is needed.
|
|
//
|
|
Offset = 0;
|
|
Bin = (PHBIN)Image;
|
|
|
|
while (Bin < (PHBIN)((PUCHAR)(Image) + Length)) {
|
|
|
|
if ( (Bin->Size > (Length-Offset)) ||
|
|
(Bin->Signature != HBIN_SIGNATURE) ||
|
|
(Bin->FileOffset != Offset)
|
|
)
|
|
{
|
|
//
|
|
// Bin is bogus
|
|
//
|
|
Status = STATUS_REGISTRY_CORRUPT;
|
|
goto ErrorExit2;
|
|
}
|
|
|
|
CurrentBin = (PHBIN)(Hive->Allocate)(Bin->Size, FALSE,CM_FIND_LEAK_TAG25);
|
|
if (CurrentBin==NULL) {
|
|
Status = STATUS_INSUFFICIENT_RESOURCES;
|
|
goto ErrorExit2; //fixfix
|
|
}
|
|
RtlCopyMemory(CurrentBin,
|
|
(PUCHAR)Image+Offset,
|
|
Bin->Size);
|
|
|
|
//
|
|
// create map entries for each block/page in bin
|
|
//
|
|
Address = (ULONG_PTR)CurrentBin;
|
|
do {
|
|
Me = HvpGetCellMap(Hive, Offset);
|
|
VALIDATE_CELL_MAP(__LINE__,Me,Hive,Offset);
|
|
Me->BlockAddress = Address;
|
|
Me->BinAddress = (ULONG_PTR)CurrentBin;
|
|
|
|
if (Address == (ULONG_PTR)CurrentBin) {
|
|
Me->BinAddress |= HMAP_NEWALLOC;
|
|
Me->MemAlloc = CurrentBin->Size;
|
|
} else {
|
|
Me->MemAlloc = 0;
|
|
}
|
|
|
|
Me->BinAddress |= HMAP_INPAGEDPOOL;
|
|
// we don't need to set this - just for debug purposes
|
|
ASSERT( (Me->CmView = NULL) == NULL );
|
|
|
|
Address += HBLOCK_SIZE;
|
|
Offset += HBLOCK_SIZE;
|
|
} while ( Address < ((ULONG_PTR)CurrentBin + CurrentBin->Size ));
|
|
|
|
if (Hive->ReadOnly == FALSE) {
|
|
|
|
//
|
|
// add free cells in the bin to the appropriate free lists
|
|
//
|
|
if ( ! HvpEnlistFreeCells(Hive,
|
|
CurrentBin,
|
|
CurrentBin->FileOffset
|
|
)) {
|
|
Status = STATUS_REGISTRY_CORRUPT;
|
|
goto ErrorExit2;
|
|
}
|
|
}
|
|
|
|
Bin = (PHBIN)((ULONG_PTR)Bin + Bin->Size);
|
|
}
|
|
|
|
return STATUS_SUCCESS;
|
|
|
|
|
|
ErrorExit2:
|
|
if (d != NULL) {
|
|
|
|
//
|
|
// directory was built and allocated, so clean it up
|
|
//
|
|
|
|
HvpFreeMap(Hive, d, 0, Tables);
|
|
(Hive->Free)(d, sizeof(HMAP_DIRECTORY));
|
|
}
|
|
|
|
ErrorExit1:
|
|
return Status;
|
|
}
|
|
|
|
NTSTATUS
|
|
HvpInitMap(
|
|
PHHIVE Hive
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Initialize the map for the Stable Volatile storage of the hive.
|
|
|
|
Following fields in hive must already be filled in:
|
|
|
|
Allocate, Free
|
|
|
|
Will initialize Storage structure of HHIVE.
|
|
|
|
Arguments:
|
|
|
|
Hive - Pointer to hive control structure to build map for.
|
|
|
|
Return Value:
|
|
|
|
STATUS_SUCCESS - it worked
|
|
STATUS_xxx - the errorneous status
|
|
|
|
--*/
|
|
{
|
|
PHBASE_BLOCK BaseBlock;
|
|
ULONG Length;
|
|
ULONG MapSlots;
|
|
ULONG Tables;
|
|
PHMAP_TABLE t = NULL;
|
|
PHMAP_DIRECTORY d = NULL;
|
|
NTSTATUS Status;
|
|
PULONG Vector = NULL;
|
|
|
|
|
|
#ifndef _CM_LDR_
|
|
CmKdPrintEx((DPFLTR_CONFIG_ID,CML_HIVE,"HvpInitMap:\n"));
|
|
CmKdPrintEx((DPFLTR_CONFIG_ID,CML_HIVE,"\tHive=%p",Hive));
|
|
#endif //_CM_LDR_
|
|
|
|
//
|
|
// Compute size of data region to be mapped
|
|
//
|
|
BaseBlock = Hive->BaseBlock;
|
|
Length = BaseBlock->Length;
|
|
if ((Length % HBLOCK_SIZE) != 0) {
|
|
Status = STATUS_REGISTRY_CORRUPT;
|
|
goto ErrorExit1;
|
|
}
|
|
MapSlots = Length / HBLOCK_SIZE;
|
|
if( MapSlots > 0 ) {
|
|
Tables = (MapSlots-1) / HTABLE_SLOTS;
|
|
} else {
|
|
Tables = 0;
|
|
}
|
|
|
|
Hive->Storage[Stable].Length = Length;
|
|
|
|
//
|
|
// allocate dirty vector if one is not already present (from HvpRecoverData)
|
|
//
|
|
|
|
if (Hive->DirtyVector.Buffer == NULL) {
|
|
Vector = (PULONG)((Hive->Allocate)(ROUND_UP(Length/HSECTOR_SIZE/8,sizeof(ULONG)), TRUE,CM_FIND_LEAK_TAG27));
|
|
if (Vector == NULL) {
|
|
Status = STATUS_NO_MEMORY;
|
|
goto ErrorExit1;
|
|
}
|
|
RtlZeroMemory(Vector, Length / HSECTOR_SIZE / 8);
|
|
RtlInitializeBitMap(&Hive->DirtyVector, Vector, Length / HSECTOR_SIZE);
|
|
Hive->DirtyAlloc = ROUND_UP(Length/HSECTOR_SIZE/8,sizeof(ULONG));
|
|
}
|
|
|
|
//
|
|
// allocate and build structure for map
|
|
//
|
|
if (Tables == 0) {
|
|
|
|
//
|
|
// Just 1 table, no need for directory
|
|
//
|
|
t = (Hive->Allocate)(sizeof(HMAP_TABLE), FALSE,CM_FIND_LEAK_TAG26);
|
|
if (t == NULL) {
|
|
Status = STATUS_INSUFFICIENT_RESOURCES;
|
|
goto ErrorExit1;
|
|
}
|
|
RtlZeroMemory(t, sizeof(HMAP_TABLE));
|
|
Hive->Storage[Stable].Map =
|
|
(PHMAP_DIRECTORY)&(Hive->Storage[Stable].SmallDir);
|
|
Hive->Storage[Stable].SmallDir = t;
|
|
|
|
} else {
|
|
|
|
//
|
|
// Need directory and multiple tables
|
|
//
|
|
d = (PHMAP_DIRECTORY)(Hive->Allocate)(sizeof(HMAP_DIRECTORY), FALSE,CM_FIND_LEAK_TAG28);
|
|
if (d == NULL) {
|
|
Status = STATUS_INSUFFICIENT_RESOURCES;
|
|
goto ErrorExit1;
|
|
}
|
|
RtlZeroMemory(d, sizeof(HMAP_DIRECTORY));
|
|
|
|
//
|
|
// Allocate tables and fill in dir
|
|
//
|
|
if (HvpAllocateMap(Hive, d, 0, Tables) == FALSE) {
|
|
Status = STATUS_INSUFFICIENT_RESOURCES;
|
|
goto ErrorExit2;
|
|
}
|
|
Hive->Storage[Stable].Map = d;
|
|
Hive->Storage[Stable].SmallDir = 0;
|
|
}
|
|
|
|
return STATUS_SUCCESS;
|
|
|
|
ErrorExit2:
|
|
if (d != NULL) {
|
|
|
|
//
|
|
// directory was built and allocated, so clean it up
|
|
//
|
|
|
|
HvpFreeMap(Hive, d, 0, Tables);
|
|
(Hive->Free)(d, sizeof(HMAP_DIRECTORY));
|
|
}
|
|
|
|
ErrorExit1:
|
|
if( Vector ) {
|
|
(Hive->Free)(Vector, ROUND_UP(Length/HSECTOR_SIZE/8,sizeof(ULONG)));
|
|
Hive->DirtyVector.Buffer = NULL;
|
|
}
|
|
return Status;
|
|
}
|
|
|
|
NTSTATUS
|
|
HvpEnlistBinInMap(
|
|
PHHIVE Hive,
|
|
ULONG Length,
|
|
PHBIN Bin,
|
|
ULONG Offset,
|
|
PVOID CmView OPTIONAL
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Creates map entries and enlist free cells for the specified bin
|
|
|
|
Arguments:
|
|
|
|
Hive - Pointer to hive control structure containing the target map
|
|
|
|
Length - the Length of the hive image
|
|
|
|
Bin - the bin to be enlisted
|
|
|
|
Offset - the offset within the hive file
|
|
|
|
CmView - pointer to the mapped view of the bin. If NULL, the bin resides in paged pool
|
|
|
|
Return Value:
|
|
|
|
STATUS_SUCCESS - it worked
|
|
STATUS_REGISTRY_CORRUPT - the bin is inconsistent
|
|
STATUS_REGISTRY_RECOVERED - if we have fixed the bin on-the-fly (self heal feature).
|
|
|
|
--*/
|
|
{
|
|
NTSTATUS Status = STATUS_SUCCESS;
|
|
ULONG BinOffset;
|
|
ULONG_PTR Address;
|
|
PHMAP_ENTRY Me;
|
|
|
|
#ifndef _CM_LDR_
|
|
CmKdPrintEx((DPFLTR_CONFIG_ID,CML_HIVE,"HvpEnlistBinInMap:\n"));
|
|
CmKdPrintEx((DPFLTR_CONFIG_ID,CML_HIVE,"\tHive=%p\t Offset=%08lx",Hive,Offset));
|
|
#endif //_CM_LDR_
|
|
|
|
#ifndef _CM_LDR_
|
|
CmKdPrintEx((DPFLTR_CONFIG_ID,CML_BIN_MAP,"HvpEnlistBinInMap: BinAddress = 0x%p\t Size = 0x%lx\n", Bin, Bin->Size));
|
|
#endif //_CM_LDR_
|
|
|
|
//
|
|
// create map entries for each block/page in bin
|
|
//
|
|
BinOffset = Offset;
|
|
for (Address = (ULONG_PTR)Bin;
|
|
Address < ((ULONG_PTR)Bin + Bin->Size);
|
|
Address += HBLOCK_SIZE
|
|
)
|
|
{
|
|
Me = HvpGetCellMap(Hive, Offset);
|
|
VALIDATE_CELL_MAP(__LINE__,Me,Hive,Offset);
|
|
Me->BlockAddress = Address;
|
|
Me->BinAddress = (ULONG_PTR)Bin;
|
|
if (Offset == BinOffset) {
|
|
Me->BinAddress |= HMAP_NEWALLOC;
|
|
Me->MemAlloc = Bin->Size;
|
|
} else {
|
|
Me->MemAlloc = 0;
|
|
}
|
|
|
|
//
|
|
// take care here !!!!!
|
|
//
|
|
if( CmView == NULL ) {
|
|
Me->BinAddress |= HMAP_INPAGEDPOOL;
|
|
// we don't need to set this - just for debug purposes
|
|
ASSERT( (Me->CmView = NULL) == NULL );
|
|
} else {
|
|
Me->BinAddress |= HMAP_INVIEW;
|
|
// this should be already set by now
|
|
//ASSERT( Me->CmView == CmView );
|
|
}
|
|
|
|
Offset += HBLOCK_SIZE;
|
|
}
|
|
|
|
if (Hive->ReadOnly == FALSE) {
|
|
|
|
//
|
|
// add free cells in the bin to the apropriate free lists
|
|
//
|
|
if ( ! HvpEnlistFreeCells(Hive, Bin, BinOffset)) {
|
|
HvCheckHiveDebug.Hive = Hive;
|
|
HvCheckHiveDebug.Status = 0xA002;
|
|
HvCheckHiveDebug.Space = Length;
|
|
HvCheckHiveDebug.MapPoint = BinOffset;
|
|
HvCheckHiveDebug.BinPoint = Bin;
|
|
if( CmDoSelfHeal() ) {
|
|
Status = STATUS_REGISTRY_RECOVERED;
|
|
} else {
|
|
Status = STATUS_REGISTRY_CORRUPT;
|
|
goto ErrorExit;
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
//
|
|
// logical consistency check
|
|
//
|
|
ASSERT(Offset == (BinOffset + Bin->Size));
|
|
|
|
ErrorExit:
|
|
return Status;
|
|
}
|
|
|
|
NTSTATUS
|
|
HvpBuildMap(
|
|
PHHIVE Hive,
|
|
PVOID Image
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Creates the map for the Stable storage of the hive, and inits
|
|
the map for the volatile storage.
|
|
|
|
Following fields in hive must already be filled in:
|
|
|
|
Allocate, Free
|
|
|
|
Will initialize Storage structure of HHIVE.
|
|
|
|
Arguments:
|
|
|
|
Hive - Pointer to hive control structure to build map for.
|
|
|
|
Image - pointer to in memory image of the hive
|
|
|
|
Return Value:
|
|
|
|
TRUE - it worked
|
|
FALSE - either hive is corrupt or no memory for map
|
|
|
|
--*/
|
|
{
|
|
PHBIN Bin;
|
|
ULONG Offset;
|
|
NTSTATUS Status;
|
|
ULONG Length;
|
|
|
|
|
|
#ifndef _CM_LDR_
|
|
CmKdPrintEx((DPFLTR_CONFIG_ID,CML_HIVE,"HvpBuildMap:\n"));
|
|
CmKdPrintEx((DPFLTR_CONFIG_ID,CML_HIVE,"\tHive=%p",Hive));
|
|
#endif //_CM_LDR_
|
|
|
|
//
|
|
// Init the map
|
|
//
|
|
Status = HvpInitMap(Hive);
|
|
|
|
if( !NT_SUCCESS(Status) ) {
|
|
//
|
|
// just return failure; HvpInitMap took care of cleanup
|
|
//
|
|
return Status;
|
|
}
|
|
|
|
//
|
|
// Fill in the map
|
|
//
|
|
Offset = 0;
|
|
Bin = (PHBIN)Image;
|
|
Length = Hive->Storage[Stable].Length;
|
|
|
|
while (Bin < (PHBIN)((PUCHAR)(Image) + Length)) {
|
|
|
|
//
|
|
// Check the validity of the bin header
|
|
//
|
|
if ( (Bin->Size > Length) ||
|
|
(Bin->Size < HBLOCK_SIZE) ||
|
|
(Bin->Signature != HBIN_SIGNATURE) ||
|
|
(Bin->FileOffset != Offset)) {
|
|
//
|
|
// Bin is bogus
|
|
//
|
|
HvCheckHiveDebug.Hive = Hive;
|
|
HvCheckHiveDebug.Status = 0xA001;
|
|
HvCheckHiveDebug.Space = Length;
|
|
HvCheckHiveDebug.MapPoint = Offset;
|
|
HvCheckHiveDebug.BinPoint = Bin;
|
|
//
|
|
// for the loader.
|
|
//
|
|
if( CmDoSelfHeal() ) {
|
|
//
|
|
// put the correct signature, fileoffset and binsize in place;
|
|
// HvEnlistBinInMap will take care of the cells consistency.
|
|
//
|
|
Bin->Signature = HBIN_SIGNATURE;
|
|
Bin->FileOffset = Offset;
|
|
if ( ((Offset + Bin->Size) > Length) ||
|
|
(Bin->Size < HBLOCK_SIZE) ||
|
|
(Bin->Size % HBLOCK_SIZE) ) {
|
|
Bin->Size = HBLOCK_SIZE;
|
|
}
|
|
//
|
|
// signal back to the caller that we have altered the hive.
|
|
//
|
|
CmMarkSelfHeal(Hive);
|
|
} else {
|
|
Status = STATUS_REGISTRY_CORRUPT;
|
|
goto ErrorExit;
|
|
}
|
|
}
|
|
|
|
//
|
|
// enlist this bin
|
|
//
|
|
Status = HvpEnlistBinInMap(Hive, Length, Bin, Offset, NULL);
|
|
//
|
|
// for the loader.
|
|
//
|
|
if( CmDoSelfHeal() && (Status == STATUS_REGISTRY_RECOVERED) ) {
|
|
CmMarkSelfHeal(Hive);
|
|
Status = STATUS_SUCCESS;
|
|
}
|
|
|
|
if( !NT_SUCCESS(Status) ) {
|
|
goto ErrorExit;
|
|
}
|
|
|
|
//
|
|
// the next bin
|
|
//
|
|
Offset += Bin->Size;
|
|
|
|
Bin = (PHBIN)((ULONG_PTR)Bin + Bin->Size);
|
|
}
|
|
|
|
return STATUS_SUCCESS;
|
|
|
|
|
|
ErrorExit:
|
|
//
|
|
// Clean up the directory table
|
|
//
|
|
#ifndef _CM_LDR_
|
|
HvpCleanMap( Hive );
|
|
#endif //_CM_LDR_
|
|
|
|
return Status;
|
|
}
|
|
|
|
BOOLEAN
|
|
HvpEnlistFreeCells(
|
|
PHHIVE Hive,
|
|
PHBIN Bin,
|
|
ULONG BinOffset
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Scan through the cells in the bin, locating the free ones.
|
|
Enlist them in the hive's free list set.
|
|
|
|
N.B. Bin MUST already be mapped when this is called.
|
|
|
|
Arguments:
|
|
|
|
Hive - pointer to hive control structure map is being built for
|
|
|
|
Bin - pointer to bin to enlist cells from
|
|
|
|
BinOffset - offset of Bin in image
|
|
|
|
Return Value:
|
|
|
|
FALSE - registry is corrupt
|
|
|
|
TRUE - it worked
|
|
|
|
--*/
|
|
{
|
|
PHCELL p;
|
|
ULONG celloffset;
|
|
ULONG size;
|
|
HCELL_INDEX cellindex;
|
|
BOOLEAN Result = TRUE;
|
|
|
|
// PERFNOTE -- Keep this in mind as a possible optimization for NT6.
|
|
// Since now the hive is loaded in chunks of bins, we can drop the
|
|
// bins that are entirely free!!!!!!
|
|
//
|
|
|
|
//
|
|
// Scan all the cells in the bin, total free and allocated, check
|
|
// for impossible pointers.
|
|
//
|
|
celloffset = sizeof(HBIN);
|
|
p = (PHCELL)((PUCHAR)Bin + sizeof(HBIN));
|
|
|
|
while (p < (PHCELL)((PUCHAR)Bin + Bin->Size)) {
|
|
|
|
//
|
|
// if free cell, check it out, add it to free list for hive
|
|
//
|
|
if (p->Size >= 0) {
|
|
|
|
size = (ULONG)p->Size;
|
|
|
|
if ( (size > Bin->Size) ||
|
|
( (PHCELL)(size + (PUCHAR)p) >
|
|
(PHCELL)((PUCHAR)Bin + Bin->Size) ) ||
|
|
((size % HCELL_PAD(Hive)) != 0) ||
|
|
(size == 0) )
|
|
{
|
|
Result = FALSE;
|
|
if( CmDoSelfHeal() ) {
|
|
//
|
|
// self heal mode; enlist the remaining of the bin as free
|
|
// also zero it out so any references into the tampered area will be
|
|
// detected and fixed by the logical check later on
|
|
//
|
|
p->Size = (LONG)((PUCHAR)((PUCHAR)Bin + Bin->Size) - (PUCHAR)p);
|
|
RtlZeroMemory((PUCHAR)p + sizeof(ULONG),p->Size - sizeof(ULONG));
|
|
size = (ULONG)p->Size;
|
|
CmMarkSelfHeal(Hive);
|
|
} else {
|
|
goto Exit;
|
|
}
|
|
}
|
|
|
|
|
|
//
|
|
// cell is free, and is not obviously corrupt, add to free list
|
|
//
|
|
celloffset = (ULONG)((PUCHAR)p - (PUCHAR)Bin);
|
|
cellindex = BinOffset + celloffset;
|
|
|
|
//
|
|
// Enlist this free cell, but do not coalesce with the next free cell
|
|
// as we haven't gotten that far yet.
|
|
//
|
|
HvpEnlistFreeCell(Hive, cellindex, size, Stable, FALSE);
|
|
|
|
} else {
|
|
|
|
size = (ULONG)(p->Size * -1);
|
|
|
|
if ( (size > Bin->Size) ||
|
|
( (PHCELL)(size + (PUCHAR)p) >
|
|
(PHCELL)((PUCHAR)Bin + Bin->Size) ) ||
|
|
((size % HCELL_PAD(Hive)) != 0) ||
|
|
(size == 0) )
|
|
{
|
|
Result = FALSE;
|
|
if( CmDoSelfHeal() ) {
|
|
//
|
|
// Self heal mode; we have no other way than to enlist this cell as a free cell
|
|
//
|
|
p->Size = (LONG)((PUCHAR)((PUCHAR)Bin + Bin->Size) - (PUCHAR)p);
|
|
RtlZeroMemory((PUCHAR)p + sizeof(ULONG),p->Size - sizeof(ULONG));
|
|
size = (ULONG)p->Size;
|
|
|
|
celloffset = (ULONG)((PUCHAR)p - (PUCHAR)Bin);
|
|
cellindex = BinOffset + celloffset;
|
|
|
|
HvpEnlistFreeCell(Hive, cellindex, size, Stable, FALSE);
|
|
CmMarkSelfHeal(Hive);
|
|
} else {
|
|
goto Exit;
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
ASSERT( ((LONG)size) >= 0);
|
|
p = (PHCELL)((PUCHAR)p + size);
|
|
}
|
|
|
|
Exit:
|
|
return Result;
|
|
}
|
|
|
|
VOID
|
|
HvpCleanMap(
|
|
PHHIVE Hive
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Cleans all the map allocations for the stable storage
|
|
|
|
Arguments:
|
|
|
|
Hive - Pointer to hive control structure to build map for.
|
|
|
|
Return Value:
|
|
|
|
None
|
|
--*/
|
|
{
|
|
ULONG Length;
|
|
ULONG MapSlots;
|
|
ULONG Tables;
|
|
PHMAP_DIRECTORY d = NULL;
|
|
|
|
//
|
|
// Free DirtyVector if any.
|
|
//
|
|
if( Hive->DirtyVector.Buffer != NULL ) {
|
|
(Hive->Free)(Hive->DirtyVector.Buffer, ROUND_UP(Hive->Storage[Stable].Length/HSECTOR_SIZE/8,sizeof(ULONG)));
|
|
Hive->DirtyVector.Buffer = NULL;
|
|
Hive->DirtyAlloc = 0;
|
|
}
|
|
//
|
|
// Compute MapSlots and Tables based on the Length
|
|
//
|
|
Length = Hive->Storage[Stable].Length;
|
|
MapSlots = Length / HBLOCK_SIZE;
|
|
if( MapSlots > 0 ) {
|
|
Tables = (MapSlots-1) / HTABLE_SLOTS;
|
|
} else {
|
|
Tables = 0;
|
|
}
|
|
|
|
if( Hive->Storage[Stable].SmallDir == 0 ) {
|
|
//
|
|
// directory was built and allocated, so clean it up
|
|
//
|
|
|
|
d = Hive->Storage[Stable].Map;
|
|
if( d != NULL ) {
|
|
HvpFreeMap(Hive, d, 0, Tables);
|
|
(Hive->Free)(d, sizeof(HMAP_DIRECTORY));
|
|
}
|
|
} else {
|
|
//
|
|
// no directory, just a smalldir
|
|
//
|
|
(Hive->Free)(Hive->Storage[Stable].SmallDir, sizeof(HMAP_TABLE));
|
|
}
|
|
|
|
Hive->Storage[Stable].SmallDir = NULL;
|
|
Hive->Storage[Stable].Map = NULL;
|
|
|
|
}
|
|
|
|
VOID
|
|
HvpFreeMap(
|
|
PHHIVE Hive,
|
|
PHMAP_DIRECTORY Dir,
|
|
ULONG Start,
|
|
ULONG End
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Sweeps through the directory Dir points to and frees Tables.
|
|
Will free Start-th through End-th entries, INCLUSIVE.
|
|
|
|
Arguments:
|
|
|
|
Hive - supplies pointer to hive control block of interest
|
|
|
|
Dir - supplies address of an HMAP_DIRECTORY structure
|
|
|
|
Start - index of first map table pointer to clean up
|
|
|
|
End - index of last map table pointer to clean up
|
|
|
|
Return Value:
|
|
|
|
NONE.
|
|
|
|
--*/
|
|
{
|
|
ULONG i;
|
|
|
|
if (End >= HDIRECTORY_SLOTS) {
|
|
End = HDIRECTORY_SLOTS - 1;
|
|
}
|
|
|
|
for (i = Start; i <= End; i++) {
|
|
if (Dir->Directory[i] != NULL) {
|
|
(Hive->Free)(Dir->Directory[i], sizeof(HMAP_TABLE));
|
|
Dir->Directory[i] = NULL;
|
|
}
|
|
}
|
|
return;
|
|
}
|
|
|
|
BOOLEAN
|
|
HvpAllocateMap(
|
|
PHHIVE Hive,
|
|
PHMAP_DIRECTORY Dir,
|
|
ULONG Start,
|
|
ULONG End
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Sweeps through the directory Dir points to and allocates Tables.
|
|
Will allocate Start-th through End-th entries, INCLUSIVE.
|
|
|
|
Does NOT clean up when out of memory, call HvpFreeMap to do that.
|
|
Arguments:
|
|
|
|
Hive - supplies pointer to hive control block of interest
|
|
|
|
Dir - supplies address of an HMAP_DIRECTORY structure
|
|
|
|
Start - index of first map table pointer to allocate for
|
|
|
|
End - index of last map table pointer to allocate for
|
|
|
|
Return Value:
|
|
|
|
TRUE - it worked
|
|
|
|
FALSE - insufficient memory
|
|
|
|
--*/
|
|
{
|
|
ULONG i,j;
|
|
PHMAP_TABLE t;
|
|
|
|
for (i = Start; i <= End; i++) {
|
|
ASSERT(Dir->Directory[i] == NULL);
|
|
t = (PHMAP_TABLE)((Hive->Allocate)(sizeof(HMAP_TABLE), FALSE,CM_FIND_LEAK_TAG29));
|
|
if (t == NULL) {
|
|
return FALSE;
|
|
}
|
|
// the zero memory stuff can be removed
|
|
RtlZeroMemory(t, sizeof(HMAP_TABLE));
|
|
for(j=0;j<HTABLE_SLOTS;j++) {
|
|
//
|
|
// Invalidate the entry
|
|
//
|
|
|
|
//
|
|
// ATTENTION : I don't really think we need this !!! <TBD>
|
|
//
|
|
|
|
t->Table[j].BinAddress = 0;
|
|
// we don't need to set this - just for debug purposes
|
|
ASSERT( (t->Table[j].CmView = NULL) == NULL );
|
|
}
|
|
|
|
Dir->Directory[i] = t;
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
ULONG
|
|
HvpGetBinMemAlloc(
|
|
IN PHHIVE Hive,
|
|
PHBIN Bin,
|
|
IN HSTORAGE_TYPE Type
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Returns the bin MemAlloc (formelly kept right in the bin) by looking at
|
|
the map. We need this to avoid touching the bins only to set their MemAlloc.
|
|
|
|
|
|
Arguments:
|
|
|
|
Hive - supplies a pointer to the hive control structure for the
|
|
hive of interest
|
|
|
|
Bin - The bin in question
|
|
|
|
Type - Stable or Volatile
|
|
|
|
Return Value:
|
|
|
|
Pointer to the new BIN if we succeeded, NULL if we failed.
|
|
|
|
--*/
|
|
{
|
|
PHMAP_ENTRY Map;
|
|
HCELL_INDEX Cell;
|
|
|
|
#if DBG
|
|
ULONG i;
|
|
PHMAP_ENTRY Me;
|
|
#endif
|
|
|
|
#ifndef _CM_LDR_
|
|
PAGED_CODE();
|
|
#endif //_CM_LDR_
|
|
|
|
ASSERT( Bin->Signature == HBIN_SIGNATURE );
|
|
|
|
Cell = Bin->FileOffset + (Type * HCELL_TYPE_MASK);
|
|
|
|
Map = HvpGetCellMap(Hive, Cell);
|
|
VALIDATE_CELL_MAP(__LINE__,Map,Hive,Cell);
|
|
|
|
#if DBG
|
|
//
|
|
// some validation code
|
|
//
|
|
for( i=0;i<Bin->Size;i+=HBLOCK_SIZE) {
|
|
Cell = Bin->FileOffset + i + (Type * HCELL_TYPE_MASK);
|
|
Me = HvpGetCellMap(Hive, Cell);
|
|
VALIDATE_CELL_MAP(__LINE__,Me,Hive,Cell);
|
|
|
|
if( i == 0 ) {
|
|
ASSERT( Me->MemAlloc != 0 );
|
|
} else {
|
|
ASSERT( Me->MemAlloc == 0 );
|
|
}
|
|
}
|
|
#endif
|
|
|
|
return Map->MemAlloc;
|
|
}
|
|
|
|
|