/*++ 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 #include #include #endif #if DBG && !defined(NTOS_KERNEL_RUNTIME) extern PVOID NtDllBase; // defined in ntos\dll\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 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. // ExAcquireSpinLock(&PsLoadedModuleSpinLock, &OldIrql); 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)Base + Entry->SizeOfImage; if (((ULONG)PcValue >= (ULONG)Base) && ((ULONG)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 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; // // 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. // if ( !RtlTryEnterCriticalSection((PRTL_CRITICAL_SECTION)NtCurrentPeb()->LoaderLock) ) { // // We could not get the loader lock, so call the system to find the image that // contains this pc // st = NtQueryVirtualMemory( NtCurrentProcess(), PcValue, MemoryBasicInformation, (PVOID)&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; } 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)Base + NtHeaders->OptionalHeader.SizeOfImage; if (((ULONG)PcValue >= (ULONG)Base) && ((ULONG)PcValue < Bounds)) { RtlLeaveCriticalSection((PRTL_CRITICAL_SECTION)NtCurrentPeb()->LoaderLock); *BaseOfImage = Base; return Base; } } } #if DBG } 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. Since we really only need DbgPrints // on DBG builds, only defined this code for DBG to keep // retail code smaller. // if ( NtDllBase != NULL ) { Base = NtDllBase; NtHeaders = RtlImageNtHeader( Base ); Bounds = (ULONG)Base + NtHeaders->OptionalHeader.SizeOfImage; if (((ULONG)PcValue >= (ULONG)Base) && ((ULONG)PcValue < Bounds)) { RtlLeaveCriticalSection((PRTL_CRITICAL_SECTION)NtCurrentPeb()->LoaderLock); *BaseOfImage = Base; return Base; } } #endif // DBG } } } except(EXCEPTION_EXECUTE_HANDLER) { NOTHING; } // // Release the Loader lock for the current process a return NULL. // RtlLeaveCriticalSection((PRTL_CRITICAL_SECTION)NtCurrentPeb()->LoaderLock); *BaseOfImage = NULL; return NULL; #endif }