Windows2003-3790/base/ntos/rtl/pctohdr.c
2020-09-30 16:53:55 +02:00

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
}