[PATCH] Fix "until LINE" in main, when "until" runs into longjmp
Andrew Burgess
aburgess@redhat.com
Wed Jul 13 10:46:10 GMT 2022
Pedro Alves <pedro@palves.net> writes:
> With a test like this:
>
> 1 #include <dlfcn.h>
> 2 int
> 3 main ()
> 4 {
> 5 dlsym (RTLD_DEFAULT, "FOO");
> 6 return 0;
> 7 }
>
> and then "start" followed by "until 6", GDB currently incorrectly
> stops inside the runtime loader, instead of line 6. Vis:
>
> ...
> Temporary breakpoint 1, main () at until.c:5
> 4 {
> (gdb) until 6
> 0x00007ffff7f0a90d in __GI__dl_catch_exception (exception=exception@entry=0x7fffffffdb00, operate=<optimized out>, args=0x7ffff7f0a90d <__GI__dl_catch_exception+109>) at dl-error-skeleton.c:206
> 206 dl-error-skeleton.c: No such file or directory.
> (gdb)
>
> The problem is related to longjmp handling -- dlsym internally
> longjmps on error. The testcase can be reduced to this:
>
> 1 #include <setjmp.h>
> 2 void func () {
> 3 jmp_buf buf;
> 4 if (setjmp (buf) == 0)
> 5 longjmp (buf, 1);
> 6 }
> 7
> 8 int main () {
> 9 func ();
> 10 return 0; /* until to here */
> 11 }
>
> and then with "start" followed by "until 10", GDB currently
> incorrectly stops at line 4 (returning from setjmp), instead of line
> 10.
>
> The problem is that the BPSTAT_WHAT_CLEAR_LONGJMP_RESUME code in
> infrun.c fails to find the initiating frame, and so infrun thinks that
> the longjmp jumped somewhere outer to "until"'s originating frame.
>
> Here:
>
> case BPSTAT_WHAT_CLEAR_LONGJMP_RESUME:
> {
> struct frame_info *init_frame;
>
> /* There are several cases to consider.
>
> 1. The initiating frame no longer exists. In this case we
> must stop, because the exception or longjmp has gone too
> far.
>
> ...
>
> init_frame = frame_find_by_id (ecs->event_thread->initiating_frame);
>
> if (init_frame) // this is NULL!
> {
> ...
> }
>
> /* For Cases 1 and 2, remove the step-resume breakpoint, if it
> exists. */
> delete_step_resume_breakpoint (ecs->event_thread);
>
> end_stepping_range (ecs); // case 1., so we stop.
> }
>
> The initiating frame is set by until_break_command ->
> set_longjmp_breakpoint. The initiating frame is supposed to be the
> frame that is selected when the command was issued, but
> until_break_command instead passes the frame id of the _caller_ frame
> by mistake. When the "until LINE" command is issued from main, the
> caller frame is the caller of main. When later infrun tries to find
> that frame by id, it fails to find it, because frame_find_by_id
> doesn't unwind past main.
>
> The bug is that we passed the caller frame's id to
> set_longjmp_breakpoint. We should have passed the selected frame's id
> instead.
LGTM.
Thanks,
Andrew
>
> Change-Id: Iaae1af7cdddf296b7c5af82c3b5b7d9b66755b1c
> ---
> gdb/breakpoint.c | 2 +-
> .../gdb.base/longjmp-until-in-main.c | 34 ++++++++++++++
> .../gdb.base/longjmp-until-in-main.exp | 44 +++++++++++++++++++
> 3 files changed, 79 insertions(+), 1 deletion(-)
> create mode 100644 gdb/testsuite/gdb.base/longjmp-until-in-main.c
> create mode 100644 gdb/testsuite/gdb.base/longjmp-until-in-main.exp
>
> diff --git a/gdb/breakpoint.c b/gdb/breakpoint.c
> index a3be12557f6..74f53368464 100644
> --- a/gdb/breakpoint.c
> +++ b/gdb/breakpoint.c
> @@ -10489,7 +10489,7 @@ until_break_command (const char *arg, int from_tty, int anywhere)
> caller_frame_id, bp_until);
> breakpoints.emplace_back (std::move (caller_breakpoint));
>
> - set_longjmp_breakpoint (tp, caller_frame_id);
> + set_longjmp_breakpoint (tp, stack_frame_id);
> lj_deleter.emplace (thread);
> }
>
> diff --git a/gdb/testsuite/gdb.base/longjmp-until-in-main.c b/gdb/testsuite/gdb.base/longjmp-until-in-main.c
> new file mode 100644
> index 00000000000..3b9ef945a34
> --- /dev/null
> +++ b/gdb/testsuite/gdb.base/longjmp-until-in-main.c
> @@ -0,0 +1,34 @@
> +/* This testcase is part of GDB, the GNU debugger.
> +
> + Copyright 2022 Free Software Foundation, Inc.
> +
> + This program is free software; you can redistribute it and/or modify
> + it under the terms of the GNU General Public License as published by
> + the Free Software Foundation; either version 3 of the License, or
> + (at your option) any later version.
> +
> + This program is distributed in the hope that it will be useful,
> + but WITHOUT ANY WARRANTY; without even the implied warranty of
> + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
> + GNU General Public License for more details.
> +
> + You should have received a copy of the GNU General Public License
> + along with this program. If not, see <http://www.gnu.org/licenses/>. */
> +
> +#include <setjmp.h>
> +
> +void
> +func ()
> +{
> + jmp_buf buf;
> +
> + if (setjmp (buf) == 0)
> + longjmp (buf, 1);
> +}
> +
> +int
> +main ()
> +{
> + func ();
> + return 0; /* until to here */
> +}
> diff --git a/gdb/testsuite/gdb.base/longjmp-until-in-main.exp b/gdb/testsuite/gdb.base/longjmp-until-in-main.exp
> new file mode 100644
> index 00000000000..33a907fb83e
> --- /dev/null
> +++ b/gdb/testsuite/gdb.base/longjmp-until-in-main.exp
> @@ -0,0 +1,44 @@
> +# Copyright 2008-2022 Free Software Foundation, Inc.
> +
> +# This program is free software; you can redistribute it and/or modify
> +# it under the terms of the GNU General Public License as published by
> +# the Free Software Foundation; either version 3 of the License, or
> +# (at your option) any later version.
> +#
> +# This program is distributed in the hope that it will be useful,
> +# but WITHOUT ANY WARRANTY; without even the implied warranty of
> +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
> +# GNU General Public License for more details.
> +#
> +# You should have received a copy of the GNU General Public License
> +# along with this program. If not, see <http://www.gnu.org/licenses/>.
> +
> +# Test "until LINE", started in the "main()" frame, where the until
> +# command runs into a longjmp that lands in a frame that is inner than
> +# main. GDB internally intercepts the longjmp, sets a breakpoint at
> +# the jump destination, and once there, decides whether to stop or
> +# ignore the breakpoint hit depending on whether the initiating frame
> +# is present on the frame chain. GDB used to have a bug where it
> +# recorded the frame of the caller of main instead of the frame of
> +# main as the initiating frame, and then later on when deciding
> +# whether the longjmp landed somewhere inner than main, since
> +# unwinding normally stops at main, GDB would fail to find the
> +# initiating frame.
> +
> +standard_testfile
> +
> +if {[prepare_for_testing "failed to prepare" ${testfile} ${srcfile}]} {
> + return
> +}
> +
> +if {![runto_main]} {
> + return
> +}
> +
> +delete_breakpoints
> +
> +set until_to_line [gdb_get_line_number "until to here"]
> +
> +gdb_test "until $until_to_line" \
> + " until to here .*" \
> + "until \$line, in main"
>
> base-commit: dd4c046506cd4da46b439a2b4f8b6d933ecbb961
> --
> 2.36.0
More information about the Gdb-patches
mailing list