Windows NT 4.0 source code leak
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

452 lines
11 KiB

/*++
Copyright (c) 1994-96 Microsoft Corporation
Module Name:
map.c
Abstract:
Implementation for the MapAndLoad API
Author:
Revision History:
--*/
#include <private.h>
BOOL
MapAndLoad(
LPSTR ImageName,
LPSTR DllPath,
PLOADED_IMAGE LoadedImage,
BOOL DotDll,
BOOL ReadOnly
)
{
HANDLE hFile;
HANDLE hMappedFile;
CHAR SearchBuffer[MAX_PATH];
DWORD dw;
LPSTR FilePart;
LPSTR OpenName;
// open and map the file.
// then fill in the loaded image descriptor
LoadedImage->hFile = INVALID_HANDLE_VALUE;
OpenName = ImageName;
dw = 0;
retry:
hFile = CreateFile(
OpenName,
ReadOnly ? GENERIC_READ : GENERIC_READ | GENERIC_WRITE,
FILE_SHARE_READ | FILE_SHARE_WRITE,
NULL,
OPEN_EXISTING,
0,
NULL
);
if ( hFile == INVALID_HANDLE_VALUE ) {
if ( !dw ) {
//
// open failed try to find the file on the search path
//
dw = SearchPath(
DllPath,
ImageName,
DotDll ? ".dll" : ".exe",
MAX_PATH,
SearchBuffer,
&FilePart
);
if ( dw && dw < MAX_PATH ) {
OpenName = SearchBuffer;
goto retry;
}
}
return FALSE;
}
if (MapIt(hFile, LoadedImage, ReadOnly) == FALSE) {
CloseHandle(hFile);
return FALSE;
} else {
LoadedImage->ModuleName = (LPSTR) MemAlloc( strlen(OpenName)+16 );
strcpy( LoadedImage->ModuleName, OpenName );
// If readonly, no need to keep the file open..
if (ReadOnly) {
CloseHandle(hFile);
}
return TRUE;
}
}
BOOL
MapIt(
HANDLE hFile,
PLOADED_IMAGE LoadedImage,
BOOL ReadOnly
)
{
HANDLE hMappedFile;
hMappedFile = CreateFileMapping(
hFile,
NULL,
ReadOnly ? PAGE_READONLY : PAGE_READWRITE,
0,
0,
NULL
);
if ( !hMappedFile ) {
return FALSE;
}
LoadedImage->MappedAddress = (PUCHAR) MapViewOfFile(
hMappedFile,
ReadOnly ? FILE_MAP_READ : FILE_MAP_WRITE,
0,
0,
0
);
CloseHandle(hMappedFile);
if (!LoadedImage->MappedAddress ||
!CalculateImagePtrs(LoadedImage)) {
return(FALSE);
}
LoadedImage->SizeOfImage = GetFileSize(hFile, NULL);
if (ReadOnly) {
LoadedImage->hFile = INVALID_HANDLE_VALUE;
} else {
LoadedImage->hFile = hFile;
}
return(TRUE);
}
BOOL
CalculateImagePtrs(
PLOADED_IMAGE LoadedImage
)
{
PIMAGE_DOS_HEADER DosHeader;
BOOL fRC;
// Everything is mapped. Now check the image and find nt image headers
fRC = TRUE; // Assume the best
__try {
DosHeader = (PIMAGE_DOS_HEADER)LoadedImage->MappedAddress;
if ((DosHeader->e_magic != IMAGE_DOS_SIGNATURE) &&
(DosHeader->e_magic != IMAGE_NT_SIGNATURE)) {
fRC = FALSE;
goto tryout;
}
if (DosHeader->e_magic == IMAGE_DOS_SIGNATURE) {
if (DosHeader->e_lfanew == 0) {
LoadedImage->fDOSImage = TRUE;
fRC = FALSE;
goto tryout;
}
LoadedImage->FileHeader = (PIMAGE_NT_HEADERS)((ULONG)DosHeader + DosHeader->e_lfanew);
} else {
// No DOS header indicates an image built w/o a dos stub
LoadedImage->FileHeader = (PIMAGE_NT_HEADERS)((ULONG)DosHeader);
}
if ( LoadedImage->FileHeader->Signature != IMAGE_NT_SIGNATURE ) {
if ( (USHORT)LoadedImage->FileHeader->Signature == (USHORT)IMAGE_OS2_SIGNATURE ||
(USHORT)LoadedImage->FileHeader->Signature == (USHORT)IMAGE_OS2_SIGNATURE_LE
) {
LoadedImage->fDOSImage = TRUE;
}
fRC = FALSE;
goto tryout;
} else {
LoadedImage->fDOSImage = FALSE;
}
// No optional header indicates an object...
if ( !LoadedImage->FileHeader->FileHeader.SizeOfOptionalHeader ) {
fRC = FALSE;
goto tryout;
}
// Image Base > 0x80000000 indicates a system image
if ( LoadedImage->FileHeader->OptionalHeader.ImageBase >= 0x80000000 ) {
LoadedImage->fSystemImage = TRUE;
} else {
LoadedImage->fSystemImage = FALSE;
}
// Check for versions < 2.50
if ( LoadedImage->FileHeader->OptionalHeader.MajorLinkerVersion < 3 &&
LoadedImage->FileHeader->OptionalHeader.MinorLinkerVersion < 5 ) {
fRC = FALSE;
goto tryout;
}
InitializeListHead( &LoadedImage->Links );
LoadedImage->Characteristics = LoadedImage->FileHeader->FileHeader.Characteristics;
LoadedImage->NumberOfSections = LoadedImage->FileHeader->FileHeader.NumberOfSections;
LoadedImage->Sections = (PIMAGE_SECTION_HEADER)((ULONG)LoadedImage->FileHeader + sizeof(IMAGE_NT_HEADERS));
LoadedImage->LastRvaSection = LoadedImage->Sections;
tryout:
if (fRC == FALSE) {
UnmapViewOfFile(LoadedImage->MappedAddress);
}
} __except ( EXCEPTION_EXECUTE_HANDLER ) {
fRC = FALSE;
}
return fRC;
}
BOOL
UnMapAndLoad(
PLOADED_IMAGE pLi
)
{
UnMapIt(pLi);
if (pLi->hFile != INVALID_HANDLE_VALUE)
CloseHandle(pLi->hFile);
return TRUE;
}
BOOL
GrowMap (
PLOADED_IMAGE pLi,
LONG lSizeOfDelta
)
{
if (pLi->hFile == INVALID_HANDLE_VALUE) {
// Can't grow read/only files.
return FALSE;
} else {
HANDLE hMappedFile;
FlushViewOfFile(pLi->MappedAddress, pLi->SizeOfImage);
UnmapViewOfFile(pLi->MappedAddress);
pLi->SizeOfImage += lSizeOfDelta;
SetFilePointer(pLi->hFile, pLi->SizeOfImage, NULL, FILE_BEGIN);
SetEndOfFile(pLi->hFile);
hMappedFile = CreateFileMapping(
pLi->hFile,
NULL,
PAGE_READWRITE,
0,
pLi->SizeOfImage,
NULL
);
if ( !hMappedFile ) {
return FALSE;
}
pLi->MappedAddress = (PUCHAR) MapViewOfFile(
hMappedFile,
FILE_MAP_WRITE,
0,
0,
0
);
CloseHandle(hMappedFile);
return TRUE;
}
}
VOID
UnMapIt(
PLOADED_IMAGE pLi
)
{
DWORD HeaderSum, CheckSum;
BOOL bl;
DWORD dw;
// Test for read-only
if (pLi->hFile == INVALID_HANDLE_VALUE) {
UnmapViewOfFile(pLi->MappedAddress);
} else {
CheckSumMappedFile( pLi->MappedAddress,
pLi->SizeOfImage,
&HeaderSum,
&CheckSum
);
pLi->FileHeader->OptionalHeader.CheckSum = CheckSum;
FlushViewOfFile(pLi->MappedAddress, pLi->SizeOfImage);
UnmapViewOfFile(pLi->MappedAddress);
if (pLi->SizeOfImage != GetFileSize(pLi->hFile, NULL)) {
dw = SetFilePointer(pLi->hFile, pLi->SizeOfImage, NULL, FILE_BEGIN);
dw = GetLastError();
bl = SetEndOfFile(pLi->hFile);
dw = GetLastError();
}
}
}
BOOL
GetImageConfigInformation(
PLOADED_IMAGE LoadedImage,
PIMAGE_LOAD_CONFIG_DIRECTORY ImageConfigInformation
)
{
PIMAGE_LOAD_CONFIG_DIRECTORY ImageConfigData;
ULONG i;
ImageConfigData = (PIMAGE_LOAD_CONFIG_DIRECTORY) ImageDirectoryEntryToData( LoadedImage->MappedAddress,
FALSE,
IMAGE_DIRECTORY_ENTRY_LOAD_CONFIG,
&i
);
if (ImageConfigData != NULL && i == sizeof( *ImageConfigData )) {
memcpy( ImageConfigInformation, ImageConfigData, sizeof( *ImageConfigData ) );
return TRUE;
}
else {
return FALSE;
}
}
BOOL
SetImageConfigInformation(
PLOADED_IMAGE LoadedImage,
PIMAGE_LOAD_CONFIG_DIRECTORY ImageConfigInformation
)
{
PIMAGE_LOAD_CONFIG_DIRECTORY ImageConfigData;
ULONG i, DirectoryAddress;
if (LoadedImage->hFile == INVALID_HANDLE_VALUE) {
return FALSE;
}
ImageConfigData = (PIMAGE_LOAD_CONFIG_DIRECTORY) ImageDirectoryEntryToData( LoadedImage->MappedAddress,
FALSE,
IMAGE_DIRECTORY_ENTRY_LOAD_CONFIG,
&i
);
if (ImageConfigData != NULL && i == sizeof( *ImageConfigData )) {
memcpy( ImageConfigData, ImageConfigInformation, sizeof( *ImageConfigData ) );
return TRUE;
}
DirectoryAddress = GetImageUnusedHeaderBytes( LoadedImage, &i );
if (i < sizeof(*ImageConfigData)) {
return FALSE;
}
LoadedImage->FileHeader->OptionalHeader.DataDirectory[ IMAGE_DIRECTORY_ENTRY_LOAD_CONFIG ].VirtualAddress = DirectoryAddress;
LoadedImage->FileHeader->OptionalHeader.DataDirectory[ IMAGE_DIRECTORY_ENTRY_LOAD_CONFIG ].Size = sizeof( *ImageConfigData );
ImageConfigData = (PIMAGE_LOAD_CONFIG_DIRECTORY)
((PCHAR)LoadedImage->MappedAddress + DirectoryAddress);
memcpy( ImageConfigData, ImageConfigInformation, sizeof( *ImageConfigData ) );
return TRUE;
}
BOOLEAN ImageLoadInit;
LIST_ENTRY ImageLoadList;
PLOADED_IMAGE
ImageLoad(
LPSTR DllName,
LPSTR DllPath
)
{
PLIST_ENTRY Head,Next;
PLOADED_IMAGE LoadedImage;
if (!ImageLoadInit) {
InitializeListHead( &ImageLoadList );
ImageLoadInit = TRUE;
}
Head = &ImageLoadList;
Next = Head->Flink;
while (Next != Head) {
LoadedImage = CONTAINING_RECORD( Next, LOADED_IMAGE, Links );
if (!_stricmp( DllName, LoadedImage->ModuleName )) {
return LoadedImage;
}
Next = Next->Flink;
}
LoadedImage = (PLOADED_IMAGE) MemAlloc( sizeof( *LoadedImage ) + strlen( DllName ) + 1 );
if (LoadedImage != NULL) {
LoadedImage->ModuleName = (LPSTR)(LoadedImage + 1);
strcpy( LoadedImage->ModuleName, DllName );
if (MapAndLoad( DllName, DllPath, LoadedImage, TRUE, TRUE )) {
InsertTailList( &ImageLoadList, &LoadedImage->Links );
return LoadedImage;
}
MemFree( LoadedImage );
LoadedImage = NULL;
}
return LoadedImage;
}
BOOL
ImageUnload(
PLOADED_IMAGE LoadedImage
)
{
if (!IsListEmpty( &LoadedImage->Links )) {
RemoveEntryList( &LoadedImage->Links );
}
UnMapAndLoad( LoadedImage );
MemFree( LoadedImage );
return TRUE;
}
BOOL
MarkImageAsRunFromSwap(
LPSTR ImageName,
ULONG Flags
)
{
return(FALSE);
}