2766 lines
83 KiB
C
2766 lines
83 KiB
C
/**
|
|
*** Copyright (C) 1996-1999 Intel Corporation. All rights reserved.
|
|
***
|
|
*** The information and source code contained herein is the exclusive
|
|
*** property of Intel Corporation and may not be disclosed, examined
|
|
*** or reproduced in whole or in part without explicit written authorization
|
|
*** from the company.
|
|
***
|
|
****************************************************************************
|
|
***
|
|
*** Thierry 09/21/99 - v16
|
|
*** WARNING: ntos\rtl\ia64\vunwind.c and sdktools\imagehlp\vwndia64.c are
|
|
*** identical. For sake of maintenance and for debug purposes,
|
|
*** please keep them as this. Thank you.
|
|
***
|
|
****************************************************************************
|
|
**/
|
|
|
|
#define _CROSS_PLATFORM_
|
|
#define _IA64REG_
|
|
#include <nt.h>
|
|
#include <ntrtl.h>
|
|
#include <nturtl.h>
|
|
#include "private.h"
|
|
#include "globals.h"
|
|
#include "ia64inst.h"
|
|
#define NOEXTAPI
|
|
#include "wdbgexts.h"
|
|
#include "ntdbg.h"
|
|
#include <stdlib.h>
|
|
|
|
#ifdef _IMAGEHLP_SOURCE_
|
|
#include <symbols.h>
|
|
|
|
#define NOT_IMAGEHLP(E)
|
|
#define FUNCTION_ENTRY_IS_IMAGE_STYLE
|
|
#define RtlVirtualUnwind VirtualUnwindIa64
|
|
#define VUW_DEBUG_PRINT OutputDebugString
|
|
|
|
#else // !_IMAGEHLP_SOURCE_
|
|
|
|
#define NOT_IMAGEHLP(E) E
|
|
#define VUW_DEBUG_PRINT DbgPrint
|
|
|
|
#endif // !_IMAGEHLP_SOURCE_
|
|
|
|
#ifdef MASK
|
|
#undef MASK
|
|
#endif // MASK
|
|
#define MASK(bp,value) (value << bp)
|
|
|
|
//
|
|
// ABI values
|
|
//
|
|
|
|
#define SVR4_ABI 0
|
|
#define HPUX_ABI 1
|
|
#define NT_ABI 2
|
|
|
|
|
|
#ifdef KERNEL_DEBUGGER
|
|
#define FUNCTION_ENTRY_IS_IMAGE_STYLE
|
|
#define RtlVirtualUnwind VirtualUnwind
|
|
#endif
|
|
|
|
#define STATE_RECORD_STACK_SIZE 32
|
|
|
|
#define SPILLSIZE_OF_FLOAT128_IN_DWORDS 4
|
|
#define SPILLSIZE_OF_ULONGLONG_IN_DWORDS 2
|
|
|
|
#define REGISTER_SIZE sizeof(ULONGLONG)
|
|
#define STATIC_REGISTER_SET_SIZE 32
|
|
#define SLOTS_PER_BUNDLE 3
|
|
|
|
#define R1_MASK 0xC0
|
|
#define R1_PREFIX 0x0
|
|
#define R1_REGION_TYPE_MASK 0x20
|
|
#define R1_LENGTH_MASK 0x1F
|
|
|
|
#define R2_MASK 0xE0
|
|
#define R2_PREFIX 0x40
|
|
|
|
#define R3_MASK 0xE0
|
|
#define R3_PREFIX 0x60
|
|
#define R3_REGION_TYPE_MASK 0x3
|
|
|
|
#define P1_MASK 0xE0
|
|
#define P1_PREFIX 0x80
|
|
#define P2_MASK 0xF0
|
|
#define P2_PREFIX 0xA0
|
|
#define P3_MASK 0xF8
|
|
#define P3_PREFIX 0xB0
|
|
#define P4_MASK 0xFF
|
|
#define P4_PREFIX 0xB8
|
|
#define P5_MASK 0xFF
|
|
#define P5_PREFIX 0xB9
|
|
#define P6_MASK 0xE0
|
|
#define P6_PREFIX 0xC0
|
|
#define P7_MASK 0xF0
|
|
#define P7_PREFIX 0xE0
|
|
#define P8_MASK 0xFF
|
|
#define P8_PREFIX 0xF0
|
|
#define P9_MASK 0xFF
|
|
#define P9_PREFIX 0xF1
|
|
#define P10_MASK 0xFF
|
|
#define P10_PREFIX 0xFF
|
|
|
|
#define B1_MASK 0xC0
|
|
#define B1_PREFIX 0x80
|
|
#define B1_TYPE_MASK 0x20
|
|
#define B1_LABEL_MASK 0x1F
|
|
#define B2_MASK 0xE0
|
|
#define B2_PREFIX 0xC0
|
|
#define B2_ECOUNT_MASK 0x1F
|
|
#define B3_MASK 0xF0
|
|
#define B3_PREFIX 0xE0
|
|
#define B4_MASK 0xF0
|
|
#define B4_PREFIX 0xF0
|
|
#define B4_TYPE_MASK 0x08
|
|
|
|
//
|
|
// P3 descriptor type
|
|
//
|
|
|
|
#define PSP_GR 0
|
|
#define RP_GR 1
|
|
#define PFS_GR 2
|
|
#define PREDS_GR 3
|
|
#define UNAT_GR 4
|
|
#define LC_GR 5
|
|
#define RP_BR 6
|
|
#define RNAT_GR 7
|
|
#define BSP_GR 8
|
|
#define BSPSTORE_GR 9
|
|
#define FPSR_GR 10
|
|
#define PRIUNAT_GR 11
|
|
|
|
//
|
|
// P7 descriptor type
|
|
//
|
|
|
|
#define MEM_STACK_F 0
|
|
#define MEM_STACK_V 1
|
|
#define SPILL_BASE 2
|
|
#define PSP_SPREL 3
|
|
#define RP_WHEN 4
|
|
#define RP_PSPREL 5
|
|
#define PFS_WHEN 6
|
|
#define PFS_PSPREL 7
|
|
#define PREDS_WHEN 8
|
|
#define PREDS_PSPREL 9
|
|
#define LC_WHEN 10
|
|
#define LC_PSPREL 11
|
|
#define UNAT_WHEN 12
|
|
#define UNAT_PSPREL 13
|
|
#define FPSR_WHEN 14
|
|
#define FPSR_PSPREL 15
|
|
|
|
//
|
|
// P8 descriptor type
|
|
//
|
|
|
|
#define PSP_PSPREL 0
|
|
#define RP_SPREL 1
|
|
#define PFS_SPREL 2
|
|
#define PREDS_SPREL 3
|
|
#define LC_SPREL 4
|
|
#define UNAT_SPREL 5
|
|
#define FPSR_SPREL 6
|
|
#define BSP_WHEN 7
|
|
#define BSP_PSPREL 8
|
|
#define BSP_SPREL 9
|
|
#define BSPSTORE_WHEN 10
|
|
#define BSPSTORE_PSPREL 11
|
|
#define BSPSTORE_SPREL 12
|
|
#define RNAT_WHEN 13
|
|
#define RNAT_PSPREL 14
|
|
#define RNAT_SPREL 15
|
|
#define PRIUNAT_WHEN 16
|
|
#define PRIUNAT_PSPREL 17
|
|
#define PRIUNAT_SPREL 18
|
|
|
|
|
|
#define STACK_POINTER_GR 12
|
|
|
|
#define FIRST_PRESERVED_GR 4
|
|
#define LAST_PRESERVED_GR 7
|
|
#define NUMBER_OF_PRESERVED_GR 4
|
|
#define NUMBER_OF_SCRATCH_GR 24
|
|
|
|
#define FIRST_LOW_PRESERVED_FR 2
|
|
#define LAST_LOW_PRESERVED_FR 5
|
|
#define NUMBER_OF_LOW_PRESERVED_FR 4
|
|
|
|
#define FIRST_HIGH_PRESERVED_FR 16
|
|
#define LAST_HIGH_PRESERVED_FR 31
|
|
#define NUMBER_OF_HIGH_PRESERVED_FR 16
|
|
#define NUMBER_OF_PRESERVED_FR (NUMBER_OF_LOW_PRESERVED_FR+NUMBER_OF_HIGH_PRESERVED_FR)
|
|
|
|
#define FIRST_PRESERVED_BR 1
|
|
#define LAST_PRESERVED_BR 5
|
|
#define NUMBER_OF_PRESERVED_BR 5
|
|
|
|
#define NUMBER_OF_PRESERVED_MISC 8
|
|
|
|
#define NUMBER_OF_PRESERVED_REGISTERS (NUMBER_OF_PRESERVED_MISC+NUMBER_OF_PRESERVED_BR)
|
|
|
|
|
|
#define REG_MISC_BASE 0
|
|
#define REG_PREDS (REG_MISC_BASE+0)
|
|
#define REG_SP (REG_MISC_BASE+1)
|
|
#define REG_PFS (REG_MISC_BASE+2)
|
|
#define REG_RP (REG_MISC_BASE+3)
|
|
#define REG_UNAT (REG_MISC_BASE+4)
|
|
#define REG_LC (REG_MISC_BASE+5)
|
|
#define REG_NATS (REG_MISC_BASE+6)
|
|
#define REG_FPSR (REG_MISC_BASE+7)
|
|
|
|
#define REG_BR_BASE (REG_MISC_BASE+NUMBER_OF_PRESERVED_MISC)
|
|
|
|
#define REG_BSP 0xff // REG_MISC_BASE+8
|
|
#define REG_BSPSTORE 0xff // REG_MISC_BASE+9
|
|
#define REG_RNAT 0xff // REG_MISC_BASE+10
|
|
|
|
//
|
|
// Where is a preserved register saved?
|
|
//
|
|
// 1. stack general register
|
|
// 2. memory stack (pspoff)
|
|
// 3. memory stack (spoff)
|
|
// 4. branch register
|
|
//
|
|
|
|
#define GENERAL_REG 0
|
|
#define PSP_RELATIVE 1
|
|
#define SP_RELATIVE 2
|
|
#define BRANCH_REG 3
|
|
|
|
|
|
#define ADD_STATE_RECORD(States, RegionLength, DescBeginIndex) \
|
|
States.Top++; \
|
|
States.Top->IsTarget = FALSE; \
|
|
States.Top->MiscMask = 0; \
|
|
States.Top->FrMask = 0; \
|
|
States.Top->GrMask = 0; \
|
|
States.Top->Label = (LABEL)0; \
|
|
States.Top->Ecount = 0; \
|
|
States.Top->RegionLen = RegionLength; \
|
|
States.Top->RegionBegin = UnwindContext.SlotCount; \
|
|
States.Top->SpWhen = 0; \
|
|
States.Top->SpAdjustment = 0; \
|
|
States.Top->SpillBase = (States.Top-1)->SpillPtr; \
|
|
States.Top->SpillPtr = (States.Top-1)->SpillPtr; \
|
|
States.Top->Previous = States.Current; \
|
|
States.Top->DescBegin = DescBeginIndex; \
|
|
States.Current = States.Top
|
|
|
|
|
|
#define VALID_LABEL_BIT_POSITION 15
|
|
|
|
#define LABEL_REGION(Region, Label) \
|
|
Region->Label = Label; \
|
|
Region->MiscMask |= (1 << VALID_LABEL_BIT_POSITION)
|
|
|
|
#define IS_REGION_LABELED(Region) \
|
|
(Region->MiscMask & (1 << VALID_LABEL_BIT_POSITION))
|
|
|
|
#define CHECK_LABEL(State, Label) \
|
|
( (IS_REGION_LABELED(State)) && (Label == State->Label) )
|
|
|
|
|
|
#define EXTRACT_NAT_FROM_UNAT(NatBit) \
|
|
NatBit = (UCHAR)((IntNats >> (((ULONG_PTR)Source & 0x1F8) >> 3)) & 0x1);
|
|
|
|
|
|
#define UW_DEBUG(x) SdbOut x
|
|
|
|
typedef struct _REGISTER_RECORD {
|
|
ULONG Where : 2; // 2-bit field
|
|
ULONG SaveOffset : 30; // 30 bits for offset, big enough?
|
|
ULONG When; // slot offset relative to region
|
|
} REGISTER_RECORD, *PREGISTER_RECORD;
|
|
|
|
typedef ULONG LABEL;
|
|
|
|
typedef struct _STATE_RECORD {
|
|
struct _STATE_RECORD *Previous; // pointer to outer nested prologue
|
|
BOOLEAN IsTarget; // TRUE if the control pc is in this prologue
|
|
UCHAR GrMask; // Mask that specifies which GRs to be restored
|
|
USHORT MiscMask; // Mask that specifies which BRs and misc. registers
|
|
// are to be restored.
|
|
// N.B. MSBit indicates Label is valid or not.
|
|
ULONG FrMask; // Mask that specifies which FRs to be restored
|
|
ULONG SpAdjustment; // size of stack frame allocated in the prologue
|
|
ULONG SpWhen; // slot offset relative to region
|
|
ULONG SpillPtr; // current spill location
|
|
ULONG SpillBase; // spill base of the region
|
|
ULONG RegionBegin; // first slot of region relative to function entry
|
|
ULONG RegionLen; // number of slots in the region
|
|
LABEL Label; // label that identifies a post-prologue state
|
|
ULONG Ecount; // number of prologue regions to pop
|
|
ULONG DescBegin; // first prologue descriptor for the region
|
|
ULONG DescEnd; // last prologue descriptor for the region
|
|
} STATE_RECORD, *PSTATE_RECORD;
|
|
|
|
typedef struct _UNWIND_CONTEXT {
|
|
REGISTER_RECORD MiscRegs[NUMBER_OF_PRESERVED_REGISTERS];
|
|
REGISTER_RECORD Float[NUMBER_OF_PRESERVED_FR];
|
|
REGISTER_RECORD Integer[NUMBER_OF_PRESERVED_GR];
|
|
BOOLEAN ActiveRegionFound;
|
|
UCHAR AlternateRp;
|
|
USHORT Version;
|
|
PUCHAR Descriptors; // beginning of descriptor data
|
|
ULONG Size; // total size of all descriptors
|
|
ULONG DescCount; // number of descriptor bytes processed
|
|
ULONG TargetSlot;
|
|
ULONG SlotCount;
|
|
} UNWIND_CONTEXT, *PUNWIND_CONTEXT;
|
|
|
|
typedef struct _STATE_RECORD_STACK {
|
|
ULONG Size;
|
|
PSTATE_RECORD Current;
|
|
PSTATE_RECORD Top;
|
|
PSTATE_RECORD Base;
|
|
} STATE_RECORD_STACK, *PSTATE_RECORD_STACK;
|
|
|
|
#define OFFSET(type, field) ((ULONG_PTR)(&((type *)0)->field))
|
|
|
|
static const USHORT MiscContextOffset[NUMBER_OF_PRESERVED_REGISTERS] = {
|
|
OFFSET(IA64_CONTEXT, Preds),
|
|
OFFSET(IA64_CONTEXT, IntSp),
|
|
OFFSET(IA64_CONTEXT, RsPFS),
|
|
OFFSET(IA64_CONTEXT, BrRp),
|
|
OFFSET(IA64_CONTEXT, ApUNAT),
|
|
OFFSET(IA64_CONTEXT, ApLC),
|
|
0,
|
|
0,
|
|
OFFSET(IA64_CONTEXT, BrS0),
|
|
OFFSET(IA64_CONTEXT, BrS1),
|
|
OFFSET(IA64_CONTEXT, BrS2),
|
|
OFFSET(IA64_CONTEXT, BrS3),
|
|
OFFSET(IA64_CONTEXT, BrS4)
|
|
};
|
|
|
|
static const USHORT MiscContextPointersOffset[NUMBER_OF_PRESERVED_REGISTERS] = {
|
|
OFFSET(IA64_KNONVOLATILE_CONTEXT_POINTERS, Preds),
|
|
OFFSET(IA64_KNONVOLATILE_CONTEXT_POINTERS, IntSp),
|
|
OFFSET(IA64_KNONVOLATILE_CONTEXT_POINTERS, RsPFS),
|
|
OFFSET(IA64_KNONVOLATILE_CONTEXT_POINTERS, BrRp),
|
|
OFFSET(IA64_KNONVOLATILE_CONTEXT_POINTERS, ApUNAT),
|
|
OFFSET(IA64_KNONVOLATILE_CONTEXT_POINTERS, ApLC),
|
|
0,
|
|
0,
|
|
OFFSET(IA64_KNONVOLATILE_CONTEXT_POINTERS, BrS0),
|
|
OFFSET(IA64_KNONVOLATILE_CONTEXT_POINTERS, BrS1),
|
|
OFFSET(IA64_KNONVOLATILE_CONTEXT_POINTERS, BrS2),
|
|
OFFSET(IA64_KNONVOLATILE_CONTEXT_POINTERS, BrS3),
|
|
OFFSET(IA64_KNONVOLATILE_CONTEXT_POINTERS, BrS4)
|
|
};
|
|
|
|
static const PSTR RegOffsetNames[NUMBER_OF_PRESERVED_REGISTERS] = {
|
|
"Preds",
|
|
"IntSp",
|
|
"RsPFS",
|
|
"BrRp",
|
|
"ApUNAT",
|
|
"ApLC",
|
|
"<invalid>",
|
|
"<invalid>",
|
|
"BrS0",
|
|
"BrS1",
|
|
"BrS2",
|
|
"BrS3",
|
|
"BrS4",
|
|
};
|
|
|
|
static const UCHAR P3RecordTypeToRegisterIndex[] =
|
|
{REG_SP, REG_RP, REG_PFS, REG_PREDS, REG_UNAT, REG_LC, REG_RP,
|
|
REG_RNAT, REG_BSP, REG_BSPSTORE, REG_FPSR};
|
|
|
|
static const UCHAR P7RecordTypeToRegisterIndex[] =
|
|
{0, REG_SP, 0, REG_SP, REG_RP, REG_RP, REG_PFS, REG_PFS, REG_PREDS,
|
|
REG_PREDS, REG_LC, REG_LC, REG_UNAT, REG_UNAT, REG_FPSR, REG_FPSR};
|
|
|
|
static const UCHAR P8RecordTypeToRegisterIndex[] =
|
|
{REG_SP, REG_RP, REG_PFS, REG_PREDS, REG_LC, REG_UNAT, REG_FPSR,
|
|
REG_BSP, REG_BSP, REG_BSP, REG_BSPSTORE, REG_BSPSTORE, REG_BSPSTORE,
|
|
REG_RNAT, REG_RNAT, REG_RNAT, REG_NATS, REG_NATS, REG_NATS};
|
|
|
|
|
|
/*++
|
|
|
|
Description:
|
|
|
|
The next code defines mechanism to handle the context of
|
|
the unwind process
|
|
|
|
--*/
|
|
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Initializes the FixupTable
|
|
|
|
Arguments:
|
|
|
|
iContext - index of the context in UnwindContext
|
|
|
|
--*/
|
|
|
|
void
|
|
Vwndia64InitFixupTable(UINT iContext)
|
|
{
|
|
if (iContext < VWNDIA64_UNWIND_CONTEXT_TABLE_SIZE)
|
|
{
|
|
PVWNDIA64_UNWIND_CONTEXT pContext = tlsvar(UnwindContext) + iContext;
|
|
ZeroMemory(pContext->FixupTable, sizeof(pContext->FixupTable));
|
|
}
|
|
}
|
|
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Adds new fixup region into the table
|
|
|
|
Arguments:
|
|
|
|
iContext - index of the context in UnwindContext
|
|
Begin, End, Fixup - fixup region data
|
|
|
|
--*/
|
|
|
|
void
|
|
Vwndia64AddFixupRegion(UINT iContext,
|
|
ULONGLONG Ip,
|
|
ULONGLONG Begin, ULONGLONG End, ULONGLONG Fixup)
|
|
{
|
|
if (iContext < VWNDIA64_UNWIND_CONTEXT_TABLE_SIZE)
|
|
{
|
|
PVWNDIA64_UNWIND_CONTEXT pContext = tlsvar(UnwindContext) + iContext;
|
|
|
|
PVWNDIA64_FUXUP_REGION FixupReg = pContext->FixupTable;
|
|
PVWNDIA64_FUXUP_REGION FixupRegLast = FixupReg +
|
|
VWNDIA64_FIXUP_TABLE_SIZE;
|
|
|
|
//
|
|
// Search for the empty region
|
|
//
|
|
while (
|
|
(FixupReg < FixupRegLast) &&
|
|
(FixupReg->Fixup &&
|
|
((FixupReg->Ip != Ip) ||
|
|
(FixupReg->Begin != Begin) ||
|
|
(FixupReg->End != End) ||
|
|
(FixupReg->Fixup != Fixup))))
|
|
{
|
|
++FixupReg;
|
|
}
|
|
|
|
if (FixupReg >= FixupRegLast) // not found (rare)
|
|
{
|
|
//
|
|
// Scroll the table (destructive)
|
|
//
|
|
MoveMemory(pContext->FixupTable,
|
|
pContext->FixupTable + 1,
|
|
sizeof(pContext->FixupTable) -
|
|
sizeof(VWNDIA64_FUXUP_REGION));
|
|
FixupReg = FixupRegLast - 1;
|
|
|
|
UW_DEBUG((2, "AddFixup: Table overflow\n"));
|
|
}
|
|
|
|
FixupReg->Ip = Ip;
|
|
FixupReg->Begin = Begin;
|
|
FixupReg->End = End;
|
|
FixupReg->Fixup = Fixup;
|
|
|
|
UW_DEBUG((2, "AddFixup: IP %016I64x, "
|
|
"%016I64x - %016I64x -> %016I64x\n",
|
|
Ip, Begin, End, Fixup));
|
|
}
|
|
else
|
|
{
|
|
UW_DEBUG((2, "AddFixup: Bad context\n"));
|
|
}
|
|
}
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Fixes the address using the specified fixup table
|
|
|
|
Arguments:
|
|
|
|
iContext - index of the context in UnwindContext
|
|
Size - size of the buffer in bytes
|
|
Addr - point to the address that have to be fixed
|
|
|
|
Return Value:
|
|
|
|
Addr contains the fixed value
|
|
|
|
--*/
|
|
|
|
void
|
|
Vwndia64FixAddress(UINT iContext, IN OUT ULONGLONG* Addr)
|
|
{
|
|
if (iContext < VWNDIA64_UNWIND_CONTEXT_TABLE_SIZE)
|
|
{
|
|
PVWNDIA64_UNWIND_CONTEXT pContext = tlsvar(UnwindContext) + iContext;
|
|
PVWNDIA64_FUXUP_REGION FixupReg = NULL;
|
|
|
|
UW_DEBUG((2, "FixAddr: Look for %016I64x\n", *Addr));
|
|
|
|
//
|
|
// find last containing region
|
|
//
|
|
{
|
|
PVWNDIA64_FUXUP_REGION FixupSearch = pContext->FixupTable;
|
|
PVWNDIA64_FUXUP_REGION FixupRegLast = FixupSearch +
|
|
VWNDIA64_FIXUP_TABLE_SIZE;
|
|
|
|
while ((FixupSearch < FixupRegLast) && FixupSearch->Fixup)
|
|
{
|
|
if ((FixupSearch->Begin <= *Addr) && (*Addr < FixupSearch->End))
|
|
{
|
|
FixupReg = FixupSearch;
|
|
}
|
|
++FixupSearch;
|
|
}
|
|
}
|
|
|
|
if (FixupReg)
|
|
{
|
|
*Addr = FixupReg->Fixup - (FixupReg->End - *Addr);
|
|
UW_DEBUG((2, " Fix to %016I64x\n", *Addr));
|
|
}
|
|
}
|
|
else
|
|
{
|
|
UW_DEBUG((2, "FixAddr: Bad context\n"));
|
|
}
|
|
}
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Checks whether specified Ip exists in fixup table
|
|
|
|
Arguments:
|
|
|
|
iContext - index of the context in UnwindContext
|
|
Ip - to search
|
|
|
|
Return Value:
|
|
|
|
Addr contains the fixed value
|
|
|
|
--*/
|
|
|
|
BOOL
|
|
Vwndia64IsFixupIp(UINT iContext, ULONGLONG Ip)
|
|
{
|
|
if (iContext < VWNDIA64_UNWIND_CONTEXT_TABLE_SIZE)
|
|
{
|
|
PVWNDIA64_UNWIND_CONTEXT pContext = tlsvar(UnwindContext) + iContext;
|
|
|
|
PVWNDIA64_FUXUP_REGION FixupSearch = pContext->FixupTable;
|
|
PVWNDIA64_FUXUP_REGION FixupRegLast = FixupSearch +
|
|
VWNDIA64_FIXUP_TABLE_SIZE;
|
|
|
|
while ((FixupSearch < FixupRegLast) &&
|
|
FixupSearch->Ip && (FixupSearch->Ip != Ip))
|
|
{
|
|
++FixupSearch;
|
|
}
|
|
|
|
return (FixupSearch->Ip == Ip);
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Initializes the new context
|
|
|
|
Return Value:
|
|
|
|
returns ID of the record (index in UnwindContext)
|
|
|
|
--*/
|
|
|
|
UINT
|
|
Vwndia64NewContext()
|
|
{
|
|
UINT* iContext = &tlsvar(UnwindContextNew);
|
|
|
|
if (*iContext >= VWNDIA64_UNWIND_CONTEXT_TABLE_SIZE)
|
|
{
|
|
*iContext = 0;
|
|
}
|
|
|
|
Vwndia64InitFixupTable(*iContext);
|
|
tlsvar(UnwindContext[*iContext]).bFailureReported = FALSE;
|
|
|
|
return (*iContext)++;
|
|
}
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Checks if the specified context index valid and
|
|
creates new context if otherwise
|
|
|
|
--*/
|
|
|
|
BOOL
|
|
Vwndia64ValidateContext(UINT* iContext)
|
|
{
|
|
if (!iContext) return FALSE;
|
|
if (*iContext >= VWNDIA64_UNWIND_CONTEXT_TABLE_SIZE)
|
|
{
|
|
*iContext = Vwndia64NewContext();
|
|
return FALSE;
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Check whether any failure have been reported
|
|
|
|
Arguments:
|
|
|
|
iContext - index of the context in UnwindContext
|
|
|
|
Return Value:
|
|
|
|
Returns true whether any failure have been reported
|
|
|
|
--*/
|
|
|
|
BOOL
|
|
Vwndia64FailureReported(UINT iContext)
|
|
{
|
|
PVWNDIA64_UNWIND_CONTEXT pContext;
|
|
|
|
pContext = tlsvar(UnwindContext);
|
|
return (iContext < VWNDIA64_UNWIND_CONTEXT_TABLE_SIZE) ?
|
|
pContext[iContext].bFailureReported : FALSE;
|
|
}
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Reports failure and sets appropriate flag is the unwind context
|
|
|
|
Arguments:
|
|
|
|
iContext - index of the context in UnwindContext;
|
|
use (-1) to report unconditionaly
|
|
|
|
Return Value:
|
|
|
|
none
|
|
|
|
--*/
|
|
|
|
void
|
|
Vwndia64ReportFailure(UINT iContext, LPCSTR szFormat, ...)
|
|
{
|
|
if ((g.SymOptions & SYMOPT_DEBUG) != SYMOPT_DEBUG)
|
|
{
|
|
return;
|
|
}
|
|
|
|
if (iContext < VWNDIA64_UNWIND_CONTEXT_TABLE_SIZE)
|
|
{
|
|
PVWNDIA64_UNWIND_CONTEXT pContext = tlsvar(UnwindContext) + iContext;
|
|
if (pContext->bFailureReported)
|
|
{
|
|
return;
|
|
}
|
|
pContext->bFailureReported = TRUE;
|
|
}
|
|
|
|
//
|
|
// Printout failure
|
|
//
|
|
{
|
|
char buf[1024];
|
|
va_list args;
|
|
|
|
va_start(args, szFormat);
|
|
CopyStrArray(buf, "DBGHELP: ");
|
|
_vsnprintf(buf + 9, sizeof(buf) - 11, szFormat, args);
|
|
va_end(args);
|
|
|
|
dprint(NULL, buf);
|
|
}
|
|
}
|
|
|
|
UCHAR
|
|
NewParsePrologueRegionPhase0 (
|
|
IN PUNWIND_CONTEXT UwContext,
|
|
IN PSTATE_RECORD StateRecord,
|
|
IN OUT PUCHAR AbiImmContext
|
|
);
|
|
|
|
VOID
|
|
NewParsePrologueRegionPhase1 (
|
|
IN PUNWIND_CONTEXT UwContext,
|
|
IN PSTATE_RECORD StateRecord
|
|
);
|
|
|
|
|
|
VOID
|
|
SrInitialize (
|
|
IN PSTATE_RECORD_STACK StateTable,
|
|
IN PSTATE_RECORD StateRecord,
|
|
IN ULONG Size
|
|
)
|
|
{
|
|
StateTable->Size = Size;
|
|
StateTable->Base = StateRecord;
|
|
StateTable->Top = StateRecord;
|
|
StateTable->Current = StateRecord;
|
|
RtlZeroMemory(StateTable->Top, sizeof(STATE_RECORD));
|
|
}
|
|
|
|
ULONG
|
|
ReadLEB128 (
|
|
IN PUCHAR Descriptors,
|
|
IN OUT PULONG CurrentDescIndex
|
|
)
|
|
{
|
|
PUCHAR Buffer;
|
|
ULONG Value;
|
|
ULONG ShiftCount = 7;
|
|
ULONG Count;
|
|
|
|
Buffer = Descriptors + *CurrentDescIndex;
|
|
Count = 1;
|
|
|
|
Value = Buffer[0] & 0x7F;
|
|
if (Buffer[0] & 0x80) {
|
|
while (TRUE) {
|
|
Value += ((Buffer[Count] & 0x7F) << ShiftCount);
|
|
if (Buffer[Count++] & 0x80) {
|
|
ShiftCount += 7;
|
|
} else {
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
*CurrentDescIndex += Count;
|
|
|
|
return Value;
|
|
}
|
|
|
|
ULONGLONG
|
|
RestorePreservedRegisterFromGR (
|
|
IN PIA64_CONTEXT Context,
|
|
IN SHORT BsFrameSize,
|
|
IN SHORT RNatSaveIndex,
|
|
IN SHORT GrNumber,
|
|
#ifdef _IMAGEHLP_SOURCE_
|
|
IN HANDLE hProcess,
|
|
IN PREAD_PROCESS_MEMORY_ROUTINE64 ReadMemory,
|
|
IN UINT iContext,
|
|
OUT BOOL *Succeed,
|
|
#else
|
|
OUT ULONG64 *SourceAddress,
|
|
#endif // _IMAGEHLP_SOURCE_
|
|
OUT PUCHAR Nat OPTIONAL
|
|
)
|
|
{
|
|
ULONGLONG Result = 0;
|
|
SHORT Offset;
|
|
SHORT Temp;
|
|
#ifdef _IMAGEHLP_SOURCE_
|
|
ULONG Size;
|
|
#endif // _IMAGEHLP_SOURCE_
|
|
|
|
#ifdef _IMAGEHLP_SOURCE_
|
|
*Succeed = FALSE;
|
|
#endif // _IMAGEHLP_SOURCE_
|
|
|
|
if (GrNumber >= STATIC_REGISTER_SET_SIZE) {
|
|
|
|
Offset = GrNumber - STATIC_REGISTER_SET_SIZE;
|
|
if ( Offset < BsFrameSize ) {
|
|
|
|
Temp = Offset + RNatSaveIndex - IA64_NAT_BITS_PER_RNAT_REG;
|
|
while (Temp >= 0) {
|
|
Offset++;
|
|
Temp -= IA64_NAT_BITS_PER_RNAT_REG;
|
|
}
|
|
Offset = Offset * sizeof(ULONGLONG);
|
|
|
|
#ifdef _IMAGEHLP_SOURCE_
|
|
{
|
|
ULONGLONG Reg = (ULONGLONG)Context->RsBSP + (ULONGLONG)Offset;
|
|
Vwndia64FixAddress(iContext, &Reg);
|
|
*Succeed = ReadMemory(hProcess, Reg,
|
|
&Result, sizeof(ULONGLONG), &Size);
|
|
}
|
|
#else
|
|
*SourceAddress = (ULONG64)(Context->RsBSP + Offset);
|
|
Result = *(PULONGLONG)(Context->RsBSP + Offset);
|
|
#endif // _IMAGEHLP_SOURCE_
|
|
|
|
} else {
|
|
|
|
UW_DEBUG((1, "ERROR: Invalid GR!\n"));
|
|
}
|
|
|
|
} else {
|
|
|
|
if (GrNumber == 0 || GrNumber == 12) {
|
|
|
|
//
|
|
// Invalid GR number -> Invalid Unwind Descriptor
|
|
//
|
|
|
|
UW_DEBUG((1, "ERROR: Invalid GR!\n"));
|
|
|
|
} else {
|
|
|
|
UW_DEBUG((1, "WARNING: Target register is not a stacked GR!\n"));
|
|
Offset = GrNumber - 1;
|
|
NOT_IMAGEHLP(*SourceAddress = (ULONG64)(&Context->IntGp + Offset));
|
|
Result = *(&Context->IntGp + Offset);
|
|
|
|
#ifdef _IMAGEHLP_SOURCE_
|
|
*Succeed = TRUE;
|
|
#endif // _IMAGEHLP_SOURCE_
|
|
|
|
}
|
|
}
|
|
|
|
if (ARGUMENT_PRESENT(Nat)) {
|
|
|
|
//
|
|
// TBD: Pick up the corresponding Nat bit
|
|
//
|
|
|
|
*Nat = (UCHAR) 0;
|
|
|
|
}
|
|
|
|
return (Result);
|
|
}
|
|
|
|
UCHAR
|
|
ParseBodyRegionDescriptors (
|
|
IN PUNWIND_CONTEXT UnwindContext,
|
|
IN PSTATE_RECORD_STACK StateTable,
|
|
IN ULONG RegionLen
|
|
)
|
|
{
|
|
LABEL Label;
|
|
UCHAR FirstByte;
|
|
BOOLEAN EcountDefined;
|
|
BOOLEAN CopyLabel;
|
|
ULONG Ecount;
|
|
ULONG SlotOffset;
|
|
PSTATE_RECORD StateTablePtr;
|
|
PUCHAR Descriptors;
|
|
|
|
CopyLabel = EcountDefined = FALSE;
|
|
Descriptors = UnwindContext->Descriptors;
|
|
|
|
while (UnwindContext->DescCount < UnwindContext->Size) {
|
|
|
|
FirstByte = Descriptors[UnwindContext->DescCount++];
|
|
|
|
if ( (FirstByte & B1_MASK) == B1_PREFIX ) {
|
|
|
|
Label = (LABEL)(FirstByte & B1_LABEL_MASK);
|
|
if (FirstByte & B1_TYPE_MASK) {
|
|
|
|
//
|
|
// copy the entry state
|
|
//
|
|
|
|
CopyLabel = TRUE;
|
|
|
|
} else {
|
|
|
|
//
|
|
// label the entry state
|
|
//
|
|
|
|
LABEL_REGION(StateTable->Top, Label);
|
|
}
|
|
|
|
UW_DEBUG((1, "Body region desc B1: copy=%d, label_num=%d\n",
|
|
FirstByte & B1_TYPE_MASK ? TRUE : FALSE, Label));
|
|
|
|
} else if ( (FirstByte & B2_MASK) == B2_PREFIX ) {
|
|
|
|
Ecount = FirstByte & B2_ECOUNT_MASK;
|
|
SlotOffset = ReadLEB128(Descriptors, &UnwindContext->DescCount);
|
|
EcountDefined = TRUE;
|
|
|
|
UW_DEBUG((1, "Epilog desc B2: ecount=%d, LEB128(slot)=%d\n",
|
|
Ecount, SlotOffset));
|
|
|
|
} else if ( (FirstByte & B3_MASK) == B3_PREFIX ) {
|
|
|
|
SlotOffset = ReadLEB128(Descriptors, &UnwindContext->DescCount);
|
|
Ecount = ReadLEB128(Descriptors, &UnwindContext->DescCount);
|
|
EcountDefined = TRUE;
|
|
|
|
UW_DEBUG((1, "Epilog desc B3: ecount=%d, LEB128 val=%d\n",
|
|
Ecount, SlotOffset));
|
|
|
|
} else if ( (FirstByte & B4_MASK) == B4_PREFIX ) {
|
|
|
|
Label = ReadLEB128(Descriptors, &UnwindContext->DescCount);
|
|
|
|
if (FirstByte & B4_TYPE_MASK) {
|
|
|
|
//
|
|
// copy the entry state
|
|
//
|
|
|
|
CopyLabel = TRUE;
|
|
|
|
} else {
|
|
|
|
//
|
|
// label the current top of stack
|
|
//
|
|
|
|
LABEL_REGION(StateTable->Top, Label);
|
|
}
|
|
|
|
UW_DEBUG((1, "Body region desc B4: copy=%d, label_num=%d\n",
|
|
FirstByte & B4_TYPE_MASK, Label));
|
|
|
|
} else {
|
|
|
|
//
|
|
// Encounter another region header record
|
|
//
|
|
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (CopyLabel) {
|
|
StateTablePtr = StateTable->Top;
|
|
while (TRUE) {
|
|
if (CHECK_LABEL(StateTablePtr, Label)) {
|
|
StateTable->Current = StateTablePtr;
|
|
break;
|
|
} else if ((StateTablePtr == StateTable->Base)) {
|
|
UW_DEBUG((1, "Undefined Label %d\n", Label));
|
|
break;
|
|
}
|
|
StateTablePtr--;
|
|
}
|
|
}
|
|
|
|
if (EcountDefined) {
|
|
|
|
Ecount++; // Ecount specifies additional level of prologue
|
|
// regions to undo (i.e. a value of 0 implies 1
|
|
// prologue region)
|
|
|
|
if (UnwindContext->ActiveRegionFound == FALSE) {
|
|
while (Ecount-- > 0) {
|
|
if (StateTable->Current->Previous) {
|
|
StateTable->Current = StateTable->Current->Previous;
|
|
}
|
|
|
|
#if DBG
|
|
else {
|
|
UW_DEBUG((1, "WARNING: Ecount is greater than the # of active prologues!\n"));
|
|
}
|
|
#endif // DBG
|
|
|
|
}
|
|
} else {
|
|
|
|
//
|
|
// control PC is in this body/epilog region
|
|
//
|
|
|
|
if ((UnwindContext->SlotCount + RegionLen - SlotOffset)
|
|
<= UnwindContext->TargetSlot)
|
|
{
|
|
PSTATE_RECORD SrPointer;
|
|
|
|
StateTable->Current->Ecount = Ecount;
|
|
SrPointer = StateTable->Current;
|
|
while (Ecount > 0) {
|
|
|
|
if (SrPointer->Previous) {
|
|
SrPointer->Ecount = Ecount;
|
|
SrPointer->SpWhen = 0;
|
|
SrPointer->SpAdjustment = 0;
|
|
SrPointer = SrPointer->Previous;
|
|
}
|
|
|
|
#if DBG
|
|
else {
|
|
UW_DEBUG((1, "WARNING: Ecount is greater than the # of active prologues!\n"));
|
|
}
|
|
#endif // DBG
|
|
Ecount--;
|
|
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return FirstByte;
|
|
}
|
|
|
|
ULONGLONG
|
|
ProcessInterruptRegion (
|
|
#ifdef _IMAGEHLP_SOURCE_
|
|
IN HANDLE hProcess,
|
|
IN PREAD_PROCESS_MEMORY_ROUTINE64 ReadMemory,
|
|
IN UINT iContext,
|
|
#else
|
|
IN PIA64_KNONVOLATILE_CONTEXT_POINTERS ContextPointers,
|
|
#endif _IMAGEHLP_SOURCE_
|
|
IN PUNWIND_CONTEXT UnwindContext,
|
|
IN PIA64_CONTEXT Context,
|
|
IN SHORT BsFrameSize,
|
|
IN SHORT RNatSaveIndex,
|
|
IN UCHAR AbiImmContext
|
|
)
|
|
{
|
|
//
|
|
// no prologue descriptor in interrupt region.
|
|
//
|
|
|
|
ULONGLONG NextPc;
|
|
ULONG Index;
|
|
SHORT TempFrameSize;
|
|
BOOLEAN Success;
|
|
#ifdef _IMAGEHLP_SOURCE_
|
|
ULONG Size;
|
|
#else
|
|
PIA64_CONTEXT PrevContext;
|
|
PVOID *Source;
|
|
PVOID Address;
|
|
#endif _IMAGEHLP_SOURCE_
|
|
|
|
|
|
if (AbiImmContext != IA64_CONTEXT_FRAME) {
|
|
|
|
PIA64_KTRAP_FRAME TrapFrame;
|
|
PIA64_KEXCEPTION_FRAME ExFrame;
|
|
#ifdef _IMAGEHLP_SOURCE_
|
|
IA64_KTRAP_FRAME TF;
|
|
IA64_KEXCEPTION_FRAME ExF;
|
|
#endif // _IMAGEHLP_SOURCE_
|
|
|
|
TrapFrame = (PIA64_KTRAP_FRAME) Context->IntSp;
|
|
#ifdef _IMAGEHLP_SOURCE_
|
|
if (!ReadMemory(hProcess, Context->IntSp, &TF, sizeof(IA64_KTRAP_FRAME), &Size))
|
|
{
|
|
return 0;
|
|
}
|
|
TrapFrame = &TF;
|
|
#endif // _IMAGEHLP_SOURCE_
|
|
|
|
Context->ApDCR = TrapFrame->ApDCR;
|
|
Context->ApUNAT = TrapFrame->ApUNAT;
|
|
Context->StFPSR = TrapFrame->StFPSR;
|
|
Context->Preds = TrapFrame->Preds;
|
|
Context->IntSp = TrapFrame->IntSp;
|
|
Context->StIPSR = TrapFrame->StIPSR;
|
|
Context->StIFS = TrapFrame->StIFS;
|
|
Context->BrRp = TrapFrame->BrRp;
|
|
Context->RsPFS = TrapFrame->RsPFS;
|
|
|
|
#ifndef _IMAGEHLP_SOURCE_
|
|
if (ARGUMENT_PRESENT(ContextPointers)) {
|
|
ContextPointers->ApUNAT = &TrapFrame->ApUNAT;
|
|
ContextPointers->IntSp = &TrapFrame->IntSp;
|
|
ContextPointers->BrRp = &TrapFrame->BrRp;
|
|
ContextPointers->RsPFS = &TrapFrame->RsPFS;
|
|
ContextPointers->Preds = &TrapFrame->Preds;
|
|
}
|
|
#endif // _IMAGEHLP_SOURCE_
|
|
|
|
switch (AbiImmContext) {
|
|
|
|
case IA64_SYSCALL_FRAME:
|
|
|
|
//
|
|
// System Call Handler Frame
|
|
//
|
|
|
|
BsFrameSize = (SHORT)(TrapFrame->StIFS >> IA64_PFS_SIZE_SHIFT);
|
|
BsFrameSize &= IA64_PFS_SIZE_MASK;
|
|
break;
|
|
|
|
case IA64_INTERRUPT_FRAME:
|
|
case IA64_EXCEPTION_FRAME:
|
|
|
|
//
|
|
// External Interrupt Frame / Exception Frame
|
|
//
|
|
|
|
BsFrameSize = (SHORT)TrapFrame->StIFS & IA64_PFS_SIZE_MASK;
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
}
|
|
|
|
RNatSaveIndex = (SHORT)(TrapFrame->RsBSP >> 3) & IA64_NAT_BITS_PER_RNAT_REG;
|
|
TempFrameSize = BsFrameSize - RNatSaveIndex;
|
|
while (TempFrameSize > 0) {
|
|
BsFrameSize++;
|
|
TempFrameSize -= IA64_NAT_BITS_PER_RNAT_REG;
|
|
}
|
|
|
|
NextPc = Ia64InsertIPSlotNumber(TrapFrame->StIIP,
|
|
((TrapFrame->StIPSR >> PSR_RI) & 0x3));
|
|
|
|
#ifdef _IMAGEHLP_SOURCE_
|
|
if (TrapFrame->RsBSP != Context->RsBSP &&
|
|
TrapFrame->RsBSP != Context->RsBSP +
|
|
BsFrameSize * sizeof(ULONGLONG)) {
|
|
ULONG64 Fixup = Context->RsBSP;
|
|
if ((TrapFrame->RsBSP & 0x1ff) ==
|
|
((Fixup + BsFrameSize * sizeof(ULONGLONG)) & 0x1ff)) {
|
|
Fixup += BsFrameSize * sizeof(ULONGLONG);
|
|
}
|
|
Vwndia64AddFixupRegion(iContext,
|
|
NextPc,
|
|
TrapFrame->RsBSPSTORE, TrapFrame->RsBSP,
|
|
Fixup);
|
|
}
|
|
#endif // _IMAGEHLP_SOURCE_
|
|
|
|
Context->RsBSP = TrapFrame->RsBSP - BsFrameSize * sizeof(ULONGLONG);
|
|
Context->RsBSPSTORE = Context->RsBSP;
|
|
Context->RsRNAT = TrapFrame->RsRNAT;
|
|
|
|
return (NextPc);
|
|
}
|
|
|
|
//
|
|
// Kernel-to-User thunk, context of the previous frame can be
|
|
// found on the user stack (i.e. context's address = sp+SCRATCH_AREA)
|
|
//
|
|
|
|
#ifdef _IMAGEHLP_SOURCE_
|
|
if (!ReadMemory(hProcess,
|
|
Context->IntSp + IA64_STACK_SCRATCH_AREA,
|
|
Context,
|
|
sizeof(IA64_CONTEXT),
|
|
&Size))
|
|
{
|
|
return 0;
|
|
}
|
|
NextPc = Ia64InsertIPSlotNumber(Context->StIIP,
|
|
((Context->StIPSR >> PSR_RI) & 0x3));
|
|
#else
|
|
|
|
PrevContext = (PIA64_CONTEXT)(Context->IntSp + IA64_STACK_SCRATCH_AREA);
|
|
|
|
RtlCopyMemory(&Context->BrRp, &PrevContext->BrRp,
|
|
(NUMBER_OF_PRESERVED_BR+3) * sizeof(ULONGLONG));
|
|
RtlCopyMemory(&Context->FltS0, &PrevContext->FltS0,
|
|
NUMBER_OF_LOW_PRESERVED_FR * sizeof(FLOAT128));
|
|
RtlCopyMemory(&Context->FltS4, &PrevContext->FltS4,
|
|
NUMBER_OF_HIGH_PRESERVED_FR * sizeof(FLOAT128));
|
|
RtlCopyMemory(&Context->IntS0, &PrevContext->IntS0,
|
|
NUMBER_OF_PRESERVED_GR * sizeof(ULONGLONG));
|
|
RtlCopyMemory(&Context->IntV0, &PrevContext->IntV0,
|
|
NUMBER_OF_SCRATCH_GR * sizeof(ULONGLONG));
|
|
Context->IntT0 = PrevContext->IntT0;
|
|
Context->IntT1 = PrevContext->IntT1;
|
|
|
|
Context->IntSp = PrevContext->IntSp;
|
|
Context->IntNats = PrevContext->IntNats;
|
|
Context->ApUNAT = PrevContext->ApUNAT;
|
|
Context->ApLC = PrevContext->ApLC;
|
|
Context->ApEC = PrevContext->ApEC;
|
|
Context->Preds = PrevContext->Preds;
|
|
Context->RsPFS = PrevContext->RsPFS;
|
|
Context->RsBSP = PrevContext->RsBSP;
|
|
Context->RsBSPSTORE = PrevContext->RsBSPSTORE;
|
|
Context->RsRSC = PrevContext->RsRSC;
|
|
Context->RsRNAT = PrevContext->RsRNAT;
|
|
Context->StIFS = PrevContext->StIFS;
|
|
Context->StIPSR = PrevContext->StIPSR;
|
|
NextPc = Ia64InsertIPSlotNumber(PrevContext->StIIP,
|
|
((PrevContext->StIPSR >> PSR_RI) & 0x3));
|
|
|
|
#endif // _IMAGEHLP_SOURCE_
|
|
|
|
return(NextPc);
|
|
}
|
|
|
|
|
|
#ifdef _IMAGEHLP_SOURCE_
|
|
|
|
BOOL
|
|
IsValidUnwindInfo64 (
|
|
const IA64_UNWIND_INFO* pUnwindInfo
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This function validates the unwind info record
|
|
|
|
Return Value:
|
|
|
|
returns false if record looks like invalid
|
|
--*/
|
|
|
|
{
|
|
if (!pUnwindInfo) return FALSE;
|
|
|
|
switch (pUnwindInfo->Version) {
|
|
case 1:
|
|
case 2:
|
|
case 3:
|
|
if (
|
|
!pUnwindInfo->DataLength ||
|
|
(pUnwindInfo->DataLength > ((1024 / sizeof(ULONG64)) * 64)))
|
|
{
|
|
return FALSE;
|
|
}
|
|
break;
|
|
default:
|
|
return FALSE;
|
|
} // switch
|
|
|
|
return TRUE;
|
|
} // IsValidUnwindInfo64
|
|
|
|
BOOL
|
|
GetUnwindInfo(
|
|
IN HANDLE hProcess,
|
|
IN ULONGLONG ImageBase,
|
|
IN ULONG UnwindInfoAddress,
|
|
|
|
IN OUT PIA64_UNWIND_INFO UnwindInfo,
|
|
IN OUT PUCHAR* Descriptors,
|
|
IN OUT PULONG DescriptorsSize,
|
|
|
|
IN PREAD_PROCESS_MEMORY_ROUTINE64 ReadMemory
|
|
)
|
|
{
|
|
ULONG BytesRead = 0;
|
|
|
|
//
|
|
// try to get UnwindInfo from the image
|
|
//
|
|
ULONG64 UnwindInfoPtr = ImageBase + UnwindInfoAddress;
|
|
|
|
if (ReadMemory(hProcess, UnwindInfoPtr,
|
|
UnwindInfo, sizeof(*UnwindInfo), &BytesRead) &&
|
|
(BytesRead == sizeof(*UnwindInfo)) &&
|
|
IsValidUnwindInfo64(UnwindInfo))
|
|
{
|
|
if (!Descriptors)
|
|
{
|
|
return TRUE;
|
|
}
|
|
|
|
if (!UnwindInfo->DataLength)
|
|
{
|
|
*Descriptors = NULL;
|
|
*DescriptorsSize = 0;
|
|
return TRUE;
|
|
}
|
|
|
|
ULONG Size = UnwindInfo->DataLength * sizeof(ULONGLONG);
|
|
*Descriptors = (PUCHAR)MemAlloc(Size);
|
|
|
|
if (*Descriptors &&
|
|
ReadMemory(hProcess,
|
|
UnwindInfoPtr + sizeof(*UnwindInfo),
|
|
*Descriptors, Size, &BytesRead) &&
|
|
BytesRead == Size)
|
|
{
|
|
if (DescriptorsSize)
|
|
{
|
|
*DescriptorsSize = Size;
|
|
}
|
|
return TRUE;
|
|
}
|
|
MemFree(*Descriptors);
|
|
}
|
|
|
|
//
|
|
// try to get UnwindInfo from the symbols
|
|
//
|
|
ULONG XDataSize;
|
|
PBYTE pXData =
|
|
(PBYTE)GetUnwindInfoFromSymbols(hProcess,
|
|
ImageBase,
|
|
UnwindInfoAddress,
|
|
&XDataSize);
|
|
|
|
if (pXData && (XDataSize >= sizeof(*UnwindInfo)))
|
|
{
|
|
memcpy(UnwindInfo, pXData, sizeof(*UnwindInfo));
|
|
|
|
if (IsValidUnwindInfo64(UnwindInfo))
|
|
{
|
|
if (!Descriptors)
|
|
{
|
|
return TRUE;
|
|
}
|
|
|
|
if (!UnwindInfo->DataLength)
|
|
{
|
|
*Descriptors = NULL;
|
|
*DescriptorsSize = 0;
|
|
return TRUE;
|
|
}
|
|
|
|
ULONG Size = UnwindInfo->DataLength * sizeof(ULONG64);
|
|
|
|
if (Size <= (XDataSize - sizeof(*UnwindInfo)))
|
|
{
|
|
*Descriptors = (PUCHAR)MemAlloc(Size);
|
|
if (*Descriptors)
|
|
{
|
|
memcpy(*Descriptors,
|
|
pXData + sizeof(*UnwindInfo),
|
|
Size);
|
|
|
|
if (DescriptorsSize)
|
|
{
|
|
*DescriptorsSize = Size;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
MemFree(*Descriptors);
|
|
}
|
|
}
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
#endif // defined(_IMAGEHLP_SOURCE_)
|
|
|
|
ULONGLONG
|
|
RtlVirtualUnwind (
|
|
#ifdef _IMAGEHLP_SOURCE_
|
|
IN HANDLE hProcess,
|
|
IN ULONGLONG ImageBase,
|
|
IN ULONGLONG ControlPc,
|
|
IN PIMAGE_IA64_RUNTIME_FUNCTION_ENTRY FunctionEntry,
|
|
OUT PIA64_CONTEXT ContextRecord,
|
|
IN PREAD_PROCESS_MEMORY_ROUTINE64 ReadMemory,
|
|
IN UINT iContext
|
|
#define ContextPointers ((PIA64_KNONVOLATILE_CONTEXT_POINTERS)0)
|
|
#else
|
|
IN ULONGLONG ImageBase,
|
|
IN ULONGLONG ControlPc,
|
|
IN PIMAGE_IA64_RUNTIME_FUNCTION_ENTRY FunctionEntry,
|
|
IN OUT PIA64_CONTEXT ContextRecord,
|
|
OUT PBOOLEAN InFunction,
|
|
OUT PFRAME_POINTERS EstablisherFrame,
|
|
IN OUT PIA64_KNONVOLATILE_CONTEXT_POINTERS ContextPointers OPTIONAL
|
|
#endif
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This function virtually unwinds the specfified function by executing its
|
|
prologue code backwards.
|
|
|
|
If the function is a leaf function, then the address where control left
|
|
the previous frame is obtained from the context record. If the function
|
|
is a nested function, but not an exception or interrupt frame, then the
|
|
prologue code is executed backwards and the address where control left
|
|
the previous frame is obtained from the updated context record.
|
|
|
|
Otherwise, an exception or interrupt entry to the system is being unwound
|
|
and an especially coded prologue restores the return address twice. Once
|
|
from the fault instruction address and once from the saved return address
|
|
register. The first restore is returned as the function value and the
|
|
second restore is placed in the updated context record.
|
|
|
|
If a context pointers record is specified, then the address where each
|
|
nonvolatile registers is restored from is recorded in the appropriate
|
|
element of the context pointers record.
|
|
|
|
Arguments:
|
|
|
|
ImageBase - Supplies the base address of the module to which the
|
|
function belongs.
|
|
|
|
ControlPc - Supplies the address where control left the specified
|
|
function.
|
|
|
|
FunctionEntry - Supplies the address of the function table entry for the
|
|
specified function.
|
|
|
|
ContextRecord - Supplies the address of a context record.
|
|
|
|
InFunction - Supplies a pointer to a variable that receives whether the
|
|
control PC is within the current function.
|
|
|
|
EstablisherFrame - Supplies a pointer to a variable that receives the
|
|
the establisher frame pointer value.
|
|
|
|
ContextPointers - Supplies an optional pointer to a context pointers
|
|
record.
|
|
|
|
Return Value:
|
|
|
|
The address where control left the previous frame is returned as the
|
|
function value.
|
|
|
|
--*/
|
|
|
|
{
|
|
#ifdef _IMAGEHLP_SOURCE_
|
|
BOOL Succeed;
|
|
#endif // _IMAGEHLP_SOURCE_
|
|
PUCHAR Descriptors = NULL;
|
|
UCHAR AbiImmContext = 0xFF;
|
|
ULONG Mask;
|
|
ULONGLONG NextPc;
|
|
ULONG RegionLen;
|
|
UCHAR FirstByte;
|
|
UCHAR Nat;
|
|
SHORT BsFrameSize; // in 8-byte units
|
|
SHORT LocalFrameSize; // in 8-byte units
|
|
SHORT TempFrameSize; // in 8-byte units
|
|
SHORT RNatSaveIndex;
|
|
ULONG i;
|
|
PULONG Buffer;
|
|
BOOLEAN IsPrologueRegion;
|
|
BOOLEAN PspRestored;
|
|
ULONGLONG PreviousIntSp;
|
|
PVOID Destination;
|
|
ULONG64 Source;
|
|
ULONG64 *CtxPtr;
|
|
ULONG64 *NatCtxPtr;
|
|
ULONG64 IntNatsSource;
|
|
ULONG64 IntNats;
|
|
ULONG Size;
|
|
ULONG DescrSize;
|
|
ULONGLONG OldTopRnat;
|
|
ULONGLONG NewTopRnat;
|
|
UNWIND_CONTEXT UnwindContext;
|
|
PSTATE_RECORD SrPointer;
|
|
STATE_RECORD_STACK StateTable;
|
|
STATE_RECORD StateRecords[STATE_RECORD_STACK_SIZE];
|
|
|
|
BsFrameSize = (SHORT)ContextRecord->StIFS & IA64_PFS_SIZE_MASK;
|
|
RNatSaveIndex = (SHORT)(ContextRecord->RsBSP >> 3) & IA64_NAT_BITS_PER_RNAT_REG;
|
|
TempFrameSize = RNatSaveIndex + BsFrameSize - IA64_NAT_BITS_PER_RNAT_REG;
|
|
while (TempFrameSize >= 0) {
|
|
BsFrameSize++;
|
|
TempFrameSize -= IA64_NAT_BITS_PER_RNAT_REG;
|
|
}
|
|
|
|
#ifdef _IMAGEHLP_SOURCE_
|
|
{
|
|
IA64_UNWIND_INFO UnwindInfo;
|
|
if (!GetUnwindInfo(hProcess, ImageBase, FunctionEntry->UnwindInfoAddress,
|
|
&UnwindInfo, &Descriptors, &Size,
|
|
ReadMemory))
|
|
{
|
|
Vwndia64ReportFailure(-1,
|
|
"Can't read Unwind Info - "
|
|
"StackWalk can not continue\n");
|
|
return 0;
|
|
}
|
|
|
|
UnwindContext.Version = UnwindInfo.Version;
|
|
}
|
|
#else
|
|
{
|
|
ULONG64 UnwindInfoPtr = ImageBase + FunctionEntry->UnwindInfoAddress;
|
|
UnwindContext.Version = ((PIA64_UNWIND_INFO)UnwindInfoPtr)->Version;
|
|
Size = ((PIA64_UNWIND_INFO)UnwindInfoPtr)->DataLength * sizeof(ULONGLONG);
|
|
Descriptors = (PUCHAR)UnwindInfoPtr + sizeof(IA64_UNWIND_INFO);
|
|
}
|
|
#endif // _IMAGEHLP_SOURCE_
|
|
|
|
UnwindContext.Size = Size;
|
|
UnwindContext.ActiveRegionFound = FALSE;
|
|
UnwindContext.AlternateRp = 0;
|
|
UnwindContext.DescCount = 0;
|
|
UnwindContext.SlotCount = 0;
|
|
UnwindContext.TargetSlot = (ULONG)(((ControlPc - FunctionEntry->BeginAddress - ImageBase) >> 4) * SLOTS_PER_BUNDLE + ((ControlPc >> 2) & 0x3));
|
|
UnwindContext.Descriptors = Descriptors;
|
|
|
|
SrInitialize(&StateTable, StateRecords, STATE_RECORD_STACK_SIZE);
|
|
|
|
if (Size) {
|
|
FirstByte = Descriptors[UnwindContext.DescCount++];
|
|
}
|
|
|
|
|
|
while ( (UnwindContext.DescCount < UnwindContext.Size) &&
|
|
(!UnwindContext.ActiveRegionFound) )
|
|
{
|
|
|
|
//
|
|
// Assume a prologue region but not an interrupt region.
|
|
//
|
|
|
|
IsPrologueRegion = TRUE;
|
|
|
|
//
|
|
// Based on the type of region header, dispatch
|
|
// to the corresponding routine that processes
|
|
// the succeeding descriptors until the next
|
|
// region header record.
|
|
//
|
|
|
|
if ((FirstByte & R1_MASK) == R1_PREFIX) {
|
|
|
|
//
|
|
// region header record in short format
|
|
//
|
|
|
|
RegionLen = FirstByte & R1_LENGTH_MASK;
|
|
|
|
if (FirstByte & R1_REGION_TYPE_MASK) {
|
|
IsPrologueRegion = FALSE;
|
|
} else {
|
|
ADD_STATE_RECORD(StateTable, RegionLen, UnwindContext.DescCount);
|
|
}
|
|
|
|
UW_DEBUG((1, "Region R1 format: body=%x, length=%d\n",
|
|
IsPrologueRegion ? 0 : 1, RegionLen));
|
|
|
|
} else if ((FirstByte & R2_MASK) == R2_PREFIX) {
|
|
|
|
//
|
|
// general prologue region header
|
|
// N.B. Skip the 2nd byte of the header and proceed to read
|
|
// the region length; the header descriptors will be
|
|
// processed again in phase 1.
|
|
//
|
|
|
|
ULONG R2DescIndex;
|
|
|
|
R2DescIndex = UnwindContext.DescCount - 1;
|
|
UnwindContext.DescCount++;
|
|
RegionLen = ReadLEB128(Descriptors, &UnwindContext.DescCount);
|
|
ADD_STATE_RECORD(StateTable, RegionLen, R2DescIndex);
|
|
UW_DEBUG((1, "Region R2: body=0, length=%d\n", RegionLen));
|
|
|
|
} else if ((FirstByte & R3_MASK) == R3_PREFIX) {
|
|
|
|
//
|
|
// region header record in long format
|
|
//
|
|
|
|
RegionLen = ReadLEB128(Descriptors, &UnwindContext.DescCount);
|
|
|
|
switch (FirstByte & R3_REGION_TYPE_MASK) {
|
|
|
|
case 0: // prologue region header
|
|
|
|
ADD_STATE_RECORD(StateTable, RegionLen, UnwindContext.DescCount);
|
|
break;
|
|
|
|
case 1: // body region header
|
|
|
|
IsPrologueRegion = FALSE;
|
|
break;
|
|
|
|
}
|
|
|
|
UW_DEBUG((1, "Region R3: body=%x, length=%d\n",
|
|
IsPrologueRegion ? 0 : 1, RegionLen));
|
|
|
|
} else {
|
|
|
|
//
|
|
// Not a region header record -> Invalid unwind descriptor.
|
|
//
|
|
|
|
UW_DEBUG((1, "Invalid unwind descriptor!\n"));
|
|
|
|
}
|
|
|
|
if (UnwindContext.TargetSlot < (UnwindContext.SlotCount + RegionLen)) {
|
|
UnwindContext.ActiveRegionFound = TRUE;
|
|
StateTable.Current->IsTarget = IsPrologueRegion;
|
|
}
|
|
|
|
if (IsPrologueRegion) {
|
|
FirstByte = NewParsePrologueRegionPhase0(&UnwindContext,
|
|
StateTable.Current,
|
|
&AbiImmContext);
|
|
} else {
|
|
FirstByte = ParseBodyRegionDescriptors(&UnwindContext,
|
|
&StateTable,
|
|
RegionLen);
|
|
}
|
|
|
|
UnwindContext.SlotCount += RegionLen;
|
|
}
|
|
|
|
//
|
|
// Restore the value of psp and save the current NatCr.
|
|
// N.B. If the value is restored from stack/bstore, turn off the
|
|
// corresponding sp bit in the saved mask associated with the
|
|
// prologue region in which psp is saved.
|
|
//
|
|
|
|
if (ARGUMENT_PRESENT(ContextPointers)) {
|
|
IntNatsSource = (ULONG64)ContextPointers->ApUNAT;
|
|
}
|
|
IntNats = ContextRecord->ApUNAT;
|
|
PreviousIntSp = ContextRecord->IntSp;
|
|
PspRestored = FALSE;
|
|
|
|
SrPointer = StateTable.Current;
|
|
while (SrPointer != StateTable.Base) {
|
|
NewParsePrologueRegionPhase1(&UnwindContext, SrPointer);
|
|
|
|
if (SrPointer->MiscMask & (1 << REG_SP)) {
|
|
if (UnwindContext.MiscRegs[REG_SP].Where == GENERAL_REG) {
|
|
PreviousIntSp = RestorePreservedRegisterFromGR (
|
|
ContextRecord,
|
|
BsFrameSize,
|
|
RNatSaveIndex,
|
|
(SHORT)UnwindContext.MiscRegs[REG_SP].SaveOffset,
|
|
#ifdef _IMAGEHLP_SOURCE_
|
|
hProcess,
|
|
ReadMemory,
|
|
iContext,
|
|
&Succeed,
|
|
#else
|
|
&Source,
|
|
#endif // _IMAGEHLP_SOURCE_
|
|
&Nat
|
|
);
|
|
#ifdef _IMAGEHLP_SOURCE_
|
|
if (!Succeed) {
|
|
return 0;
|
|
}
|
|
|
|
UW_DEBUG((1, "Restored IntSp to %I64x\n", PreviousIntSp));
|
|
|
|
#endif // _IMAGEHLP_SOURCE_
|
|
|
|
} else {
|
|
|
|
Source = ContextRecord->IntSp + UnwindContext.MiscRegs[REG_SP].SaveOffset*4;
|
|
#ifdef _IMAGEHLP_SOURCE_
|
|
if (!ReadMemory(hProcess, (ULONG64)(Source), &PreviousIntSp, sizeof(ULONGLONG), &Size)) {
|
|
return 0;
|
|
}
|
|
#else
|
|
PreviousIntSp = *(PULONGLONG)Source;
|
|
#endif // _IMAGEHLP_SOURCE_
|
|
EXTRACT_NAT_FROM_UNAT(Nat);
|
|
|
|
}
|
|
ContextRecord->IntNats &= ~(0x1 << STACK_POINTER_GR);
|
|
ContextRecord->IntNats |= (Nat << STACK_POINTER_GR);
|
|
SrPointer->MiscMask &= ~(1 << REG_SP);
|
|
if (ARGUMENT_PRESENT(ContextPointers)) {
|
|
CtxPtr = (ULONG64 *)((ULONG_PTR)ContextPointers +
|
|
MiscContextPointersOffset[REG_SP]);
|
|
*CtxPtr = Source;
|
|
}
|
|
PspRestored = TRUE;
|
|
}
|
|
if (PspRestored == FALSE) {
|
|
PreviousIntSp += SrPointer->SpAdjustment * 4;
|
|
}
|
|
SrPointer = SrPointer->Previous;
|
|
}
|
|
|
|
if (AbiImmContext != 0xFF) {
|
|
|
|
ContextRecord->IntSp = PreviousIntSp; // trap/context frame address
|
|
NextPc = ProcessInterruptRegion(
|
|
#ifdef _IMAGEHLP_SOURCE_
|
|
hProcess,
|
|
ReadMemory,
|
|
iContext,
|
|
#else
|
|
ContextPointers,
|
|
#endif _IMAGEHLP_SOURCE_
|
|
&UnwindContext,
|
|
ContextRecord,
|
|
BsFrameSize,
|
|
RNatSaveIndex,
|
|
AbiImmContext);
|
|
|
|
goto FastExit;
|
|
}
|
|
|
|
//
|
|
// Restore the contents of any preserved registers saved in this frame.
|
|
//
|
|
|
|
SrPointer = StateTable.Current;
|
|
while (SrPointer != StateTable.Base) {
|
|
|
|
Mask = SrPointer->MiscMask;
|
|
UW_DEBUG((1, "MiscMask = 0x%x\n", Mask));
|
|
|
|
for (i = 0; i < NUMBER_OF_PRESERVED_REGISTERS; i++) {
|
|
Destination = (PVOID)((ULONG_PTR)ContextRecord + MiscContextOffset[i]);
|
|
if (Mask & 0x1) {
|
|
|
|
if (ARGUMENT_PRESENT(ContextPointers)) {
|
|
CtxPtr = (ULONG64 *)((ULONG_PTR)ContextPointers +
|
|
MiscContextPointersOffset[i]);
|
|
Source = *CtxPtr;
|
|
}
|
|
|
|
if (UnwindContext.MiscRegs[i].Where == GENERAL_REG) {
|
|
|
|
*(PULONGLONG)Destination =
|
|
RestorePreservedRegisterFromGR (
|
|
ContextRecord,
|
|
BsFrameSize,
|
|
RNatSaveIndex,
|
|
(SHORT)UnwindContext.MiscRegs[i].SaveOffset,
|
|
#ifdef _IMAGEHLP_SOURCE_
|
|
hProcess,
|
|
ReadMemory,
|
|
iContext,
|
|
&Succeed,
|
|
#else
|
|
&Source,
|
|
#endif // _IMAGEHLP_SOURCE_
|
|
NULL
|
|
);
|
|
#ifdef _IMAGEHLP_SOURCE_
|
|
if (!Succeed) {
|
|
*(PULONGLONG)Destination = 0;
|
|
} else {
|
|
UW_DEBUG((1, "Restored %s to %I64x\n",
|
|
RegOffsetNames[i],
|
|
*(PULONGLONG)Destination));
|
|
}
|
|
|
|
#endif // _IMAGEHLP_SOURCE_
|
|
|
|
} else if (UnwindContext.MiscRegs[i].Where == BRANCH_REG) {
|
|
|
|
//
|
|
// restore return pointer from branch register
|
|
//
|
|
|
|
USHORT Offset;
|
|
|
|
Offset = (USHORT)UnwindContext.MiscRegs[i].SaveOffset-FIRST_PRESERVED_BR;
|
|
Source = (ULONG64)(&ContextRecord->BrS0 + Offset);
|
|
#ifdef _IMAGEHLP_SOURCE_
|
|
Vwndia64FixAddress(iContext, &Source);
|
|
|
|
if (!ReadMemory(hProcess, (ULONG64)(Source), Destination, sizeof(ULONGLONG), &Size)) {
|
|
*(PULONGLONG)Destination = 0;
|
|
} else {
|
|
UW_DEBUG((1, "Restored %s to %I64x\n",
|
|
RegOffsetNames[i],
|
|
*(PULONGLONG)Destination));
|
|
}
|
|
#else
|
|
*(PULONGLONG)Destination = *(PULONGLONG)(Source);
|
|
#endif // _IMAGEHLP_SOURCE_
|
|
|
|
} else if (UnwindContext.MiscRegs[i].Where == PSP_RELATIVE) {
|
|
|
|
if ((SrPointer->Ecount == 0) || (UnwindContext.MiscRegs[i].SaveOffset <= (IA64_STACK_SCRATCH_AREA/sizeof(ULONG)))) {
|
|
Source = PreviousIntSp + IA64_STACK_SCRATCH_AREA
|
|
- UnwindContext.MiscRegs[i].SaveOffset*4;
|
|
|
|
if (i == REG_NATS) {
|
|
Destination = (PVOID)&IntNats;
|
|
IntNatsSource = Source;
|
|
}
|
|
|
|
#ifdef _IMAGEHLP_SOURCE_
|
|
if (!ReadMemory(hProcess, (ULONG64)(Source), Destination, sizeof(ULONGLONG), &Size)) {
|
|
*(PULONGLONG)Destination = 0;
|
|
} else {
|
|
UW_DEBUG((1, "Restored %s to %I64x\n",
|
|
RegOffsetNames[i],
|
|
*(PULONGLONG)Destination));
|
|
}
|
|
#else
|
|
*(PULONGLONG)Destination = *(PULONGLONG)(Source);
|
|
#endif // _IMAGEHLP_SOURCE_
|
|
}
|
|
|
|
} else if (UnwindContext.MiscRegs[i].Where == SP_RELATIVE) {
|
|
|
|
//
|
|
// Make the necessary adjustment depending on whether
|
|
// the preserved register is saved before or after the
|
|
// stack pointer has been adjusted in this prologue.
|
|
//
|
|
|
|
if (UnwindContext.MiscRegs[i].When >= SrPointer->SpWhen && SrPointer->RegionLen != 0)
|
|
{
|
|
Source = ContextRecord->IntSp
|
|
+ UnwindContext.MiscRegs[i].SaveOffset*4;
|
|
}
|
|
else {
|
|
Source = ContextRecord->IntSp + SrPointer->SpAdjustment*4
|
|
+ UnwindContext.MiscRegs[i].SaveOffset*4;
|
|
}
|
|
|
|
if (i == REG_NATS) {
|
|
Destination = (PVOID)&IntNats;
|
|
IntNatsSource = Source;
|
|
}
|
|
|
|
#ifdef _IMAGEHLP_SOURCE_
|
|
if (!ReadMemory(hProcess, (ULONG64)(Source), Destination, sizeof(ULONGLONG), &Size)) {
|
|
*(PULONGLONG)Destination = 0;
|
|
} else {
|
|
UW_DEBUG((1, "Restored %s to %I64x\n",
|
|
RegOffsetNames[i],
|
|
*(PULONGLONG)Destination));
|
|
}
|
|
#else
|
|
*(PULONGLONG)Destination = *(PULONGLONG)(Source);
|
|
#endif // _IMAGEHLP_SOURCE_
|
|
}
|
|
|
|
if (ARGUMENT_PRESENT(ContextPointers) && (i != REG_NATS)) {
|
|
*CtxPtr = Source;
|
|
}
|
|
|
|
} else if (Mask == 0) {
|
|
|
|
//
|
|
// No more registers to restore
|
|
//
|
|
|
|
break;
|
|
}
|
|
|
|
Mask = Mask >> 1;
|
|
}
|
|
|
|
//
|
|
// Restore preserved FRs (f2 - f5, f16 - f31)
|
|
//
|
|
|
|
Mask = SrPointer->FrMask;
|
|
Destination = (PVOID)&ContextRecord->FltS0;
|
|
CtxPtr = (ULONG64 *)&ContextPointers->FltS0;
|
|
|
|
UW_DEBUG((1, "FrMask = 0x%x\n", Mask));
|
|
for (i = 0; i < NUMBER_OF_PRESERVED_FR; i++) {
|
|
if (Mask & 0x1) {
|
|
|
|
if ((SrPointer->Ecount == 0) || (UnwindContext.Float[i].SaveOffset <= (IA64_STACK_SCRATCH_AREA/sizeof(ULONG)))) {
|
|
Source = PreviousIntSp + IA64_STACK_SCRATCH_AREA
|
|
- UnwindContext.Float[i].SaveOffset*4;
|
|
#ifdef _IMAGEHLP_SOURCE_
|
|
if (!ReadMemory(hProcess, (ULONG64)(Source), Destination, sizeof(FLOAT128), &Size)) {
|
|
*(PULONGLONG)Destination = 0;
|
|
}
|
|
#else
|
|
*(FLOAT128 *)Destination = *(FLOAT128 *)Source;
|
|
#endif // _IMAGEHLP_SOURCE_
|
|
|
|
if (ARGUMENT_PRESENT(ContextPointers)) {
|
|
*CtxPtr = Source;
|
|
}
|
|
}
|
|
|
|
} else if (Mask == 0) {
|
|
break;
|
|
}
|
|
|
|
Mask = Mask >> 1;
|
|
|
|
if (i == (NUMBER_OF_LOW_PRESERVED_FR - 1)) {
|
|
Destination = (PVOID)&ContextRecord->FltS4;
|
|
CtxPtr = (ULONG64 *)(&ContextPointers->FltS4);
|
|
} else {
|
|
Destination = (PVOID)((FLOAT128 *)Destination+1);
|
|
CtxPtr++;
|
|
}
|
|
}
|
|
|
|
//
|
|
// Restore preserved GRs (r4 - r7)
|
|
//
|
|
|
|
Mask = SrPointer->GrMask;
|
|
Destination = (PVOID)&ContextRecord->IntS0;
|
|
CtxPtr = (ULONG64 *)&ContextPointers->IntS0;
|
|
NatCtxPtr = (ULONG64 *)&ContextPointers->IntS0Nat;
|
|
|
|
UW_DEBUG((1, "GrMask = 0x%x\n", Mask));
|
|
for (i = 0; i < NUMBER_OF_PRESERVED_GR; i++)
|
|
{
|
|
if (Mask & 0x1) {
|
|
|
|
if ((SrPointer->Ecount == 0) || (UnwindContext.Integer[i].SaveOffset <= (IA64_STACK_SCRATCH_AREA/sizeof(ULONG)))) {
|
|
Source = PreviousIntSp + IA64_STACK_SCRATCH_AREA
|
|
- UnwindContext.Integer[i].SaveOffset*4;
|
|
|
|
#ifdef _IMAGEHLP_SOURCE_
|
|
if (!ReadMemory(hProcess, (ULONG64)(Source), Destination, sizeof(ULONGLONG), &Size)) {
|
|
*(PULONGLONG)Destination = 0;
|
|
}
|
|
#else
|
|
*(PULONGLONG)Destination = *(PULONGLONG)Source;
|
|
#endif // _IMAGEHLP_SOURCE_
|
|
EXTRACT_NAT_FROM_UNAT(Nat);
|
|
Nat = (UCHAR)((IntNats >> (((ULONG_PTR)Source & 0x1F8) >> 3)) & 0x1);
|
|
ContextRecord->IntNats &= ~(0x1 << (i+FIRST_PRESERVED_GR));
|
|
ContextRecord->IntNats |= (Nat << (i+FIRST_PRESERVED_GR));
|
|
|
|
#ifndef _IMAGEHLP_SOURCE_
|
|
if (ARGUMENT_PRESENT(ContextPointers)) {
|
|
*CtxPtr = Source;
|
|
*NatCtxPtr = IntNatsSource;
|
|
}
|
|
#endif
|
|
}
|
|
|
|
} else if (Mask == 0) {
|
|
break;
|
|
}
|
|
|
|
Mask = Mask >> 1;
|
|
Destination = (PVOID)((PULONGLONG)Destination+1);
|
|
CtxPtr++;
|
|
NatCtxPtr++;
|
|
}
|
|
|
|
ContextRecord->IntSp += SrPointer->SpAdjustment * 4;
|
|
SrPointer = SrPointer->Previous;
|
|
}
|
|
|
|
ContextRecord->IntSp = PreviousIntSp;
|
|
|
|
//
|
|
// Restore the value of the epilogue count from the PFS
|
|
//
|
|
|
|
ContextRecord->ApEC = (ContextRecord->RsPFS >> IA64_PFS_EC_SHIFT) &
|
|
~(((ULONGLONG)1 << IA64_PFS_EC_SIZE) - 1);
|
|
if (ARGUMENT_PRESENT(ContextPointers)) {
|
|
ContextPointers->ApEC = ContextPointers->RsPFS;
|
|
}
|
|
|
|
|
|
FastExit:
|
|
|
|
NOT_IMAGEHLP(*InFunction = TRUE);
|
|
NOT_IMAGEHLP(EstablisherFrame->MemoryStackFp = ContextRecord->IntSp);
|
|
NOT_IMAGEHLP(EstablisherFrame->BackingStoreFp = ContextRecord->RsBSP);
|
|
|
|
#ifdef _IMAGEHLP_SOURCE_
|
|
if (Descriptors)
|
|
MemFree(Descriptors);
|
|
#endif // _IMAGEHLP_SOURCE_
|
|
|
|
if (AbiImmContext == 0xFF) {
|
|
|
|
NextPc = *(&ContextRecord->BrRp + UnwindContext.AlternateRp);
|
|
#ifndef _IMAGEHLP_SOURCE_
|
|
NextPc = RtlIa64InsertIPSlotNumber((NextPc-0x10), 2);
|
|
#endif // _IMAGEHLP_SOURCE_
|
|
|
|
//
|
|
// determine the local frame size of previous frame and compute
|
|
// the new bsp.
|
|
//
|
|
|
|
OldTopRnat = (ContextRecord->RsBSP+(BsFrameSize-1)*8) | IA64_RNAT_ALIGNMENT;
|
|
|
|
ContextRecord->StIFS = MASK(IA64_IFS_V, (ULONGLONG)1) | ContextRecord->RsPFS;
|
|
BsFrameSize = (SHORT)ContextRecord->StIFS & IA64_PFS_SIZE_MASK;
|
|
LocalFrameSize = (SHORT)(ContextRecord->StIFS >> IA64_PFS_SIZE_SHIFT) & IA64_PFS_SIZE_MASK;
|
|
TempFrameSize = LocalFrameSize - RNatSaveIndex;
|
|
while (TempFrameSize > 0) {
|
|
LocalFrameSize++;
|
|
BsFrameSize++;
|
|
TempFrameSize -= IA64_NAT_BITS_PER_RNAT_REG;
|
|
}
|
|
ContextRecord->RsBSP -= LocalFrameSize * 8;
|
|
ContextRecord->RsBSPSTORE = ContextRecord->RsBSP;
|
|
|
|
//
|
|
// determine if the RNAT field needs to be updated.
|
|
//
|
|
|
|
NewTopRnat = (ContextRecord->RsBSP+(BsFrameSize-1)*8) | IA64_RNAT_ALIGNMENT;
|
|
|
|
if (NewTopRnat < OldTopRnat) {
|
|
|
|
#ifdef _IMAGEHLP_SOURCE_
|
|
Destination = &ContextRecord->RsRNAT;
|
|
Source = NewTopRnat;
|
|
if (!ReadMemory(hProcess, (ULONG64)Source, Destination, 8, &Size)) {
|
|
*(PULONGLONG)Destination = 0;
|
|
}
|
|
#else
|
|
ContextRecord->RsRNAT = *(PULONGLONG)(NewTopRnat);
|
|
#endif // _IMAGEHLP_SOURCE_
|
|
|
|
}
|
|
}
|
|
|
|
#ifdef _IMAGEHLP_SOURCE_
|
|
UW_DEBUG((1, "NextPc = 0x%I64x, PSP = 0x%I64x, BSP = 0x%I64x\n",
|
|
(ULONGLONG)NextPc,
|
|
(ULONGLONG)ContextRecord->IntSp,
|
|
(ULONGLONG)ContextRecord->RsBSP));
|
|
#else
|
|
UW_DEBUG((1, "NextPc = 0x%I64x, PSP = 0x%I64x, BSP = 0x%I64x\n",
|
|
(ULONGLONG)NextPc,
|
|
EstablisherFrame->MemoryStackFp,
|
|
EstablisherFrame->BackingStoreFp));
|
|
#endif // _IMAGEHLP_SOURCE_
|
|
return (NextPc);
|
|
}
|
|
|
|
UCHAR
|
|
NewParsePrologueRegionPhase0 (
|
|
IN PUNWIND_CONTEXT UwContext,
|
|
IN PSTATE_RECORD State,
|
|
IN OUT PUCHAR AbiImmContext
|
|
)
|
|
{
|
|
PUCHAR Desc = UwContext->Descriptors;
|
|
ULONG Offset;
|
|
ULONG FrameSize;
|
|
ULONG Index;
|
|
UCHAR RecType;
|
|
UCHAR FirstByte;
|
|
UCHAR SecondByte;
|
|
ULONG GrSave;
|
|
ULONG TempMask;
|
|
ULONG i;
|
|
|
|
FirstByte = 0;
|
|
|
|
while (UwContext->DescCount < UwContext->Size) {
|
|
|
|
FirstByte = Desc[UwContext->DescCount++];
|
|
|
|
if ( (FirstByte & P1_MASK) == P1_PREFIX) {
|
|
|
|
continue;
|
|
|
|
} else if ( (FirstByte & P2_MASK) == P2_PREFIX ) {
|
|
|
|
UwContext->DescCount++;
|
|
|
|
} else if ( (FirstByte & P3_MASK) == P3_PREFIX ) {
|
|
|
|
UwContext->DescCount++;
|
|
|
|
} else if ( (FirstByte & P4_MASK) == P4_PREFIX ) {
|
|
|
|
UwContext->DescCount += ((State->RegionLen+3) >> 2);
|
|
|
|
} else if ( (FirstByte & P5_MASK) == P5_PREFIX ) {
|
|
|
|
UwContext->DescCount += 3;
|
|
|
|
} else if ( (FirstByte & P6_MASK) == P6_PREFIX ) {
|
|
|
|
continue;
|
|
|
|
} else if ( (FirstByte & P7_MASK) == P7_PREFIX ) {
|
|
|
|
RecType = FirstByte & ~P7_MASK;
|
|
|
|
switch (RecType) {
|
|
|
|
case MEM_STACK_F:
|
|
|
|
Offset = ReadLEB128(Desc, &UwContext->DescCount);
|
|
FrameSize = ReadLEB128(Desc, &UwContext->DescCount);
|
|
|
|
if (UwContext->TargetSlot > (UwContext->SlotCount+Offset) || State->RegionLen == 0)
|
|
{
|
|
State->SpAdjustment += FrameSize*4;
|
|
State->SpWhen = Offset;
|
|
}
|
|
break;
|
|
|
|
case SPILL_BASE:
|
|
|
|
State->SpillBase = ReadLEB128(Desc, &UwContext->DescCount);
|
|
State->SpillPtr = State->SpillBase;
|
|
break;
|
|
|
|
case MEM_STACK_V:
|
|
case RP_WHEN:
|
|
case PFS_WHEN:
|
|
case PREDS_WHEN:
|
|
case LC_WHEN:
|
|
case UNAT_WHEN:
|
|
case FPSR_WHEN:
|
|
|
|
Offset = ReadLEB128(Desc, &UwContext->DescCount);
|
|
if ((State->IsTarget) &&
|
|
(UwContext->TargetSlot > (UwContext->SlotCount+Offset)))
|
|
{
|
|
Index = P7RecordTypeToRegisterIndex[RecType];
|
|
if (!(State->MiscMask & (1 << Index))) {
|
|
State->MiscMask |= MASK(Index,1);
|
|
UwContext->MiscRegs[Index].When = Offset;
|
|
} else {
|
|
UW_DEBUG((1, "Duplicate descriptors,"));
|
|
UW_DEBUG((1, "unwinder may produce incorrect result!\n"));
|
|
}
|
|
}
|
|
UW_DEBUG((1, "Prolog P7: type=%d slot= %d\n", RecType, Offset));
|
|
break;
|
|
|
|
case PSP_SPREL:
|
|
case RP_PSPREL:
|
|
case PFS_PSPREL:
|
|
case PREDS_PSPREL:
|
|
case LC_PSPREL:
|
|
case UNAT_PSPREL:
|
|
case FPSR_PSPREL:
|
|
|
|
Offset = ReadLEB128(Desc, &UwContext->DescCount);
|
|
break;
|
|
|
|
default:
|
|
|
|
UW_DEBUG((1, "Invalid record type for descriptor P7!\n"));
|
|
|
|
}
|
|
|
|
} else if ( (FirstByte & P8_MASK) == P8_PREFIX ) {
|
|
|
|
RecType = Desc[UwContext->DescCount++];
|
|
|
|
switch (RecType) {
|
|
|
|
case PSP_PSPREL:
|
|
case RP_SPREL:
|
|
case PFS_SPREL:
|
|
case PREDS_SPREL:
|
|
case LC_SPREL:
|
|
case UNAT_SPREL:
|
|
case FPSR_SPREL:
|
|
case BSP_PSPREL:
|
|
case BSP_SPREL:
|
|
case BSPSTORE_PSPREL:
|
|
case BSPSTORE_SPREL:
|
|
case RNAT_PSPREL:
|
|
case RNAT_SPREL:
|
|
case PRIUNAT_PSPREL:
|
|
case PRIUNAT_SPREL:
|
|
|
|
Offset = ReadLEB128(Desc, &UwContext->DescCount);
|
|
UW_DEBUG((1, "Prolog P8: type=%d slot= %d\n", RecType, Offset));
|
|
break;
|
|
|
|
case BSP_WHEN:
|
|
case BSPSTORE_WHEN:
|
|
case RNAT_WHEN:
|
|
case PRIUNAT_WHEN:
|
|
|
|
Offset = ReadLEB128(Desc, &UwContext->DescCount);
|
|
if ((State->IsTarget) &&
|
|
(UwContext->TargetSlot > (UwContext->SlotCount+Offset)))
|
|
{
|
|
Index = P7RecordTypeToRegisterIndex[RecType];
|
|
if (!(State->MiscMask & (1 << Index))) {
|
|
State->MiscMask |= MASK(Index,1);
|
|
UwContext->MiscRegs[Index].When = Offset;
|
|
} else {
|
|
UW_DEBUG((1, "Duplicate descriptors,"));
|
|
UW_DEBUG((1, "unwinder may produce incorrect result!\n"));
|
|
}
|
|
}
|
|
UW_DEBUG((1, "Prolog P8: type=%d slot= %d\n", RecType, Offset));
|
|
break;
|
|
|
|
default:
|
|
|
|
UW_DEBUG((1, "Invalid record type for descriptor P8!\n"));
|
|
|
|
}
|
|
|
|
} else if ( (FirstByte & P9_MASK) == P9_PREFIX ) {
|
|
|
|
UwContext->DescCount += 2;
|
|
VUW_DEBUG_PRINT("Format P9 not supported yet!\n");
|
|
|
|
} else if ( (FirstByte & P10_MASK) == P10_PREFIX ) {
|
|
|
|
UCHAR Abi = Desc[UwContext->DescCount++];
|
|
UCHAR Context = Desc[UwContext->DescCount++];
|
|
|
|
*AbiImmContext = Context;
|
|
|
|
if (Abi != NT_ABI) {
|
|
VUW_DEBUG_PRINT("Unknown ABI unwind descriptor\n");
|
|
}
|
|
|
|
} else {
|
|
|
|
//
|
|
// Encounter another region header record
|
|
//
|
|
|
|
break;
|
|
}
|
|
}
|
|
|
|
State->DescEnd = UwContext->DescCount - 2;
|
|
|
|
return FirstByte;
|
|
}
|
|
|
|
VOID
|
|
NewParsePrologueRegionPhase1 (
|
|
IN PUNWIND_CONTEXT UwContext,
|
|
IN PSTATE_RECORD State
|
|
)
|
|
{
|
|
ULONG FrameSize;
|
|
ULONG Offset;
|
|
ULONG GrSave;
|
|
ULONG BrBase;
|
|
ULONG Index;
|
|
ULONG Count;
|
|
UCHAR RecType;
|
|
UCHAR FirstByte, SecondByte; // 1st & 2nd bytes of a region header record
|
|
ULONG DescIndex;
|
|
ULONG ImaskBegin;
|
|
UCHAR NextBr, NextGr, NextFr;
|
|
USHORT MiscMask;
|
|
ULONG TempMask;
|
|
ULONG FrMask = 0;
|
|
UCHAR BrMask = 0;
|
|
UCHAR GrMask = 0;
|
|
PUCHAR Desc = UwContext->Descriptors;
|
|
BOOLEAN SpillMaskOmitted = TRUE;
|
|
USHORT i;
|
|
|
|
DescIndex = State->DescBegin;
|
|
|
|
FirstByte = Desc[DescIndex];
|
|
|
|
if ((FirstByte & R2_MASK) == R2_PREFIX) {
|
|
|
|
//
|
|
// general prologue region header; need to process it first
|
|
//
|
|
|
|
DescIndex++;
|
|
SecondByte = Desc[DescIndex++];
|
|
MiscMask = ((FirstByte & 0x7) << 1) | ((SecondByte & 0x80) >> 7);
|
|
GrSave = SecondByte & 0x7F;
|
|
ReadLEB128(Desc, &DescIndex); // advance the descriptor index
|
|
|
|
if (GrSave < STATIC_REGISTER_SET_SIZE) {
|
|
UW_DEBUG((1, "Invalid unwind descriptor!\n"));
|
|
}
|
|
|
|
UW_DEBUG((1, "Region R2: rmask=%x,grsave=%d,length=%d\n",
|
|
MiscMask, GrSave, State->RegionLen));
|
|
|
|
Count = 0;
|
|
for (Index = REG_PREDS; Index <= REG_RP; Index++) {
|
|
if (MiscMask & 0x1) {
|
|
if (!(State->IsTarget) ||
|
|
(State->MiscMask & MASK(Index,1)))
|
|
{
|
|
UwContext->MiscRegs[Index].Where = GENERAL_REG;
|
|
UwContext->MiscRegs[Index].SaveOffset = GrSave+Count;
|
|
UwContext->MiscRegs[Index].When = 0;
|
|
State->MiscMask |= MASK(Index,1);
|
|
}
|
|
Count++;
|
|
}
|
|
MiscMask = MiscMask >> 1;
|
|
}
|
|
}
|
|
|
|
while (DescIndex <= State->DescEnd) {
|
|
|
|
FirstByte = Desc[DescIndex++];
|
|
|
|
if ( (FirstByte & P1_MASK) == P1_PREFIX) {
|
|
|
|
BrMask = FirstByte & ~P1_MASK;
|
|
State->MiscMask |= (BrMask << REG_BR_BASE);
|
|
|
|
UW_DEBUG((1, "Prolog P1: brmask=%x\n", BrMask));
|
|
|
|
for (Count = REG_BR_BASE;
|
|
Count < REG_BR_BASE+NUMBER_OF_PRESERVED_BR;
|
|
Count++)
|
|
{
|
|
if (BrMask & 0x1) {
|
|
UwContext->MiscRegs[Count].Where = PSP_RELATIVE;
|
|
UwContext->MiscRegs[Count].When = State->RegionLen;
|
|
}
|
|
BrMask = BrMask >> 1;
|
|
}
|
|
|
|
} else if ( (FirstByte & P2_MASK) == P2_PREFIX ) {
|
|
|
|
SecondByte = Desc[DescIndex++];
|
|
GrSave = SecondByte & 0x7F;
|
|
BrMask = ((FirstByte & ~P2_MASK) << 1) | ((SecondByte & 0x80) >> 7);
|
|
UW_DEBUG((1, "Prolog P2: brmask=%x reg base=%d\n", BrMask, GrSave));
|
|
|
|
State->MiscMask |= (BrMask << REG_BR_BASE);
|
|
|
|
for (Count = REG_BR_BASE;
|
|
Count < REG_BR_BASE+NUMBER_OF_PRESERVED_BR;
|
|
Count++)
|
|
{
|
|
if (BrMask & 0x1) {
|
|
UwContext->MiscRegs[Count].Where = GENERAL_REG;
|
|
UwContext->MiscRegs[Count].SaveOffset = GrSave++;
|
|
}
|
|
BrMask = BrMask >> 1;
|
|
}
|
|
|
|
} else if ( (FirstByte & P3_MASK) == P3_PREFIX ) {
|
|
|
|
SecondByte = Desc[DescIndex++];
|
|
RecType = ((SecondByte & 0x80) >> 7) | ((FirstByte & 0x7) << 1);
|
|
Index = P3RecordTypeToRegisterIndex[RecType];
|
|
|
|
if (RecType == RP_BR)
|
|
{
|
|
UwContext->AlternateRp = SecondByte & 0x7F;
|
|
}
|
|
else if (!(State->IsTarget) || (State->MiscMask & MASK(Index,1)))
|
|
{
|
|
UwContext->MiscRegs[Index].Where = GENERAL_REG;
|
|
UwContext->MiscRegs[Index].SaveOffset = SecondByte & 0x7F;
|
|
UwContext->MiscRegs[Index].When = 0;
|
|
State->MiscMask |= MASK(Index,1);
|
|
|
|
UW_DEBUG((1, "Prolog P3: type=%d reg=%d\n",
|
|
RecType, UwContext->MiscRegs[Index].SaveOffset));
|
|
}
|
|
|
|
} else if ( (FirstByte & P4_MASK) == P4_PREFIX ) {
|
|
|
|
SpillMaskOmitted = FALSE;
|
|
ImaskBegin = DescIndex;
|
|
DescIndex += ((State->RegionLen+3) >> 2);
|
|
|
|
} else if ( (FirstByte & P5_MASK) == P5_PREFIX ) {
|
|
|
|
GrMask = (Desc[DescIndex] & 0xF0) >> 4;
|
|
FrMask = ((ULONG)(Desc[DescIndex] & 0xF) << 16) |
|
|
((ULONG)Desc[DescIndex+1] << 8) |
|
|
((ULONG)Desc[DescIndex+2]);
|
|
|
|
DescIndex += 3; // increment the descriptor index
|
|
|
|
State->GrMask |= GrMask;
|
|
State->FrMask |= FrMask;
|
|
|
|
UW_DEBUG((1, "Prolog P5: grmask = %x, frmask = %x\n",
|
|
State->GrMask, State->FrMask));
|
|
|
|
} else if ( (FirstByte & P6_MASK) == P6_PREFIX ) {
|
|
|
|
if (FirstByte & 0x10) {
|
|
|
|
GrMask = FirstByte & 0xF;
|
|
State->GrMask |= GrMask;
|
|
|
|
} else {
|
|
|
|
FrMask = FirstByte & 0xF;
|
|
State->FrMask |= FrMask;
|
|
|
|
}
|
|
|
|
UW_DEBUG((1, "Prolog P6: is_gr = %d, mask = %x\n",
|
|
(FirstByte & 0x10) ? 1 : 0,
|
|
(FirstByte & 0x10) ? State->GrMask : State->FrMask));
|
|
|
|
} else if ( (FirstByte & P7_MASK) == P7_PREFIX ) {
|
|
|
|
RecType = FirstByte & ~P7_MASK;
|
|
|
|
switch (RecType) {
|
|
|
|
case PSP_SPREL:
|
|
|
|
//
|
|
// sp-relative location
|
|
//
|
|
|
|
Index = P7RecordTypeToRegisterIndex[RecType];
|
|
Offset = ReadLEB128(Desc, &DescIndex);
|
|
if (!(State->IsTarget) || (State->MiscMask & MASK(Index,1)))
|
|
{
|
|
UwContext->MiscRegs[Index].Where = SP_RELATIVE;
|
|
UwContext->MiscRegs[Index].SaveOffset = Offset;
|
|
if (!(State->MiscMask & MASK(Index,1))) {
|
|
UwContext->MiscRegs[Index].When = State->RegionLen;
|
|
State->MiscMask |= MASK(Index,1);
|
|
}
|
|
}
|
|
UW_DEBUG((1, "Prolog P7: type=%d spoff = %d\n", RecType, Offset));
|
|
break;
|
|
|
|
|
|
case RP_PSPREL:
|
|
case PFS_PSPREL:
|
|
case PREDS_PSPREL:
|
|
case LC_PSPREL:
|
|
case UNAT_PSPREL:
|
|
case FPSR_PSPREL:
|
|
|
|
//
|
|
// psp-relative location
|
|
//
|
|
|
|
Index = P7RecordTypeToRegisterIndex[RecType];
|
|
Offset = ReadLEB128(Desc, &DescIndex);
|
|
if (!(State->IsTarget) || (State->MiscMask & MASK(Index,1)))
|
|
{
|
|
UwContext->MiscRegs[Index].Where = PSP_RELATIVE;
|
|
UwContext->MiscRegs[Index].SaveOffset = Offset;
|
|
UwContext->MiscRegs[Index].When = 0;
|
|
State->MiscMask |= MASK(Index,1);
|
|
}
|
|
UW_DEBUG((1, "Prolog P7: type=%d pspoff= %d\n", RecType, Offset));
|
|
break;
|
|
|
|
case MEM_STACK_V:
|
|
case RP_WHEN:
|
|
case PFS_WHEN:
|
|
case PREDS_WHEN:
|
|
case LC_WHEN:
|
|
case UNAT_WHEN:
|
|
case FPSR_WHEN:
|
|
|
|
//
|
|
// Nevermind processing these descriptors because they
|
|
// have been taken care of in phase 0
|
|
//
|
|
|
|
Offset = ReadLEB128(Desc, &DescIndex);
|
|
break;
|
|
|
|
case MEM_STACK_F:
|
|
|
|
Offset = ReadLEB128(Desc, &DescIndex);
|
|
FrameSize = ReadLEB128(Desc, &DescIndex);
|
|
|
|
UW_DEBUG((1, "Prolog P7: type=%d Slot=%d FrameSize=%d\n",
|
|
RecType, Offset, FrameSize));
|
|
break;
|
|
|
|
case SPILL_BASE:
|
|
|
|
State->SpillBase = ReadLEB128(Desc, &DescIndex);
|
|
State->SpillPtr = State->SpillBase;
|
|
UW_DEBUG((1, "Prolog P7: type=%d, spillbase=%d\n",
|
|
RecType, State->SpillBase));
|
|
break;
|
|
|
|
default:
|
|
|
|
UW_DEBUG((1, "invalid unwind descriptors\n"));
|
|
|
|
}
|
|
|
|
} else if ( (FirstByte & P8_MASK) == P8_PREFIX ) {
|
|
|
|
RecType = Desc[DescIndex++];
|
|
|
|
switch (RecType) {
|
|
|
|
case PSP_PSPREL:
|
|
VUW_DEBUG_PRINT("Unsupported Unwind Descriptor!\n");
|
|
break;
|
|
|
|
case RP_SPREL:
|
|
case PFS_SPREL:
|
|
case PREDS_SPREL:
|
|
case LC_SPREL:
|
|
case UNAT_SPREL:
|
|
case FPSR_SPREL:
|
|
case BSP_SPREL:
|
|
case BSPSTORE_SPREL:
|
|
case RNAT_SPREL:
|
|
case PRIUNAT_SPREL:
|
|
|
|
//
|
|
// sp-relative location
|
|
//
|
|
|
|
Index = P8RecordTypeToRegisterIndex[RecType];
|
|
Offset = ReadLEB128(Desc, &DescIndex);
|
|
if (!(State->IsTarget) || (State->MiscMask & MASK(Index,1)))
|
|
{
|
|
UwContext->MiscRegs[Index].Where = SP_RELATIVE;
|
|
UwContext->MiscRegs[Index].SaveOffset = Offset;
|
|
if (!(State->MiscMask & MASK(Index,1))) {
|
|
UwContext->MiscRegs[Index].When=State->RegionLen;
|
|
State->MiscMask |= MASK(Index,1);
|
|
}
|
|
}
|
|
UW_DEBUG((1, "Prolog P8: type=%d spoff= %d\n", RecType, Offset));
|
|
break;
|
|
|
|
case BSP_PSPREL:
|
|
case BSPSTORE_PSPREL:
|
|
case RNAT_PSPREL:
|
|
case PRIUNAT_PSPREL:
|
|
|
|
//
|
|
// psp-relative location
|
|
//
|
|
|
|
Index = P8RecordTypeToRegisterIndex[RecType];
|
|
Offset = ReadLEB128(Desc, &DescIndex);
|
|
if (!(State->IsTarget) || (State->MiscMask & MASK(Index,1)))
|
|
{
|
|
UwContext->MiscRegs[Index].Where = PSP_RELATIVE;
|
|
UwContext->MiscRegs[Index].SaveOffset = Offset;
|
|
UwContext->MiscRegs[Index].When = 0;
|
|
State->MiscMask |= MASK(Index,1);
|
|
}
|
|
UW_DEBUG((1, "Prolog P8: type=%d pspoff= %d\n", RecType, Offset));
|
|
break;
|
|
|
|
case BSP_WHEN:
|
|
case BSPSTORE_WHEN:
|
|
case RNAT_WHEN:
|
|
case PRIUNAT_WHEN:
|
|
|
|
//
|
|
// Nevermind processing these descriptors because they
|
|
// have been taken care of in phase 0
|
|
//
|
|
|
|
Offset = ReadLEB128(Desc, &DescIndex);
|
|
break;
|
|
|
|
default:
|
|
|
|
UW_DEBUG((1, "Invalid record type for descriptor P8!\n"));
|
|
|
|
}
|
|
|
|
} else if ( (FirstByte & P9_MASK) == P9_PREFIX ) {
|
|
|
|
DescIndex += 2;
|
|
VUW_DEBUG_PRINT("Format P9 not supported yet!\n");
|
|
|
|
} else if ( (FirstByte & P10_MASK) == P10_PREFIX ) {
|
|
|
|
UCHAR Abi = Desc[DescIndex++];
|
|
UCHAR Context = Desc[DescIndex++];
|
|
|
|
} else {
|
|
|
|
UW_DEBUG((1, "Invalid descriptor!\n"));
|
|
|
|
}
|
|
}
|
|
|
|
GrMask = State->GrMask;
|
|
FrMask = State->FrMask;
|
|
BrMask = State->MiscMask >> REG_BR_BASE;
|
|
|
|
if (!(GrMask | FrMask | BrMask)) {
|
|
|
|
return;
|
|
|
|
} else if (SpillMaskOmitted && !(State->IsTarget)) {
|
|
|
|
//
|
|
// When spillmask is omitted, floating point registers, general
|
|
// registers, and then branch regisers are spilled in order.
|
|
// They are not modified in the prologue region; therefore, there
|
|
// is no need to restore their contents when the control ip is
|
|
// in this prologue region.
|
|
//
|
|
|
|
// 1. floating point registers
|
|
|
|
State->SpillPtr &= ~(SPILLSIZE_OF_FLOAT128_IN_DWORDS - 1);
|
|
NextFr = NUMBER_OF_PRESERVED_FR - 1;
|
|
while (FrMask & 0xFFFFF) {
|
|
if (FrMask & 0x80000) {
|
|
State->SpillPtr += SPILLSIZE_OF_FLOAT128_IN_DWORDS;
|
|
UwContext->Float[NextFr].SaveOffset = State->SpillPtr;
|
|
}
|
|
FrMask = FrMask << 1;
|
|
NextFr--;
|
|
}
|
|
|
|
// 2. branch registers
|
|
|
|
NextBr = REG_BR_BASE + NUMBER_OF_PRESERVED_BR - 1;
|
|
while (BrMask & 0x1F) {
|
|
if (BrMask & 0x10) {
|
|
if (UwContext->MiscRegs[NextBr].Where == PSP_RELATIVE) {
|
|
State->SpillPtr += SPILLSIZE_OF_ULONGLONG_IN_DWORDS;
|
|
UwContext->MiscRegs[NextBr].SaveOffset = State->SpillPtr;
|
|
}
|
|
}
|
|
BrMask = BrMask << 1;
|
|
NextBr--;
|
|
}
|
|
|
|
// 3. general registers
|
|
|
|
NextGr = NUMBER_OF_PRESERVED_GR - 1;
|
|
while (GrMask & 0xF) {
|
|
if (GrMask & 0x8) {
|
|
State->SpillPtr += SPILLSIZE_OF_ULONGLONG_IN_DWORDS;
|
|
UwContext->Integer[NextGr].SaveOffset = State->SpillPtr;
|
|
}
|
|
GrMask = GrMask << 1;
|
|
NextGr--;
|
|
}
|
|
|
|
} else if (SpillMaskOmitted && State->IsTarget) {
|
|
|
|
State->GrMask = 0;
|
|
State->FrMask = 0;
|
|
State->MiscMask &= MASK(REG_BR_BASE, 1) - 1;
|
|
|
|
} else if (SpillMaskOmitted == FALSE) {
|
|
|
|
ULONG Length;
|
|
|
|
if (State->IsTarget) {
|
|
|
|
//
|
|
// control ip is in the prologue region; clear the masks
|
|
// and then process the imask to determine which preserved
|
|
// Gr/Fr/Br have been saved and set the corresponding bits.
|
|
//
|
|
|
|
State->GrMask = 0;
|
|
State->FrMask = 0;
|
|
State->MiscMask &= MASK(REG_BR_BASE, 1) - 1;
|
|
Length = UwContext->TargetSlot - State->RegionBegin;
|
|
} else {
|
|
Length = State->RegionLen;
|
|
}
|
|
|
|
NextGr = NUMBER_OF_PRESERVED_GR - 1;
|
|
NextBr = NUMBER_OF_PRESERVED_BR - 1;
|
|
NextFr = NUMBER_OF_PRESERVED_FR - 1;
|
|
for (Count = 0; Count < Length; Count++) {
|
|
|
|
if ((Count % 4) == 0) {
|
|
FirstByte = Desc[ImaskBegin++];
|
|
} else {
|
|
FirstByte = FirstByte << 2;
|
|
}
|
|
|
|
switch (FirstByte & 0xC0) {
|
|
|
|
case 0x40: // 0x01 - save next fr
|
|
|
|
while ( !(FrMask & 0x80000) && (NextFr > 0) ) {
|
|
NextFr--;
|
|
FrMask = FrMask << 1;
|
|
}
|
|
|
|
UW_DEBUG((1, "spilled register FS%lx\n", (ULONG)NextFr));
|
|
|
|
State->FrMask |= MASK(NextFr,1);
|
|
UwContext->Float[NextFr].When = Count;
|
|
State->SpillPtr += SPILLSIZE_OF_ULONGLONG_IN_DWORDS;
|
|
State->SpillPtr &= ~(SPILLSIZE_OF_FLOAT128_IN_DWORDS - 1);
|
|
State->SpillPtr += SPILLSIZE_OF_FLOAT128_IN_DWORDS;
|
|
UwContext->Float[NextFr].SaveOffset = State->SpillPtr;
|
|
|
|
NextFr--;
|
|
FrMask = FrMask << 1;
|
|
break;
|
|
|
|
case 0x80: // 0x10 - save next gr
|
|
|
|
while ( !(GrMask & 0x8) && (NextGr > 0) ) {
|
|
NextGr--;
|
|
GrMask = GrMask << 1;
|
|
}
|
|
|
|
UW_DEBUG((1, "spilled register S%lx\n", (ULONG)NextGr));
|
|
|
|
State->GrMask |= MASK(NextGr,1);
|
|
UwContext->Integer[NextGr].When = Count;
|
|
State->SpillPtr += SPILLSIZE_OF_ULONGLONG_IN_DWORDS;
|
|
UwContext->Integer[NextGr].SaveOffset = State->SpillPtr;
|
|
|
|
NextGr--;
|
|
GrMask = GrMask << 1;
|
|
break;
|
|
|
|
case 0xC0: // 0x11 - save next br
|
|
|
|
while ( !(BrMask & 0x10) && (NextBr > 0) ) {
|
|
NextBr--;
|
|
BrMask = BrMask << 1;
|
|
}
|
|
|
|
UW_DEBUG((1, "spilled register BS%lx\n", (ULONG)NextBr));
|
|
|
|
Index = REG_BR_BASE + NextBr;
|
|
State->MiscMask |= MASK(Index,1);
|
|
UwContext->MiscRegs[Index].When = Count;
|
|
if (UwContext->MiscRegs[Index].Where == PSP_RELATIVE) {
|
|
State->SpillPtr += SPILLSIZE_OF_ULONGLONG_IN_DWORDS;
|
|
UwContext->MiscRegs[Index].SaveOffset = State->SpillPtr;
|
|
}
|
|
|
|
NextBr--;
|
|
BrMask = BrMask << 1;
|
|
break;
|
|
|
|
default: // 0x00 - save no register
|
|
break;
|
|
|
|
}
|
|
}
|
|
}
|
|
}
|