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

602 lines
14 KiB
C++

/*++
Copyright (c) 1999-2001 Microsoft Corporation
Module Name:
mmap.cpp
Abstract:
Implementation of memory map class.
Author:
Matthew D Hendel (math) 16-Sep-1999
Revision History:
--*/
#include "ntsdp.hpp"
MappedMemoryMap::MappedMemoryMap(void)
{
m_RegionCount = 0;
m_List = NULL;
}
MappedMemoryMap::~MappedMemoryMap(void)
{
PMEMORY_MAP_ENTRY Entry;
PMEMORY_MAP_ENTRY Next;
Entry = m_List;
while ( Entry != NULL )
{
Next = Entry->Next;
free ( Entry );
Entry = Next;
}
}
PMEMORY_MAP_ENTRY
MappedMemoryMap::AddMapEntry(ULONG64 BaseOfRegion, ULONG SizeOfRegion,
PVOID Buffer, PVOID UserData,
BOOL AllowOverlap)
{
PMEMORY_MAP_ENTRY PrevEntry;
PMEMORY_MAP_ENTRY MapEntry;
MapEntry = (MEMORY_MAP_ENTRY *)malloc ( sizeof ( MEMORY_MAP_ENTRY ) );
if (!MapEntry)
{
return NULL;
}
MapEntry->BaseOfRegion = BaseOfRegion;
MapEntry->SizeOfRegion = SizeOfRegion;
MapEntry->Region = Buffer;
MapEntry->UserData = UserData;
MapEntry->AllowOverlap = AllowOverlap;
MapEntry->Next = NULL;
//
// Insert the element.
//
PrevEntry = FindPreceedingRegion(BaseOfRegion);
if ( PrevEntry == NULL )
{
//
// Insert at head.
//
MapEntry->Next = m_List;
m_List = MapEntry;
}
else
{
//
// Insert in order.
//
MapEntry->Next = PrevEntry->Next;
PrevEntry->Next = MapEntry;
}
m_RegionCount++;
return MapEntry;
}
HRESULT
MappedMemoryMap::AddRegion(
IN ULONG64 BaseOfRegion,
IN ULONG SizeOfRegion,
IN PVOID Buffer,
IN PVOID UserData,
IN BOOL AllowOverlap
)
{
//
// The region size cannot be zero.
//
if (SizeOfRegion == 0)
{
ErrOut("**** MappedMemoryMap::AddRegion: Empty region being added.\n");
return S_OK;
}
if (IsBadReadPtr(Buffer, SizeOfRegion))
{
ErrOut("**** MappedMemoryMap::AddRegion: Mapping too small to map "
"%s:%X from %p\n", FormatAddr64(BaseOfRegion),
SizeOfRegion, Buffer);
return E_INVALIDARG;
}
ULONG64 EndOfRegion;
PMEMORY_MAP_ENTRY Entry;
ULONG Size;
while (SizeOfRegion > 0)
{
//
// Find the first overlapping region.
// We need to rescan the whole list as it
// may have changed due to insertion of fragments
// of the new region.
//
EndOfRegion = BaseOfRegion + SizeOfRegion;
for (Entry = m_List; Entry != NULL; Entry = Entry->Next)
{
if (EndOfRegion > Entry->BaseOfRegion &&
Entry->BaseOfRegion >= BaseOfRegion)
{
if (AllowOverlap || Entry->AllowOverlap)
{
// Overlap can occur when a stack, ethread or
// eprocess is taken from static data in an image.
// For example, the x86 idle process is static
// data in ntoskrnl. Triage dumps contain the
// eprocess data and it gets mapped and can overlap
// with the ntoskrnl image.
#if 0
WarnOut("WARNING: Allowing overlapped region %s - %s\n",
FormatAddr64(BaseOfRegion),
FormatAddr64(BaseOfRegion + SizeOfRegion - 1));
#endif
Entry = NULL;
}
break;
}
}
if (Entry == NULL ||
BaseOfRegion < Entry->BaseOfRegion)
{
// There's a portion of the beginning of the new
// region which does not overlap so add it and
// trim the description.
Size = Entry == NULL ? SizeOfRegion :
(ULONG)(Entry->BaseOfRegion - BaseOfRegion);
if (!AddMapEntry(BaseOfRegion, Size, Buffer, UserData,
AllowOverlap))
{
return E_OUTOFMEMORY;
}
if (Size == SizeOfRegion)
{
// None of the region overlapped so we're done.
return S_OK;
}
BaseOfRegion += Size;
SizeOfRegion -= Size;
Buffer = (PUCHAR)Buffer + Size;
}
//
// Now handle the overlap.
//
if (SizeOfRegion > Entry->SizeOfRegion)
{
Size = Entry->SizeOfRegion;
}
else
{
Size = SizeOfRegion;
}
int Compare;
__try
{
Compare = memcmp(Buffer, Entry->Region, Size);
}
__except(MappingExceptionFilter(GetExceptionInformation()))
{
return HRESULT_FROM_NT(GetExceptionCode());
}
if (Compare)
{
ErrOut("**** MappedMemoryMap::AddRegion: "
"Conflicting region %s - %s\n",
FormatAddr64(BaseOfRegion),
FormatAddr64(BaseOfRegion + SizeOfRegion - 1));
return HR_REGION_CONFLICT;
}
// Overlap region data matched so it's OK to just
// trim the overlap out of the new region and move
// on to the next possible overlap.
BaseOfRegion += Size;
SizeOfRegion -= Size;
Buffer = (PUCHAR)Buffer + Size;
}
return S_OK;
}
BOOL
MappedMemoryMap::ReadMemory(
IN ULONG64 BaseOfRange,
IN OUT PVOID Buffer,
IN ULONG SizeOfRange,
OUT ULONG * BytesRead
)
/*++
Routine Description:
Read memory from the memory map. ReadMemory can read across regions, as
long as there is no unallocated space between regions.
This routine will do a partial read if the region ends before
SizeOfRange bytes have been read. In that case BytesRead will return the
number of bytes actually read.
Arguments:
BaseOfRange - The base address where we want to read memory from.
SizeOfRange - The length of the region to read memory from.
Buffer - A pointer to a buffer to read memory into.
BytesRead - On success, the number of bytes successfully read.
Return Values:
TRUE - If any number of bytes were successfully read from the memory map.
FALSE - If no bytes were read.
--*/
{
ULONG BytesToReadFromRegion;
PMEMORY_MAP_ENTRY Entry;
ULONG64 BaseToRead;
ULONG SizeToRead;
PBYTE BufferForRead;
ULONG_PTR OffsetToRead;
ULONG AvailSize;
//
// We return TRUE if we read any bytes successfully and FALSE if not.
//
*BytesRead = 0;
BaseToRead = BaseOfRange;
SizeToRead = SizeOfRange;
BufferForRead = (PBYTE) Buffer;
do
{
Entry = FindContainingRegion(BaseToRead);
if ( !Entry )
{
if (*BytesRead)
{
return TRUE;
}
else
{
return FALSE;
}
}
PMEMORY_MAP_ENTRY NextEntry = Entry->Next;
// Due to overlap there may be other entries
// that need to be processed even before the
// end of the containing region.
AvailSize = Entry->SizeOfRegion;
while (NextEntry != NULL)
{
if (NextEntry->BaseOfRegion > BaseToRead)
{
ULONG64 EntryDiff =
NextEntry->BaseOfRegion - Entry->BaseOfRegion;
if (EntryDiff < AvailSize)
{
AvailSize = (ULONG)EntryDiff;
}
break;
}
NextEntry = NextEntry->Next;
}
if (BaseToRead + SizeToRead > Entry->BaseOfRegion + AvailSize)
{
BytesToReadFromRegion = (ULONG)
(Entry->BaseOfRegion - BaseToRead) + AvailSize;
}
else
{
BytesToReadFromRegion = SizeToRead;
}
OffsetToRead = (ULONG_PTR) (BaseToRead - Entry->BaseOfRegion);
__try
{
RtlCopyMemory (BufferForRead,
(PBYTE)Entry->Region + OffsetToRead,
BytesToReadFromRegion
);
}
__except(MappingExceptionFilter(GetExceptionInformation()))
{
return FALSE;
}
*BytesRead += BytesToReadFromRegion;
BaseToRead += BytesToReadFromRegion;
SizeToRead -= BytesToReadFromRegion;
BufferForRead += BytesToReadFromRegion;
} while ( SizeToRead );
return TRUE;
}
BOOL
MappedMemoryMap::GetRegionInfo(
IN ULONG64 Addr,
OUT ULONG64* BaseOfRegion, OPTIONAL
OUT ULONG* SizeOfRegion, OPTIONAL
OUT PVOID* Buffer, OPTIONAL
OUT PVOID* UserData OPTIONAL
)
/*++
Routine Description:
Get information about the region containing the address Addr.
Arguments:
Addr - An address that is contained within some region in the map.
BaseOfRegion - Pointer to a buffer to return the region base.
SizeOfRegion - Pointer to a buffer to retutn the region size.
Buffer - Pointer to a buffer to return the region buffer pointer.
UserData - Pointer to a buffer to return the region client param.
Return Values:
TRUE - On success.
FALSE - On failure.
--*/
{
PMEMORY_MAP_ENTRY Entry;
Entry = FindContainingRegion(Addr);
if ( !Entry )
{
return FALSE;
}
if ( BaseOfRegion )
{
*BaseOfRegion = Entry->BaseOfRegion;
}
if ( SizeOfRegion )
{
*SizeOfRegion = Entry->SizeOfRegion;
}
if ( Buffer )
{
*Buffer = Entry->Region;
}
if ( UserData )
{
*UserData = Entry->UserData;
}
return TRUE;
}
BOOL
MappedMemoryMap::GetNextRegion(
IN ULONG64 Addr,
OUT PULONG64 Next
)
{
PMEMORY_MAP_ENTRY Entry;
Entry = m_List;
while (Entry != NULL)
{
//
// Assuming they're in order.
//
if (Entry->BaseOfRegion > Addr)
{
*Next = Entry->BaseOfRegion;
return TRUE;
}
Entry = Entry->Next;
}
return FALSE;
}
BOOL
MappedMemoryMap::RemoveRegion(
IN ULONG64 BaseOfRegion,
IN ULONG SizeOfRegion
)
{
// XXX drewb - This should carve the given region out of
// any existing regions. Right now we don't need general
// removal functionality though so only handle the case
// where the requested removal is an exact single region
// match.
PMEMORY_MAP_ENTRY PrevEntry;
PMEMORY_MAP_ENTRY Entry;
PrevEntry = FindPreceedingRegion(BaseOfRegion);
if (PrevEntry != NULL)
{
Entry = PrevEntry->Next;
}
else
{
Entry = m_List;
}
if (Entry == NULL)
{
ErrOut("MappedMemoryMap::RemoveRegion NULL region for %s:%x\n",
FormatAddr64(BaseOfRegion), SizeOfRegion);
return FALSE;
}
else if (Entry->BaseOfRegion != BaseOfRegion ||
Entry->SizeOfRegion != SizeOfRegion)
{
ErrOut("MappedMemoryMap::RemoveRegion region mismatch: "
"%s:%x vs. entry %s:%x\n",
FormatAddr64(BaseOfRegion), SizeOfRegion,
FormatAddr64(Entry->BaseOfRegion), Entry->SizeOfRegion);
return FALSE;
}
if (PrevEntry == NULL)
{
m_List = Entry->Next;
}
else
{
PrevEntry->Next = Entry->Next;
}
free(Entry);
m_RegionCount--;
return TRUE;
}
BOOL
MappedMemoryMap::GetRegionByUserData(
IN PVOID UserData,
OUT PULONG64 Base,
OUT PULONG Size
)
{
PMEMORY_MAP_ENTRY Entry;
Entry = m_List;
while (Entry != NULL)
{
if (Entry->UserData == UserData)
{
*Base = Entry->BaseOfRegion;
*Size = Entry->SizeOfRegion;
return TRUE;
}
Entry = Entry->Next;
}
return FALSE;
}
//
// Private functions
//
PMEMORY_MAP_ENTRY
MappedMemoryMap::FindPreceedingRegion(
IN ULONG64 BaseOfRegion
)
{
PMEMORY_MAP_ENTRY PrevEntry;
PMEMORY_MAP_ENTRY Entry;
PrevEntry = NULL;
Entry = m_List;
while (Entry != NULL)
{
//
// Assuming they're in order.
//
if ( Entry->BaseOfRegion >= BaseOfRegion )
{
return PrevEntry;
}
PrevEntry = Entry;
Entry = Entry->Next;
}
return PrevEntry;
}
PMEMORY_MAP_ENTRY
MappedMemoryMap::FindContainingRegion(
IN ULONG64 Addr
)
{
PMEMORY_MAP_ENTRY Entry;
PMEMORY_MAP_ENTRY ReturnEntry = NULL;
Entry = m_List;
//
// We may have overlapping regions, so keep going until we find
// the most precise one (assumed to be the one we care about)
//
while ( Entry != NULL )
{
if ( Entry->BaseOfRegion <= Addr &&
Addr < Entry->BaseOfRegion + Entry->SizeOfRegion)
{
ReturnEntry = Entry;
}
else if (ReturnEntry != NULL &&
Addr >= ReturnEntry->BaseOfRegion + ReturnEntry->SizeOfRegion)
{
// Optimization - we can stop searching if we've already
// found a block and have now left its region. We can't
// stop as long as we're in its region as there may
// be more exact overlaps anywhere within the entire region.
break;
}
Entry = Entry->Next;
}
return ReturnEntry;
}