diff options
author | Wei-cheng Wang <cole945@gmail.com> | 2015-01-17 14:30:59 +0800 |
---|---|---|
committer | Cole Wang <cole945@gmail.com> | 2015-01-17 19:48:22 +0800 |
commit | cf90fd9a07e8998540bf74f293d348a6653ac120 (patch) | |
tree | 21337cfdef7297895e07abba7a54aa0c6a0c25a9 /gdb/ppc-linux-tdep.c | |
parent | b4cdae6fe51e532e0b1069c6960b14a610182d14 (diff) | |
download | binutils-gdb-cf90fd9a07e8998540bf74f293d348a6653ac120.tar.gz |
Skip-trampoline for PowerPC reverse-stepping.
Diffstat (limited to 'gdb/ppc-linux-tdep.c')
-rw-r--r-- | gdb/ppc-linux-tdep.c | 58 |
1 files changed, 39 insertions, 19 deletions
diff --git a/gdb/ppc-linux-tdep.c b/gdb/ppc-linux-tdep.c index dcfd3bb9f97..13bb479fe3a 100644 --- a/gdb/ppc-linux-tdep.c +++ b/gdb/ppc-linux-tdep.c @@ -51,6 +51,7 @@ #include "linux-tdep.h" #include "linux-record.h" #include "record-full.h" +#include "infrun.h" #include "stap-probe.h" #include "ax.h" @@ -314,31 +315,50 @@ ppc_skip_trampoline_code (struct frame_info *frame, CORE_ADDR pc) struct gdbarch_tdep *tdep = gdbarch_tdep (gdbarch); enum bfd_endian byte_order = gdbarch_byte_order (gdbarch); CORE_ADDR target = 0; + int scan_limit, i; - if (ppc_insns_match_pattern (frame, pc, powerpc32_plt_stub, insnbuf)) - { - /* Insn pattern is - lis r11, xxxx - lwz r11, xxxx(r11) - Branch target is in r11. */ - - target = (ppc_insn_d_field (insnbuf[0]) << 16) - | ppc_insn_d_field (insnbuf[1]); - target = read_memory_unsigned_integer (target, 4, byte_order); - } + scan_limit = 1; + /* When reverse-debugging, scan backward to check whether we are + in the middle of trampoline code. */ + if (execution_direction == EXEC_REVERSE) + scan_limit = 4; /* At more 4 instructions. */ - if (ppc_insns_match_pattern (frame, pc, powerpc32_plt_stub_so, insnbuf)) + for (i = 0; i < scan_limit; i++) { - /* Insn pattern is - lwz r11, xxxx(r30) - Branch target is in r11. */ + if (ppc_insns_match_pattern (frame, pc, powerpc32_plt_stub, insnbuf)) + { + /* Insn pattern is + lis r11, xxxx + lwz r11, xxxx(r11) + Branch target is in r11. */ + + target = (ppc_insn_d_field (insnbuf[0]) << 16) + | ppc_insn_d_field (insnbuf[1]); + target = read_memory_unsigned_integer (target, 4, byte_order); + } + else if (ppc_insns_match_pattern (frame, pc, powerpc32_plt_stub_so, + insnbuf)) + { + /* Insn pattern is + lwz r11, xxxx(r30) + Branch target is in r11. */ + + target = get_frame_register_unsigned (frame, + tdep->ppc_gp0_regnum + 30) + + ppc_insn_d_field (insnbuf[0]); + target = read_memory_unsigned_integer (target, 4, byte_order); + } + else + { + /* Scan backward one more instructions if doesn't match. */ + pc -= 4; + continue; + } - target = get_frame_register_unsigned (frame, tdep->ppc_gp0_regnum + 30) - + ppc_insn_d_field (insnbuf[0]); - target = read_memory_unsigned_integer (target, 4, byte_order); + return target; } - return target; + return 0; } /* Wrappers to handle Linux-only registers. */ |