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

1123 lines
32 KiB
C++

//----------------------------------------------------------------------------
//
// Stack walking support.
//
// Copyright (C) Microsoft Corporation, 1997-2002.
//
//----------------------------------------------------------------------------
#include "ntsdp.hpp"
BOOL g_AllowCorStack = TRUE;
BOOL g_DebugCorStack;
IMAGE_IA64_RUNTIME_FUNCTION_ENTRY g_EpcRfeBuffer;
PIMAGE_IA64_RUNTIME_FUNCTION_ENTRY g_EpcRfe;
PFPO_DATA
SynthesizeKnownFpo(PSTR Symbol, ULONG64 OffStart, ULONG64 Disp)
{
static ULONG64 s_Nr2, s_Lu2, s_Eh3, s_Kuit;
if (!s_Nr2 || !s_Lu2 || !s_Eh3 || !s_Kuit)
{
GetOffsetFromSym(g_Process, "nt!_NLG_Return2", &s_Nr2, NULL);
GetOffsetFromSym(g_Process, "nt!_local_unwind2", &s_Lu2, NULL);
GetOffsetFromSym(g_Process, "nt!_except_handler3", &s_Eh3, NULL);
GetOffsetFromSym(g_Process,
"nt!KiUnexpectedInterruptTail", &s_Kuit, NULL);
}
if (OffStart == s_Nr2 || OffStart == s_Lu2)
{
static FPO_DATA s_Lu2Fpo;
s_Lu2Fpo.ulOffStart = (ULONG)OffStart;
s_Lu2Fpo.cbProcSize = 0x68;
s_Lu2Fpo.cdwLocals = 4;
s_Lu2Fpo.cdwParams = 0;
s_Lu2Fpo.cbProlog = 0;
s_Lu2Fpo.cbRegs = 3;
s_Lu2Fpo.fHasSEH = 0;
s_Lu2Fpo.fUseBP = 0;
s_Lu2Fpo.reserved = 0;
s_Lu2Fpo.cbFrame = FRAME_FPO;
return &s_Lu2Fpo;
}
else if (OffStart == s_Eh3)
{
static FPO_DATA s_Eh3Fpo;
s_Eh3Fpo.ulOffStart = (ULONG)OffStart;
s_Eh3Fpo.cbProcSize = 0xbd;
s_Eh3Fpo.cdwLocals = 2;
s_Eh3Fpo.cdwParams = 4;
s_Eh3Fpo.cbProlog = 3;
s_Eh3Fpo.cbRegs = 4;
s_Eh3Fpo.fHasSEH = 0;
s_Eh3Fpo.fUseBP = 0;
s_Eh3Fpo.reserved = 0;
s_Eh3Fpo.cbFrame = FRAME_NONFPO;
return &s_Eh3Fpo;
}
else if (OffStart == s_Kuit)
{
//
// KiUnexpectedInterruptTail has three special stubs
// following it for CommonDispatchException[0-2]Args.
// These stubs set up for the appropriate number of
// arguments and then call CommonDispatchException.
// They do not have symbols or FPO data so fake some
// up if we're in the region immediately after KUIT.
//
PFPO_DATA KuitData = (PFPO_DATA)
SymFunctionTableAccess(g_Process->m_SymHandle, OffStart);
if (KuitData != NULL &&
Disp >= (ULONG64)KuitData->cbProcSize &&
Disp < (ULONG64)KuitData->cbProcSize + 0x20)
{
static FPO_DATA s_CdeStubFpo;
s_CdeStubFpo.ulOffStart = (ULONG)OffStart;
s_CdeStubFpo.cbProcSize = 0x10;
s_CdeStubFpo.cdwLocals = 0;
s_CdeStubFpo.cdwParams = 0;
s_CdeStubFpo.cbProlog = 0;
s_CdeStubFpo.cbRegs = 0;
s_CdeStubFpo.fHasSEH = 0;
s_CdeStubFpo.fUseBP = 0;
s_CdeStubFpo.reserved = 0;
s_CdeStubFpo.cbFrame = FRAME_TRAP;
return &s_CdeStubFpo;
}
}
return NULL;
}
PFPO_DATA
SynthesizeFpoDataForModule(DWORD64 PCAddr)
{
DWORD64 Offset;
CHAR symbuf[MAX_SYMBOL_LEN];
GetSymbol(PCAddr, symbuf, sizeof(symbuf), &Offset);
if (Offset == PCAddr)
{
// No symbol.
return NULL;
}
PFPO_DATA KnownFpo =
SynthesizeKnownFpo(symbuf, PCAddr - Offset, Offset);
if (KnownFpo != NULL)
{
return KnownFpo;
}
// Not a known symbol so no FPO is available.
return NULL;
}
PFPO_DATA
SynthesizeFpoDataForFastSyscall(ULONG64 Offset)
{
static FPO_DATA s_FastFpo;
// XXX drewb - Temporary until the fake user-shared
// module is worked out.
s_FastFpo.ulOffStart = (ULONG)Offset;
s_FastFpo.cbProcSize = X86_SHARED_SYSCALL_SIZE;
s_FastFpo.cdwLocals = 0;
s_FastFpo.cdwParams = 0;
s_FastFpo.cbProlog = 0;
s_FastFpo.cbRegs = 0;
s_FastFpo.fHasSEH = 0;
s_FastFpo.fUseBP = 0;
s_FastFpo.reserved = 0;
s_FastFpo.cbFrame = FRAME_FPO;
return &s_FastFpo;
}
PFPO_DATA
ModifyFpoRecord(ImageInfo* Image, PFPO_DATA FpoData)
{
if (FpoData->cdwLocals == 80)
{
static ULONG64 s_CommonDispatchException;
// Some versions of CommonDispatchException have
// the wrong locals size, which screws up stack
// traces. Detect and fix up these problems.
if (s_CommonDispatchException == 0)
{
GetOffsetFromSym(g_Process,
"nt!CommonDispatchException",
&s_CommonDispatchException,
NULL);
}
if (Image->m_BaseOfImage + FpoData->ulOffStart ==
s_CommonDispatchException)
{
static FPO_DATA s_CdeFpo;
s_CdeFpo = *FpoData;
s_CdeFpo.cdwLocals = 20;
FpoData = &s_CdeFpo;
}
}
else if (FpoData->cdwLocals == 0 && FpoData->cdwParams == 0 &&
FpoData->cbRegs == 3)
{
static ULONG64 s_KiSwapThread;
// KiSwapThread has shrink-wrapping so that three registers
// are pushed in only a portion of the code. Unfortunately,
// the most important place in the code -- the call to
// KiSwapContext -- is outside of this region and therefore
// the register count is wrong much more often than it's
// correct. Switch the register count to two to make it
// correct more often than wrong.
if (s_KiSwapThread == 0)
{
GetOffsetFromSym(g_Process,
"nt!KiSwapThread", &s_KiSwapThread, NULL);
}
if (Image->m_BaseOfImage + FpoData->ulOffStart ==
s_KiSwapThread)
{
static FPO_DATA s_KstFpo;
s_KstFpo = *FpoData;
s_KstFpo.cbRegs = 2;
FpoData = &s_KstFpo;
}
}
else if (FpoData->fHasSEH)
{
static FPO_DATA s_SehFpo;
s_SehFpo = *FpoData;
s_SehFpo.cbFrame = FRAME_NONFPO;
FpoData = &s_SehFpo;
}
return FpoData;
}
PFPO_DATA
FindFpoDataForModule(DWORD64 PCAddr)
/*++
Routine Description:
Locates the fpo data structure in the process's linked list for the
requested module.
Arguments:
PCAddr - address contained in the program counter
Return Value:
null - could not locate the entry
valid address - found the entry at the adress retured
--*/
{
ProcessInfo* Process;
ImageInfo* Image;
PFPO_DATA FpoData;
Process = g_Process;
Image = Process->m_ImageHead;
FpoData = 0;
while (Image)
{
if ((PCAddr >= Image->m_BaseOfImage) &&
(PCAddr < Image->m_BaseOfImage + Image->m_SizeOfImage))
{
FpoData = (PFPO_DATA)
SymFunctionTableAccess(g_Process->m_SymHandle, PCAddr);
if (!FpoData)
{
FpoData = SynthesizeFpoDataForModule(PCAddr);
}
else
{
FpoData = ModifyFpoRecord(Image, FpoData);
}
return FpoData;
}
Image = Image->m_Next;
}
ULONG64 FscBase;
switch(IsInFastSyscall(PCAddr, &FscBase))
{
case FSC_FOUND:
return SynthesizeFpoDataForFastSyscall(FscBase);
}
// the function is not part of any known loaded image
return NULL;
}
LPVOID
SwFunctionTableAccess(
HANDLE hProcess,
ULONG64 AddrBase
)
{
static IMAGE_IA64_RUNTIME_FUNCTION_ENTRY s_Ia64;
static _IMAGE_RUNTIME_FUNCTION_ENTRY s_Amd64;
PVOID pife;
if (g_Machine->m_ExecTypes[0] == IMAGE_FILE_MACHINE_I386)
{
return (LPVOID)FindFpoDataForModule( AddrBase );
}
pife = SymFunctionTableAccess64(hProcess, AddrBase);
switch(g_Machine->m_ExecTypes[0])
{
case IMAGE_FILE_MACHINE_IA64:
if (pife)
{
s_Ia64 = *(PIMAGE_IA64_RUNTIME_FUNCTION_ENTRY)pife;
return &s_Ia64;
}
else
{
if (IS_KERNEL_TARGET(g_Target) &&
(AddrBase >= IA64_MM_EPC_VA) &&
(AddrBase < (IA64_MM_EPC_VA + IA64_PAGE_SIZE)))
{
return g_EpcRfe;
}
else
{
return NULL;
}
}
break;
case IMAGE_FILE_MACHINE_AMD64:
if (pife)
{
s_Amd64 = *(_PIMAGE_RUNTIME_FUNCTION_ENTRY)pife;
return &s_Amd64;
}
break;
}
return NULL;
}
DWORD64
SwTranslateAddress(
HANDLE hProcess,
HANDLE hThread,
LPADDRESS64 lpaddress
)
{
//
// don't support 16bit stacks
//
return 0;
}
BOOL
SwReadMemory(
HANDLE hProcess,
ULONG64 BaseAddress,
LPVOID lpBuffer,
DWORD nSize,
LPDWORD lpNumberOfBytesRead
)
{
DBG_ASSERT(hProcess == OS_HANDLE(g_Process->m_SysHandle));
if (IS_KERNEL_TARGET(g_Target))
{
DWORD BytesRead;
HRESULT Status;
if ((LONG_PTR)lpNumberOfBytesRead == -1)
{
if (g_Target->m_MachineType == IMAGE_FILE_MACHINE_I386)
{
BaseAddress += g_Target->m_TypeInfo.SizeTargetContext;
}
Status = g_Target->ReadControl(CURRENT_PROC,
(ULONG)BaseAddress,
lpBuffer,
nSize,
&BytesRead);
return Status == S_OK;
}
}
if (g_Target->ReadVirtual(g_Process, BaseAddress, lpBuffer, nSize,
lpNumberOfBytesRead) != S_OK)
{
// Make sure bytes read is zero.
if (lpNumberOfBytesRead != NULL)
{
*lpNumberOfBytesRead = 0;
}
return FALSE;
}
else
{
return TRUE;
}
}
BOOL
SwReadMemory32(
HANDLE hProcess,
ULONG dwBaseAddress,
LPVOID lpBuffer,
DWORD nSize,
LPDWORD lpNumberOfBytesRead
)
{
return SwReadMemory(hProcess,
EXTEND64(dwBaseAddress),
lpBuffer,
nSize,
lpNumberOfBytesRead);
}
DWORD64
GetKernelModuleBase(
ULONG64 Address
)
{
ModuleInfo* ModIter;
if (!(ModIter = g_Target->GetModuleInfo(FALSE)) ||
ModIter->Initialize(g_Thread) != S_OK)
{
return 0;
}
// We only want module base and size.
ModIter->m_InfoLevel = MODULE_INFO_BASE_SIZE;
for (;;)
{
MODULE_INFO_ENTRY ModEntry;
ZeroMemory(&ModEntry, sizeof(ModEntry));
if (ModIter->GetEntry(&ModEntry) != S_OK)
{
break;
}
if (Address >= ModEntry.Base &&
Address < ModEntry.Base + ModEntry.Size)
{
return ModEntry.Base;
}
}
return 0;
}
DWORD64
SwGetModuleBase(
HANDLE hProcess,
ULONG64 Address
)
{
if (g_Machine->m_ExecTypes[0] == IMAGE_FILE_MACHINE_IA64 &&
IS_KERNEL_TARGET(g_Target) &&
(Address >= IA64_MM_EPC_VA) &&
(Address < (IA64_MM_EPC_VA + IA64_PAGE_SIZE)))
{
Address -= (IA64_MM_EPC_VA - g_Target->m_SystemCallVirtualAddress);
}
ImageInfo* Image = g_Process->FindImageByOffset(Address, TRUE);
if (Image)
{
return Image->m_BaseOfImage;
}
// This might be the JIT output for managed code.
// There's no 'base' as such but we need to return
// something non-zero to indicate this code is known.
// There won't be any FPO information so the module base
// value isn't that important. Just return the actual address.
if (g_Machine->m_ExecTypes[0] == IMAGE_FILE_MACHINE_I386 &&
g_Process->IsCorCode(Address) == S_OK)
{
return Address;
}
// If no regular module was found we need to look in
// the dynamic function tables to see if an entry
// there matches.
ULONG64 DynBase = g_Target->
GetDynamicFunctionTableBase(g_Process, Address);
if (DynBase)
{
return DynBase;
}
if (IS_KERNEL_TARGET(g_Target))
{
// If no modules have been loaded there's still a possibility
// of getting a kernel stack trace (without symbols) by going
// after the module base directly. This also makes it possible
// to get a stack trace when there are no symbols available.
if (g_Process->m_ImageHead == NULL)
{
return GetKernelModuleBase( Address );
}
}
return 0;
}
DWORD
SwGetModuleBase32(
HANDLE hProcess,
DWORD Address
)
{
return (DWORD)SwGetModuleBase(hProcess, Address);
}
void
PrintStackTraceHeaderLine(
ULONG Flags
)
{
if ( (Flags & DEBUG_STACK_COLUMN_NAMES) == 0 )
{
return;
}
StartOutLine(DEBUG_OUTPUT_NORMAL, OUT_LINE_NO_TIMESTAMP);
if (Flags & DEBUG_STACK_FRAME_NUMBERS)
{
dprintf(" # ");
}
if (Flags & DEBUG_STACK_FRAME_MEMORY_USAGE)
{
dprintf(" Memory ");
}
if (Flags & DEBUG_STACK_FRAME_ADDRESSES)
{
g_Machine->PrintStackFrameAddressesTitle(Flags);
}
if (Flags & DEBUG_STACK_ARGUMENTS)
{
g_Machine->PrintStackArgumentsTitle(Flags);
}
g_Machine->PrintStackCallSiteTitle(Flags);
dprintf("\n");
}
VOID
PrintStackFrame(
PDEBUG_STACK_FRAME StackFrame,
PDEBUG_STACK_FRAME PrevFrame,
ULONG Flags
)
{
ULONG64 Displacement;
ULONG64 InstructionOffset = StackFrame->InstructionOffset;
SYMBOL_INFO_AND_NAME SymInfo;
if (g_Machine->m_ExecTypes[0] == IMAGE_FILE_MACHINE_IA64 &&
IS_KERNEL_TARGET(g_Target) &&
(InstructionOffset >= IA64_MM_EPC_VA) &&
(InstructionOffset < (IA64_MM_EPC_VA + IA64_PAGE_SIZE)))
{
InstructionOffset = InstructionOffset -
(IA64_MM_EPC_VA - g_Target->m_SystemCallVirtualAddress);
}
GetSymbolInfo(InstructionOffset, NULL, 0,
SymInfo, &Displacement);
StartOutLine(DEBUG_OUTPUT_NORMAL, OUT_LINE_NO_TIMESTAMP);
if (Flags & DEBUG_STACK_FRAME_NUMBERS)
{
dprintf("%02lx ", StackFrame->FrameNumber);
}
if (Flags & DEBUG_STACK_FRAME_MEMORY_USAGE)
{
if (PrevFrame)
{
g_Machine->PrintStackFrameMemoryUsage(StackFrame, PrevFrame);
}
else
{
dprintf(" ");
}
}
if (Flags & DEBUG_STACK_FRAME_ADDRESSES)
{
g_Machine->PrintStackFrameAddresses(Flags, StackFrame);
}
if (Flags & DEBUG_STACK_ARGUMENTS)
{
g_Machine->PrintStackArguments(Flags, StackFrame);
}
g_Machine->PrintStackCallSite(Flags, StackFrame,
SymInfo, SymInfo->Name,
Displacement);
if (Flags & DEBUG_STACK_SOURCE_LINE)
{
OutputLineAddr(InstructionOffset, " [%s @ %d]");
}
dprintf( "\n" );
}
BOOL
CheckFrameValidity(PDEBUG_STACK_FRAME Frame)
{
//
// If the current frame's IP is not in any loaded module
// it's likely that we won't be able to unwind.
//
// If the current frame's IP is in a loaded module and
// that module does not have symbols it's very possible
// that the stack trace will be incorrect since the
// debugger has to guess about how to unwind the stack.
// Non-x86 architectures have unwind info in the images
// themselves so restrict this check to x86.
//
if (g_Machine->m_ExecTypes[0] == IMAGE_FILE_MACHINE_I386 &&
Frame->InstructionOffset != -1 &&
!Frame->FuncTableEntry &&
g_Process->IsCorCode(Frame->InstructionOffset) != S_OK)
{
IMAGEHLP_MODULE64 Mod;
Mod.SizeOfStruct = sizeof(Mod);
if (!SymGetModuleInfo64(g_Process->m_SymHandle,
Frame->InstructionOffset, &Mod))
{
WarnOut("WARNING: Frame IP not in any known module. "
"Following frames may be wrong.\n");
return TRUE;
}
else if (Mod.SymType == SymNone || Mod.SymType == SymExport ||
Mod.SymType == SymDeferred)
{
WarnOut("WARNING: Stack unwind information not available. "
"Following frames may be wrong.\n");
return TRUE;
}
}
return FALSE;
}
VOID
PrintStackTrace(
ULONG NumFrames,
PDEBUG_STACK_FRAME StackFrames,
ULONG Flags
)
{
ULONG i;
BOOL SymWarning = FALSE;
PrintStackTraceHeaderLine(Flags);
for (i = 0; i < NumFrames; i++)
{
if (!SymWarning && NumFrames > 1)
{
SymWarning = CheckFrameValidity(StackFrames + i);
}
PrintStackFrame(StackFrames + i,
i > 0 ? (StackFrames + (i - 1)) : NULL,
Flags);
}
}
HRESULT
UnwindCorFrame(ICorDataStackWalk* CorStack,
PCROSS_PLATFORM_CONTEXT Context,
PDEBUG_STACK_FRAME DbgFrame,
LPSTACKFRAME64 VirtFrame)
{
HRESULT Status;
ADDRESS64 PreInstr, PreStack, PreFrame;
ADDRESS64 PostInstr, PostStack, PostFrame;
CorDataFrameType CorFrameType;
g_Machine->
GetStackDefaultsFromContext(Context,
&PreInstr, &PreStack, &PreFrame);
if ((Status = CorStack->
SetFrameContext(g_Target->m_TypeInfo.SizeTargetContext,
(BYTE*)Context)) != S_OK ||
(Status = CorStack->
GetFrameDescription(&CorFrameType, NULL, 0)) != S_OK)
{
return Status;
}
if (CorFrameType != DAC_FRAME_COR_METHOD_FRAME)
{
return S_FALSE;
}
if ((Status = CorStack->UnwindFrame()) != S_OK ||
(Status = CorStack->
GetFrameContext(g_Target->m_TypeInfo.SizeTargetContext,
(BYTE*)Context)) != S_OK)
{
return Status;
}
g_Machine->
GetStackDefaultsFromContext(Context,
&PostInstr, &PostStack, &PostFrame);
if (g_DebugCorStack)
{
dprintf(" COR Pre i %08X s %08X f %08x\n"
" Post i %08X s %08X f %08x\n",
(ULONG)PreInstr.Offset,
(ULONG)PreStack.Offset,
(ULONG)PreFrame.Offset,
(ULONG)PostInstr.Offset,
(ULONG)PostStack.Offset,
(ULONG)PostFrame.Offset);
}
DbgFrame->InstructionOffset = PreInstr.Offset;
DbgFrame->ReturnOffset = PostInstr.Offset;
DbgFrame->FrameOffset = PreFrame.Offset;
DbgFrame->StackOffset = PreStack.Offset;
DbgFrame->FuncTableEntry = 0;
DbgFrame->Virtual = FALSE;
ZeroMemory(DbgFrame->Reserved,
sizeof(DbgFrame->Reserved));
ZeroMemory(DbgFrame->Params,
sizeof(DbgFrame->Params));
// Prepare the StackWalk64 frame in case it's
// used later. Setting Virtual to FALSE should
// force the stack walker to reinitialize.
VirtFrame->Virtual = FALSE;
VirtFrame->AddrPC = PostInstr;
VirtFrame->AddrFrame = PostFrame;
VirtFrame->AddrStack = PostStack;
return S_OK;
}
DWORD
StackTrace(
DebugClient* Client,
ULONG64 FramePointer,
ULONG64 StackPointer,
ULONG64 InstructionPointer,
ULONG PointerDefaults,
PDEBUG_STACK_FRAME StackFrames,
ULONG NumFrames,
ULONG64 ExtThread,
ULONG Flags,
BOOL EstablishingScope
)
{
STACKFRAME64 VirtualFrame;
DWORD i;
CROSS_PLATFORM_CONTEXT Context;
BOOL SymWarning = FALSE;
ULONG X86Ebp;
ULONG Seg;
ADDRESS64 DefInstr, DefStack, DefFrame;
HRESULT Status;
ICorDataStackWalk* CorStack;
if (!EstablishingScope)
{
RequireCurrentScope();
}
if (g_Machine->GetContextState(MCTX_FULL) != S_OK)
{
return 0;
}
Context = *GetScopeOrMachineContext();
g_Machine->GetStackDefaultsFromContext(&Context, &DefInstr,
&DefStack, &DefFrame);
//
// let's start clean
//
ZeroMemory( StackFrames, sizeof(StackFrames[0]) * NumFrames );
ZeroMemory( &VirtualFrame, sizeof(VirtualFrame) );
if (IS_KERNEL_TARGET(g_Target))
{
//
// if debugger was initialized at boot, usermode addresses needed for
// stack traces on IA64 were not available. Try it now:
//
if (g_Machine->m_ExecTypes[0] == IMAGE_FILE_MACHINE_IA64 &&
!g_Target->m_KdDebuggerData.KeUserCallbackDispatcher)
{
g_Target->QueryKernelInfo(g_Thread, FALSE);
}
ULONG64 ThreadData;
// If no explicit thread is given then we use the
// current thread. However, the current thread is only
// valid if the current thread is the event thread since
// tracing back into user mode requires that the appropriate
// user-mode memory state be active.
if (ExtThread != 0)
{
ThreadData = ExtThread;
}
else if (g_Thread != g_EventThread ||
g_Process != g_EventProcess ||
g_Process->
GetImplicitThreadData(g_Thread, &ThreadData) != S_OK)
{
ThreadData = 0;
}
VirtualFrame.KdHelp.Thread = ThreadData;
VirtualFrame.KdHelp.ThCallbackStack = ThreadData ?
g_Target->m_KdDebuggerData.ThCallbackStack : 0;
VirtualFrame.KdHelp.KiCallUserMode =
g_Target->m_KdDebuggerData.KiCallUserMode;
VirtualFrame.KdHelp.NextCallback =
g_Target->m_KdDebuggerData.NextCallback;
VirtualFrame.KdHelp.KeUserCallbackDispatcher =
g_Target->m_KdDebuggerData.KeUserCallbackDispatcher;
VirtualFrame.KdHelp.FramePointer =
g_Target->m_KdDebuggerData.FramePointer;
VirtualFrame.KdHelp.SystemRangeStart = g_Target->m_SystemRangeStart;
}
//
// setup the program counter
//
if (PointerDefaults & STACK_INSTR_DEFAULT)
{
VirtualFrame.AddrPC = DefInstr;
}
else
{
VirtualFrame.AddrPC.Mode = AddrModeFlat;
Seg = g_Machine->GetSegRegNum(SEGREG_CODE);
VirtualFrame.AddrPC.Segment =
Seg ? (WORD)g_Machine->FullGetVal32(Seg) : 0;
VirtualFrame.AddrPC.Offset = InstructionPointer;
}
//
// setup the frame pointer
//
if (PointerDefaults & STACK_FRAME_DEFAULT)
{
VirtualFrame.AddrFrame = DefFrame;
}
else
{
VirtualFrame.AddrFrame.Mode = AddrModeFlat;
Seg = g_Machine->GetSegRegNum(SEGREG_STACK);
VirtualFrame.AddrFrame.Segment =
Seg ? (WORD)g_Machine->FullGetVal32(Seg) : 0;
VirtualFrame.AddrFrame.Offset = FramePointer;
}
VirtualFrame.AddrBStore = VirtualFrame.AddrFrame;
X86Ebp = (ULONG)VirtualFrame.AddrFrame.Offset;
//
// setup the stack pointer
//
if (PointerDefaults & STACK_STACK_DEFAULT)
{
VirtualFrame.AddrStack = DefStack;
}
else
{
VirtualFrame.AddrStack.Mode = AddrModeFlat;
Seg = g_Machine->GetSegRegNum(SEGREG_STACK);
VirtualFrame.AddrStack.Segment =
Seg ? (WORD)g_Machine->FullGetVal32(Seg) : 0;
VirtualFrame.AddrStack.Offset = StackPointer;
}
if (g_Machine->m_ExecTypes[0] == IMAGE_FILE_MACHINE_IA64 &&
IS_KERNEL_TARGET(g_Target) &&
g_Target->m_SystemCallVirtualAddress)
{
PVOID FunctionEntry;
FunctionEntry = SwFunctionTableAccess
(g_Process->m_SymHandle, g_Target->m_SystemCallVirtualAddress);
if (FunctionEntry != NULL)
{
RtlCopyMemory(&g_EpcRfeBuffer, FunctionEntry,
sizeof(IMAGE_IA64_RUNTIME_FUNCTION_ENTRY));
g_EpcRfe = &g_EpcRfeBuffer;
}
else
{
g_EpcRfe = NULL;
}
}
if (g_AllowCorStack &&
g_Machine->m_ExecTypes[0] == IMAGE_FILE_MACHINE_I386)
{
// XXX drewb - No good way to know what runtime thread
// to use, so always use the current one.
CorStack = g_Process->StartCorStack(0);
}
else
{
// Non-x86 stacks can always be walked by StackWalk64
// because of high-quality unwind information.
CorStack = NULL;
}
for (i = 0; i < NumFrames; i++)
{
Status = S_FALSE;
if (g_DebugCorStack)
{
dprintf("Frame %d\n", i);
}
StackFrames[i].FrameNumber = i;
if (i > 0 && CorStack)
{
if (FAILED(Status = UnwindCorFrame(CorStack,
&Context,
&StackFrames[i],
&VirtualFrame)))
{
ErrOut("Managed frame unwind failed, %s\n",
FormatStatusCode(Status));
break;
}
}
if (Status == S_FALSE)
{
// SwReadMemory doesn't currently use the thread handle
// but send in something reasonable in case of future changes.
if (!StackWalk64(g_Machine->m_ExecTypes[0],
OS_HANDLE(g_Process->m_SysHandle),
OS_HANDLE(g_Thread->m_Handle),
&VirtualFrame,
&Context,
SwReadMemory,
SwFunctionTableAccess,
SwGetModuleBase,
SwTranslateAddress))
{
break;
}
StackFrames[i].InstructionOffset = VirtualFrame.AddrPC.Offset;
StackFrames[i].ReturnOffset = VirtualFrame.AddrReturn.Offset;
StackFrames[i].FrameOffset = VirtualFrame.AddrFrame.Offset;
StackFrames[i].StackOffset = VirtualFrame.AddrStack.Offset;
StackFrames[i].FuncTableEntry =
(ULONG64)VirtualFrame.FuncTableEntry;
StackFrames[i].Virtual = VirtualFrame.Virtual;
// NOTE - we have more reserved space in the DEBUG_STACK_FRAME
memcpy(StackFrames[i].Reserved, VirtualFrame.Reserved,
sizeof(VirtualFrame.Reserved));
memcpy(StackFrames[i].Params, VirtualFrame.Params,
sizeof(VirtualFrame.Params));
if (g_Machine->m_ExecTypes[0] == IMAGE_FILE_MACHINE_IA64 &&
IS_KERNEL_TARGET(g_Target))
{
if ((VirtualFrame.AddrPC.Offset >= IA64_MM_EPC_VA) &&
(VirtualFrame.AddrPC.Offset <
(IA64_MM_EPC_VA + IA64_PAGE_SIZE)))
{
VirtualFrame.AddrPC.Offset -=
(IA64_MM_EPC_VA -
g_Target->m_SystemCallVirtualAddress);
}
if ((i != 0) &&
(StackFrames[i - 1].InstructionOffset >= IA64_MM_EPC_VA) &&
(VirtualFrame.AddrPC.Offset <
(IA64_MM_EPC_VA + IA64_PAGE_SIZE)))
{
StackFrames[i - 1].ReturnOffset =
VirtualFrame.AddrPC.Offset;
}
}
else if (g_Machine->m_ExecTypes[0] == IMAGE_FILE_MACHINE_I386)
{
if (StackFrames[i].FuncTableEntry)
{
PFPO_DATA FpoData = (PFPO_DATA)
StackFrames[i].FuncTableEntry;
if ((FpoData->cbFrame == FRAME_FPO) && X86Ebp &&
(!FpoData->fUseBP &&
((SAVE_EBP(&StackFrames[i]) >> 32) != 0xEB)))
{
// EBP tag, so stack walker doesn't get confused
SAVE_EBP(&StackFrames[i]) = X86Ebp + 0xEB00000000;
}
}
X86Ebp = Context.X86Context.Ebp;
}
}
if (Flags)
{
if (i == 0)
{
PrintStackTraceHeaderLine(Flags);
}
if (!SymWarning && NumFrames > 1)
{
SymWarning = CheckFrameValidity(StackFrames + i);
}
PrintStackFrame(StackFrames + i,
i > 0 ? (StackFrames + (i - 1)) : NULL,
Flags);
if (Flags & DEBUG_STACK_NONVOLATILE_REGISTERS)
{
g_Machine->PrintStackNonvolatileRegisters(Flags,
StackFrames + i,
&Context, i);
}
}
}
RELEASE(CorStack);
return i;
}
VOID
DoStackTrace(
DebugClient* Client,
ULONG64 FramePointer,
ULONG64 StackPointer,
ULONG64 InstructionPointer,
ULONG PointerDefaults,
ULONG NumFrames,
ULONG TraceFlags
)
{
PDEBUG_STACK_FRAME StackFrames;
ULONG NumFramesToRead;
DWORD FrameCount;
if (NumFrames == 0)
{
NumFrames = g_DefaultStackTraceDepth;
}
if (TraceFlags & RAW_STACK_DUMP)
{
DBG_ASSERT(TraceFlags == RAW_STACK_DUMP);
NumFramesToRead = 1;
}
else
{
NumFramesToRead = NumFrames;
}
StackFrames = (PDEBUG_STACK_FRAME)
malloc( sizeof(StackFrames[0]) * NumFramesToRead );
if (!StackFrames)
{
ErrOut( "could not allocate memory for stack trace\n" );
return;
}
if (g_Machine->m_Ptr64 &&
(TraceFlags & DEBUG_STACK_ARGUMENTS) &&
!(TraceFlags & DEBUG_STACK_FUNCTION_INFO))
{
TraceFlags |= DEBUG_STACK_FRAME_ADDRESSES_RA_ONLY;
}
FrameCount = StackTrace(Client,
FramePointer,
StackPointer,
InstructionPointer,
PointerDefaults,
StackFrames,
NumFramesToRead,
0,
TraceFlags,
FALSE);
if (FrameCount == 0)
{
ErrOut( "could not fetch any stack frames\n" );
free(StackFrames);
return;
}
if (TraceFlags & RAW_STACK_DUMP)
{
// Starting with the frame pointer, dump NumFrames DWORD's
// and the symbol if possible.
ADDR StartAddr;
ADDRFLAT(&StartAddr, StackFrames[0].FrameOffset);
DumpDwordMemory(&StartAddr, NumFrames, TRUE);
}
free(StackFrames);
}