240 lines
6.9 KiB
C
240 lines
6.9 KiB
C
/*++
|
|
|
|
Copyright (c) 1992 Microsoft Corporation
|
|
|
|
Module Name:
|
|
|
|
pctohdr.c
|
|
|
|
Abstract:
|
|
|
|
This module implements code to locate the file header for an image or
|
|
dll given a PC value that lies within the image.
|
|
|
|
N.B. This routine is conditionalized for user mode and kernel mode.
|
|
|
|
Author:
|
|
|
|
Steve Wood (stevewo) 18-Aug-1989
|
|
|
|
Environment:
|
|
|
|
User Mode or Kernel Mode
|
|
|
|
Revision History:
|
|
|
|
--*/
|
|
|
|
#if defined(NTOS_KERNEL_RUNTIME)
|
|
#include "ntos.h"
|
|
#else
|
|
#include <nt.h>
|
|
#include <ntrtl.h>
|
|
#include <nturtl.h>
|
|
#endif
|
|
|
|
#if !defined(NTOS_KERNEL_RUNTIME)
|
|
extern PVOID NtDllBase; // defined in ntdll\ldrinit.c
|
|
#endif
|
|
|
|
PVOID
|
|
RtlPcToFileHeader(
|
|
IN PVOID PcValue,
|
|
OUT PVOID *BaseOfImage
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This function returns the base of an image that contains the
|
|
specified PcValue. An image contains the PcValue if the PcValue
|
|
is within the ImageBase, and the ImageBase plus the size of the
|
|
virtual image.
|
|
|
|
Arguments:
|
|
|
|
PcValue - Supplies a PcValue. All of the modules mapped into the
|
|
calling processes address space are scanned to compute which
|
|
module contains the PcValue.
|
|
|
|
BaseOfImage - Returns the base address for the image containing the
|
|
PcValue. This value must be added to any relative addresses in
|
|
the headers to locate portions of the image.
|
|
|
|
Return Value:
|
|
|
|
NULL - No image was found that contains the PcValue.
|
|
|
|
NON-NULL - Returns the base address of the image that contain the
|
|
PcValue.
|
|
|
|
--*/
|
|
|
|
{
|
|
|
|
#if defined(NTOS_KERNEL_RUNTIME)
|
|
|
|
extern LIST_ENTRY PsLoadedModuleList;
|
|
extern KSPIN_LOCK PsLoadedModuleSpinLock;
|
|
|
|
PVOID Base;
|
|
ULONG_PTR Bounds;
|
|
PLDR_DATA_TABLE_ENTRY Entry;
|
|
PLIST_ENTRY Next;
|
|
KIRQL OldIrql;
|
|
|
|
//
|
|
// Acquire the loaded module list spinlock and scan the list for the
|
|
// specified PC value if the list has been initialized.
|
|
//
|
|
|
|
OldIrql = KeGetCurrentIrql();
|
|
if (OldIrql < DISPATCH_LEVEL) {
|
|
KeRaiseIrqlToDpcLevel();
|
|
}
|
|
|
|
ExAcquireSpinLockAtDpcLevel(&PsLoadedModuleSpinLock);
|
|
Next = PsLoadedModuleList.Flink;
|
|
if (Next != NULL) {
|
|
while (Next != &PsLoadedModuleList) {
|
|
Entry = CONTAINING_RECORD(Next,
|
|
LDR_DATA_TABLE_ENTRY,
|
|
InLoadOrderLinks);
|
|
|
|
Next = Next->Flink;
|
|
Base = Entry->DllBase;
|
|
Bounds = (ULONG_PTR)Base + Entry->SizeOfImage;
|
|
if (((ULONG_PTR)PcValue >= (ULONG_PTR)Base) && ((ULONG_PTR)PcValue < Bounds)) {
|
|
ExReleaseSpinLock(&PsLoadedModuleSpinLock, OldIrql);
|
|
*BaseOfImage = Base;
|
|
return Base;
|
|
}
|
|
}
|
|
}
|
|
|
|
//
|
|
// Release the loaded module list spin lock and return NULL.
|
|
//
|
|
|
|
ExReleaseSpinLock(&PsLoadedModuleSpinLock, OldIrql);
|
|
*BaseOfImage = NULL;
|
|
return NULL;
|
|
|
|
#else
|
|
|
|
PVOID Base;
|
|
ULONG_PTR Bounds;
|
|
PLDR_DATA_TABLE_ENTRY Entry;
|
|
PLIST_ENTRY ModuleListHead;
|
|
PLIST_ENTRY Next;
|
|
PIMAGE_NT_HEADERS NtHeaders;
|
|
PPEB Peb;
|
|
PTEB Teb;
|
|
MEMORY_BASIC_INFORMATION MemInfo;
|
|
NTSTATUS st;
|
|
ULONG LoaderLockDisposition;
|
|
PVOID LockCookie = NULL;
|
|
|
|
//
|
|
// Acquire the Loader lock for the current process and scan the loaded
|
|
// module list for the specified PC value if all the data structures
|
|
// have been initialized.
|
|
//
|
|
|
|
LdrLockLoaderLock(LDR_LOCK_LOADER_LOCK_FLAG_TRY_ONLY | LDR_LOCK_LOADER_LOCK_FLAG_RAISE_ON_ERRORS, &LoaderLockDisposition, &LockCookie);
|
|
|
|
if (LoaderLockDisposition == LDR_LOCK_LOADER_LOCK_DISPOSITION_LOCK_NOT_ACQUIRED) {
|
|
//
|
|
// We could not get the loader lock, so call the system to find the image that
|
|
// contains this pc
|
|
//
|
|
|
|
st = NtQueryVirtualMemory(
|
|
NtCurrentProcess(),
|
|
PcValue,
|
|
MemoryBasicInformation,
|
|
&MemInfo,
|
|
sizeof(MemInfo),
|
|
NULL);
|
|
if ( !NT_SUCCESS(st) ) {
|
|
MemInfo.AllocationBase = NULL;
|
|
} else {
|
|
if ( MemInfo.Type == MEM_IMAGE ) {
|
|
try {
|
|
*BaseOfImage = MemInfo.AllocationBase;
|
|
} except (EXCEPTION_EXECUTE_HANDLER) {
|
|
MemInfo.AllocationBase = NULL;
|
|
}
|
|
} else {
|
|
MemInfo.AllocationBase = NULL;;
|
|
}
|
|
}
|
|
return MemInfo.AllocationBase;
|
|
}
|
|
|
|
// If we *did* get the loader lock, let's avoid the syscall and search the tables.
|
|
__try {
|
|
Teb = NtCurrentTeb();
|
|
if (Teb != NULL) {
|
|
Peb = Teb->ProcessEnvironmentBlock;
|
|
if (Peb->Ldr != NULL) {
|
|
ModuleListHead = &Peb->Ldr->InLoadOrderModuleList;
|
|
Next = ModuleListHead->Flink;
|
|
if (Next != NULL) {
|
|
while (Next != ModuleListHead) {
|
|
Entry = CONTAINING_RECORD(Next, LDR_DATA_TABLE_ENTRY, InLoadOrderLinks);
|
|
Next = Next->Flink;
|
|
Base = Entry->DllBase;
|
|
NtHeaders = RtlImageNtHeader(Base);
|
|
Bounds = (ULONG_PTR)Base + NtHeaders->OptionalHeader.SizeOfImage;
|
|
if (((ULONG_PTR)PcValue >= (ULONG_PTR)Base) && ((ULONG_PTR)PcValue < Bounds)) {
|
|
goto Done;
|
|
}
|
|
}
|
|
}
|
|
|
|
} else {
|
|
|
|
//
|
|
// ( Peb->Ldr == NULL )
|
|
//
|
|
// If called during process intialization before the Ldr
|
|
// module list has been setup, code executing must be in
|
|
// NTDLL module. If NtDllBase is non-NULL and the PcValue
|
|
// falls into the NTDLL range, return a valid Base. This
|
|
// allows DbgPrint's during LdrpInitializeProcess to work
|
|
// on RISC machines.
|
|
//
|
|
|
|
if (NtDllBase != NULL) {
|
|
Base = NtDllBase;
|
|
NtHeaders = RtlImageNtHeader(Base);
|
|
if (NtHeaders == NULL) {
|
|
Base = NULL;
|
|
goto Done;
|
|
}
|
|
|
|
Bounds = (ULONG_PTR)Base + NtHeaders->OptionalHeader.SizeOfImage;
|
|
if (((ULONG_PTR)PcValue >= (ULONG_PTR)Base) && ((ULONG_PTR)PcValue < Bounds))
|
|
goto Done;
|
|
}
|
|
}
|
|
}
|
|
|
|
Base = NULL;
|
|
Done:
|
|
;
|
|
} __except(EXCEPTION_EXECUTE_HANDLER) {
|
|
Base = NULL;
|
|
}
|
|
|
|
LdrUnlockLoaderLock(LDR_UNLOCK_LOADER_LOCK_FLAG_RAISE_ON_ERRORS, LockCookie);
|
|
|
|
*BaseOfImage = Base;
|
|
return Base;
|
|
|
|
#endif
|
|
|
|
}
|