gdb: Implement the init_reg dwarf2_frame_ops for amd64
Andrew Burgess
aburgess@redhat.com
Mon Dec 6 15:13:19 GMT 2021
* Pedro Alves <pedro@palves.net> [2021-12-03 17:45:04 +0000]:
> On 2021-12-02 15:56, Andrew Burgess via Gdb-patches wrote:
> > * Djordje Todorovic <Djordje.Todorovic@syrmia.com> [2021-11-18 12:42:32 +0000]:
> >
> >> Hi Andrew,
> >>
> >> Please find the patch, rebased on top of your improvement.
> >>
> >> Thanks,
> >> Djordje
> >>
> >> ---
> >> gdb/amd64-tdep.c | 37 +++++++++++++
> >> gdb/frame.c | 9 +++-
> >> .../gdb.arch/amd64-invalid-stack-middle.exp | 8 +--
> >> gdb/testsuite/gdb.arch/amd64-not-saved-regs.c | 28 ++++++++++
> >> .../gdb.arch/amd64-not-saved-regs.exp | 52 +++++++++++++++++++
> >> gdb/testsuite/gdb.dwarf2/dw2-op-out-param.exp | 4 +-
> >> .../gdb.dwarf2/dw2-reg-undefined.exp | 8 +--
> >> gdb/testsuite/gdb.mi/mi-reg-undefined.exp | 4 +-
> >> 8 files changed, 136 insertions(+), 14 deletions(-)
> >> create mode 100644 gdb/testsuite/gdb.arch/amd64-not-saved-regs.c
> >> create mode 100644 gdb/testsuite/gdb.arch/amd64-not-saved-regs.exp
> >> ?
> >> diff --git a/gdb/amd64-tdep.c b/gdb/amd64-tdep.c
> >> index 7c67359678b..a6b544145fb 100644
> >> --- a/gdb/amd64-tdep.c
> >> +++ b/gdb/amd64-tdep.c
> >> @@ -25,6 +25,8 @@
> >> #include "arch-utils.h"
> >> #include "block.h"
> >> #include "dummy-frame.h"
> >> +#include "dwarf2.h"
> >> +#include "dwarf2/frame.h"
> >> #include "frame.h"
> >> #include "frame-base.h"
> >> #include "frame-unwind.h"
> >> @@ -3103,6 +3105,38 @@ amd64_in_indirect_branch_thunk (struct gdbarch *gdbarch, CORE_ADDR pc)
> >> AMD64_RIP_REGNUM);
> >> }
> >>
> >> +/* Implement the "init_reg" dwarf2_frame_ops method. */
> >> +
> >> +static void
> >> +amd64_dwarf2_frame_init_reg (struct gdbarch *gdbarch, int regnum,
> >> + struct dwarf2_frame_state_reg *reg,
> >> + struct frame_info *this_frame)
> >> +{
> >> + switch (regnum)
> >> + {
> >> + case AMD64_RIP_REGNUM:
> >> + reg->how = DWARF2_FRAME_REG_FN;
> >> + return;
> >> +
> >> + case AMD64_RSP_REGNUM:
> >> + reg->how = DWARF2_FRAME_REG_CFA;
> >> + return;
> >> +
> >> + /* Caller-saved registers. */
> >> + case AMD64_RAX_REGNUM:
> >> + case AMD64_RDI_REGNUM:
> >> + case AMD64_RSI_REGNUM:
> >> + case AMD64_RDX_REGNUM:
> >> + case AMD64_RCX_REGNUM:
> >> + case AMD64_R8_REGNUM:
> >> + case AMD64_R9_REGNUM:
> >> + case AMD64_R10_REGNUM:
> >> + case AMD64_R11_REGNUM:
> >> + reg->how = DWARF2_FRAME_REG_UNDEFINED;
> >> + return;
> >> + }
> >> +}
> >
> > I believe that this is the System-V ABI, which is not the only ABI
> > used on x86-64. From what I've read I believe that Windows uses a
> > slightly different ABI, where $rsi and $rdi are not callee saved.
> >
> > I think that we might consider moving this function from the general
> > amd64-tdep.c to something like amd64-linux-tdep.c, and register this
> > in the function amd64_linux_init_abi.
> >
> > Of course, this will mean that other System-V like targets would miss
> > out for now, but maybe the function itself could be renamed to
> > something like amd64_system_v_dwarf2_frame_init_reg, and left in
> > amd64-tdep.c, then from amd64-linux-tdep.c we can register that
> > function. And in future, other system-v like targets can also
> > register the function.
> >
> >> +
> >> void
> >> amd64_init_abi (struct gdbarch_info info, struct gdbarch *gdbarch,
> >> const target_desc *default_tdesc)
> >> @@ -3217,6 +3251,9 @@ amd64_init_abi (struct gdbarch_info info, struct gdbarch *gdbarch,
> >> set_gdbarch_stab_reg_to_regnum (gdbarch, amd64_dwarf_reg_to_regnum);
> >> set_gdbarch_dwarf2_reg_to_regnum (gdbarch, amd64_dwarf_reg_to_regnum);
> >>
> >> + /* Frame handling. */
> >> + dwarf2_frame_set_init_reg (gdbarch, amd64_dwarf2_frame_init_reg);
> >> +
> >> /* We don't override SDB_REG_RO_REGNUM, since COFF doesn't seem to
> >> be in use on any of the supported AMD64 targets. */
> >>
> >> diff --git a/gdb/frame.c b/gdb/frame.c
> >> index 2a899fc494f..da12ed36e02 100644
> >> --- a/gdb/frame.c
> >> +++ b/gdb/frame.c
> >> @@ -2315,9 +2315,14 @@ get_prev_frame_always (struct frame_info *this_frame)
> >> }
> >> catch (const gdb_exception_error &ex)
> >> {
> >> - if (ex.error == MEMORY_ERROR)
> >> + if (ex.error == MEMORY_ERROR || ex.error == OPTIMIZED_OUT_ERROR)
> >> {
> >> - this_frame->stop_reason = UNWIND_MEMORY_ERROR;
> >> + if (ex.error == MEMORY_ERROR)
> >> + this_frame->stop_reason = UNWIND_MEMORY_ERROR;
> >> + else
> >> + /* This is for the OPTIMIZED_OUT_ERROR case. */
> >> + this_frame->stop_reason = UNWIND_UNAVAILABLE;
>
> Hmm? Nak on this part, it makes no sense to me, but I may be missing something.
> Optimized out should never be converted to UNWIND_UNAVAILABLE.
I had a misunderstanding about the meaning on unavailable within GDB
(see below), but given that, maybe we need another unwind stop reason,
UNWIND_OPTIMIZED_OUT? For when the information required to unwind has
been optimised away?
>
> The patch is missing a commit log describing it / providing a rationale,
> so I'm kind of lost here. Could you add that, please?
>
> >> +
> >> if (ex.message != NULL)
> >> {
> >> char *stop_string;
> >> diff --git a/gdb/testsuite/gdb.arch/amd64-invalid-stack-middle.exp b/gdb/testsuite/gdb.arch/amd64-invalid-stack-middle.exp
> >> index f9e590d83bb..6ae1c9c1322 100644
> >> --- a/gdb/testsuite/gdb.arch/amd64-invalid-stack-middle.exp
> >> +++ b/gdb/testsuite/gdb.arch/amd64-invalid-stack-middle.exp
> >> @@ -42,11 +42,11 @@ if ![runto breakpt] {
> >> return -1
> >> }
> >>
> >> -gdb_test "bt" "^bt\r\n#0 +breakpt *\\(\\) \[^\r\n\]*\r\n#1 +0x\[0-9a-f\]+ in func5\[^\r\n\]*\r\n#2 +0x\[0-9a-f\]+ in func4\[^\r\n\]*\r\n#3 +0x\[0-9a-f\]+ in func3\[^\r\n\]*\r\nBacktrace stopped: Cannot access memory at address 0x\[0-9a-f\]+" \
> >> - "first backtrace, with error message"
> >> +gdb_test "bt" "^bt\r\n#0 +breakpt *\\(\\) \[^\r\n\]*\r\n#1 +0x\[0-9a-f\]+ in func5\[^\r\n\]*\r\n#2 +0x\[0-9a-f\]+ in func4\[^\r\n\]*\r\n#3 +0x\[0-9a-f\]+ in func3\[^\r\n\]*\r\nBacktrace stopped: value has been optimized out" \
> >> + "first backtrace, with error message"
> >
> > I think the new "value has been optimized out" message is better than
> > what we had previously ("Cannot access memory at address 0x....")
> > which I guess was caused by trying to access through an unavailable
> > register.
> >
> > That said, I remember a long time ago there was a whole big discussion
> > about whether registers that were not saved should be described as
> > "optimized out" or not. In the end we added the "<not saved>" message
> > to cover this case.
>
> I don't recall a big discussion about it, but yes, an optimized out
> register is presented as <not saved>, as that's what it means.
>
> >
> > So I wonder if we should instead be ending the backtrace with
> > something like: "Backtrace stopped: value is not available"? I took
> > that exact text from require_available (value.c) but anything that
> > talks about available rather than optimized out might be better.
>
> I would rather this said "register not saved" or something
> around it. Maybe even mention the register name. That's because "unavailable"
> has a different meaning from optimized out / not saved -- <unavailable> is for
> when e.g., we're looking at a traceframe, and some register/memory we need to
> display the value wasn't collected, it's not in the traceframe. Or a
> similar thing with a trimmed corefile. Or e.g., the register exists in the machine,
> but the kernel is running an older kernel missing a PTRACE op to get at the register.
> I.e., it's for when the value actually exists, but we don't have a way to get at it.
>
> >
> > Though I don't know how hard it would be to achieve that result. I
> > guess we'd need to start in dwarf2_frame_prev_register where we map
> > DWARF2_FRAME_REG_UNDEFINED onto optimized_out - maybe that needs to
> > map to an unavailable value instead, but that feels like it might have
> > huge knock on consequences ... we don't want optimized out variables
> > to suddenly start reporting themselves as unavailable instead...
>
> Please don't.
No, that would be the wrong thing to do given the intended use of
unavailable. I knew unavailable was used for traceframes, but thought
that it was also used for registers that hadn't been saved between
frames. Given the explanation you gave above then using optimized out
for unsaved registers is what we expect.
Thanks for correcting me on this.
Andrew
More information about the Gdb-patches
mailing list