mirror of
https://github.com/tobast/libunwind-eh_elf.git
synced 2024-12-23 20:03:43 +01:00
(linux_interrupt): Add "marker" argument. Support both old and new
Linux kernel "struct pt_regs" layout. (update_frame_state): Handle ABI_MARKER_OLD_LINUX_SIGTRAMP and ABI_MARKRE_OLD_LINUX_INTERRUPT. Cast values of type unw_word_t to (unsigned long) before printing them. (Logical change 1.91)
This commit is contained in:
parent
3053ee9de1
commit
f3b8566331
1 changed files with 13 additions and 5 deletions
|
@ -60,7 +60,7 @@ linux_sigtramp (struct cursor *c, unw_word_t *num_regsp)
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline int
|
static inline int
|
||||||
linux_interrupt (struct cursor *c, unw_word_t *num_regsp)
|
linux_interrupt (struct cursor *c, unw_word_t *num_regsp, int marker)
|
||||||
{
|
{
|
||||||
#if defined(UNW_LOCAL_ONLY)
|
#if defined(UNW_LOCAL_ONLY)
|
||||||
/* Perhaps libunwind will some day become the Linux kernel unwinder.
|
/* Perhaps libunwind will some day become the Linux kernel unwinder.
|
||||||
|
@ -69,6 +69,7 @@ linux_interrupt (struct cursor *c, unw_word_t *num_regsp)
|
||||||
return -UNW_EINVAL;
|
return -UNW_EINVAL;
|
||||||
#else
|
#else
|
||||||
unw_word_t sc_addr, num_regs;
|
unw_word_t sc_addr, num_regs;
|
||||||
|
ia64_loc_t pfs_loc;
|
||||||
|
|
||||||
sc_addr = c->sigcontext_addr = c->sp + 0x10;
|
sc_addr = c->sigcontext_addr = c->sp + 0x10;
|
||||||
|
|
||||||
|
@ -78,7 +79,11 @@ linux_interrupt (struct cursor *c, unw_word_t *num_regsp)
|
||||||
num_regs = 0;
|
num_regs = 0;
|
||||||
|
|
||||||
/* do what can't be described by unwind directives: */
|
/* do what can't be described by unwind directives: */
|
||||||
c->loc[IA64_REG_PFS] = IA64_LOC_ADDR (sc_addr + LINUX_PT_PFS_OFF, 0);
|
if (marker == ABI_MARKER_OLD_LINUX_INTERRUPT)
|
||||||
|
pfs_loc = IA64_LOC_ADDR (sc_addr + LINUX_OLD_PT_PFS_OFF, 0);
|
||||||
|
else
|
||||||
|
pfs_loc = IA64_LOC_ADDR (sc_addr + LINUX_PT_PFS_OFF, 0);
|
||||||
|
c->loc[IA64_REG_PFS] = pfs_loc;
|
||||||
*num_regsp = num_regs; /* size of frame */
|
*num_regsp = num_regs; /* size of frame */
|
||||||
return 0;
|
return 0;
|
||||||
#endif
|
#endif
|
||||||
|
@ -174,7 +179,8 @@ check_rbs_switch (struct cursor *c)
|
||||||
|| (ret = ia64_get (c, c->loc[IA64_REG_BSPSTORE], &saved_bspstore)))
|
|| (ret = ia64_get (c, c->loc[IA64_REG_BSPSTORE], &saved_bspstore)))
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
else if (c->abi_marker == ABI_MARKER_LINUX_SIGTRAMP
|
else if ((c->abi_marker == ABI_MARKER_LINUX_SIGTRAMP
|
||||||
|
|| c->abi_marker == ABI_MARKER_OLD_LINUX_SIGTRAMP)
|
||||||
&& !IA64_IS_REG_LOC (c->loc[IA64_REG_BSP])
|
&& !IA64_IS_REG_LOC (c->loc[IA64_REG_BSP])
|
||||||
&& (IA64_GET_ADDR (c->loc[IA64_REG_BSP])
|
&& (IA64_GET_ADDR (c->loc[IA64_REG_BSP])
|
||||||
== c->sigcontext_addr + LINUX_SC_AR_BSP_OFF))
|
== c->sigcontext_addr + LINUX_SC_AR_BSP_OFF))
|
||||||
|
@ -250,14 +256,16 @@ update_frame_state (struct cursor *c)
|
||||||
switch (c->abi_marker)
|
switch (c->abi_marker)
|
||||||
{
|
{
|
||||||
case ABI_MARKER_LINUX_SIGTRAMP:
|
case ABI_MARKER_LINUX_SIGTRAMP:
|
||||||
|
case ABI_MARKER_OLD_LINUX_SIGTRAMP:
|
||||||
c->as->abi = ABI_LINUX;
|
c->as->abi = ABI_LINUX;
|
||||||
if ((ret = linux_sigtramp (c, &num_regs)) < 0)
|
if ((ret = linux_sigtramp (c, &num_regs)) < 0)
|
||||||
return ret;
|
return ret;
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
case ABI_MARKER_OLD_LINUX_INTERRUPT:
|
||||||
case ABI_MARKER_LINUX_INTERRUPT:
|
case ABI_MARKER_LINUX_INTERRUPT:
|
||||||
c->as->abi = ABI_LINUX;
|
c->as->abi = ABI_LINUX;
|
||||||
if ((ret = linux_interrupt (c, &num_regs)) < 0)
|
if ((ret = linux_interrupt (c, &num_regs, c->abi_marker)) < 0)
|
||||||
return ret;
|
return ret;
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
@ -273,7 +281,7 @@ update_frame_state (struct cursor *c)
|
||||||
return -UNW_EINVAL;
|
return -UNW_EINVAL;
|
||||||
}
|
}
|
||||||
debug (100, "%s: sigcontext_addr=%lx (ret=%d)\n",
|
debug (100, "%s: sigcontext_addr=%lx (ret=%d)\n",
|
||||||
__FUNCTION__, c->sigcontext_addr, ret);
|
__FUNCTION__, (unsigned long) c->sigcontext_addr, ret);
|
||||||
|
|
||||||
c->sigcontext_off = c->sigcontext_addr - c->sp;
|
c->sigcontext_off = c->sigcontext_addr - c->sp;
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue