diff options
-rw-r--r-- | gdb/ChangeLog | 6 | ||||
-rw-r--r-- | gdb/doc/ChangeLog | 7 | ||||
-rw-r--r-- | gdb/doc/gdb.texinfo | 67 | ||||
-rw-r--r-- | gdb/infrun.c | 7 | ||||
-rw-r--r-- | gdb/testsuite/ChangeLog | 11 | ||||
-rw-r--r-- | gdb/testsuite/gdb.base/sigstep.c | 7 | ||||
-rw-r--r-- | gdb/testsuite/gdb.base/sigstep.exp | 55 |
7 files changed, 141 insertions, 19 deletions
diff --git a/gdb/ChangeLog b/gdb/ChangeLog index 32f788c..8a34118 100644 --- a/gdb/ChangeLog +++ b/gdb/ChangeLog @@ -1,3 +1,9 @@ +2014-10-27 Pedro Alves <palves@redhat.com> + + * infrun.c (handle_signal_stop): Also skip handlers when a random + signal arrives while handling a "stepi" or a "nexti". Set the + thread's 'step_after_step_resume_breakpoint' flag. + 2014-10-27 Luis Machado <lgustavo@codesourcery.com> * arm-tdep.c (INSN_S_L_BIT_NUM): Document. diff --git a/gdb/doc/ChangeLog b/gdb/doc/ChangeLog index f1b2329..f60fd8f 100644 --- a/gdb/doc/ChangeLog +++ b/gdb/doc/ChangeLog @@ -1,3 +1,10 @@ +2014-10-27 Pedro Alves <palves@redhat.com> + + * gdb.texinfo (Continuing and Stepping): Add cross reference to + info on stepping and signal handlers. + (Signals): Explain stepping and signal handlers. Add context + index entry, and cross references. + 2014-10-20 Simon Marchi <simon.marchi@ericsson.com> * python.texi (Breakpoints In Python): Add parenthesis after diff --git a/gdb/doc/gdb.texinfo b/gdb/doc/gdb.texinfo index a1b8ac7..15c2908 100644 --- a/gdb/doc/gdb.texinfo +++ b/gdb/doc/gdb.texinfo @@ -5079,7 +5079,9 @@ line of source code, or one machine instruction (depending on what particular command you use). Either when continuing or when stepping, your program may stop even sooner, due to a breakpoint or a signal. (If it stops due to a signal, you may want to use @code{handle}, or use -@samp{signal 0} to resume execution. @xref{Signals, ,Signals}.) +@samp{signal 0} to resume execution (@pxref{Signals, ,Signals}), +or you may step into the signal's handler (@pxref{stepping and signal +handlers}).) @table @code @kindex continue @@ -5573,6 +5575,66 @@ a result of the fatal signal once it saw the signal. To prevent this, you can continue with @samp{signal 0}. @xref{Signaling, ,Giving your Program a Signal}. +@cindex stepping and signal handlers +@anchor{stepping and signal handlers} + +@value{GDBN} optimizes for stepping the mainline code. If a signal +that has @code{handle nostop} and @code{handle pass} set arrives while +a stepping command (e.g., @code{stepi}, @code{step}, @code{next}) is +in progress, @value{GDBN} lets the signal handler run and then resumes +stepping the mainline code once the signal handler returns. In other +words, @value{GDBN} steps over the signal handler. This prevents +signals that you've specified as not interesting (with @code{handle +nostop}) from changing the focus of debugging unexpectedly. Note that +the signal handler itself may still hit a breakpoint, stop for another +signal that has @code{handle stop} in effect, or for any other event +that normally results in stopping the stepping command sooner. Also +note that @value{GDBN} still informs you that the program received a +signal if @code{handle print} is set. + +@anchor{stepping into signal handlers} + +If you set @code{handle pass} for a signal, and your program sets up a +handler for it, then issuing a stepping command, such as @code{step} +or @code{stepi}, when your program is stopped due to the signal will +step @emph{into} the signal handler (if the target supports that). + +Likewise, if you use the @code{queue-signal} command to queue a signal +to be delivered to the current thread when execution of the thread +resumes (@pxref{Signaling, ,Giving your Program a Signal}), then a +stepping command will step into the signal handler. + +Here's an example, using @code{stepi} to step to the first instruction +of @code{SIGUSR1}'s handler: + +@smallexample +(@value{GDBP}) handle SIGUSR1 +Signal Stop Print Pass to program Description +SIGUSR1 Yes Yes Yes User defined signal 1 +(@value{GDBP}) c +Continuing. + +Program received signal SIGUSR1, User defined signal 1. +main () sigusr1.c:28 +28 p = 0; +(@value{GDBP}) si +sigusr1_handler () at sigusr1.c:9 +9 @{ +@end smallexample + +The same, but using @code{queue-signal} instead of waiting for the +program to receive the signal first: + +@smallexample +(@value{GDBP}) n +28 p = 0; +(@value{GDBP}) queue-signal SIGUSR1 +(@value{GDBP}) si +sigusr1_handler () at sigusr1.c:9 +9 @{ +(@value{GDBP}) +@end smallexample + @cindex extra signal information @anchor{extra signal information} @@ -16654,6 +16716,9 @@ be used to pass a signal whose handling state has been set to @code{nopass} @end table @c @end group +@xref{stepping into signal handlers}, for information on how stepping +commands behave when the thread has a signal queued. + @node Returning @section Returning from a Function diff --git a/gdb/infrun.c b/gdb/infrun.c index 90a3123..df053e2 100644 --- a/gdb/infrun.c +++ b/gdb/infrun.c @@ -4463,9 +4463,9 @@ handle_signal_stop (struct execution_control_state *ecs) return; } - if (ecs->event_thread->control.step_range_end != 0 - && ecs->event_thread->suspend.stop_signal != GDB_SIGNAL_0 - && pc_in_thread_step_range (stop_pc, ecs->event_thread) + if (ecs->event_thread->suspend.stop_signal != GDB_SIGNAL_0 + && (pc_in_thread_step_range (stop_pc, ecs->event_thread) + || ecs->event_thread->control.step_range_end == 1) && frame_id_eq (get_stack_frame_id (frame), ecs->event_thread->control.step_stack_frame_id) && ecs->event_thread->control.step_resume_breakpoint == NULL) @@ -4485,6 +4485,7 @@ handle_signal_stop (struct execution_control_state *ecs) "single-step range\n"); insert_hp_step_resume_breakpoint_at_frame (frame); + ecs->event_thread->step_after_step_resume_breakpoint = 1; /* Reset trap_expected to ensure breakpoints are re-inserted. */ ecs->event_thread->control.trap_expected = 0; keep_going (ecs); diff --git a/gdb/testsuite/ChangeLog b/gdb/testsuite/ChangeLog index 2eea791..5cc7b0b 100644 --- a/gdb/testsuite/ChangeLog +++ b/gdb/testsuite/ChangeLog @@ -1,3 +1,14 @@ +2014-10-27 Pedro Alves <palves@redhat.com> + + * gdb.base/sigstep.c (dummy): New global. + (main): Issue a couple writes to the new global. + * gdb.base/sigstep.exp (get_next_pc, test_skip_handler): New + procedures. + (skip_over_handler): Use test_skip_handler. + (top level): Call skip_over_handler for stepi and nexti too. + (breakpoint_over_handler): Use test_skip_handler. + (top level): Call breakpoint_over_handler for stepi and nexti too. + 2014-10-27 Yao Qi <yao@codesourcery.com> * gdb.trace/tfile.c (adjust_function_address) diff --git a/gdb/testsuite/gdb.base/sigstep.c b/gdb/testsuite/gdb.base/sigstep.c index aa2384a..cc69184 100644 --- a/gdb/testsuite/gdb.base/sigstep.c +++ b/gdb/testsuite/gdb.base/sigstep.c @@ -24,6 +24,7 @@ #include <errno.h> static volatile int done; +static volatile int dummy; static void handler (int sig) @@ -74,8 +75,10 @@ main () return 1; } } - /* Wait. */ - while (!done); + /* Wait. Issue a couple writes to a dummy volatile var to be + reasonably sure our simple "get-next-pc" logic doesn't + stumble on branches. */ + dummy = 0; dummy = 0; while (!done); done = 0; } return 0; diff --git a/gdb/testsuite/gdb.base/sigstep.exp b/gdb/testsuite/gdb.base/sigstep.exp index 184d46e..53152b8 100644 --- a/gdb/testsuite/gdb.base/sigstep.exp +++ b/gdb/testsuite/gdb.base/sigstep.exp @@ -269,9 +269,38 @@ proc skip_to_handler_entry { i } { gdb_test "clear *handler" ".*" "$prefix; clear handler" } -skip_to_handler_entry step -skip_to_handler_entry next -skip_to_handler_entry continue +foreach cmd {"stepi" "nexti" "step" "next" "continue"} { + skip_to_handler_entry $cmd +} + +# Get the address of where a single-step should land. + +proc get_next_pc {test} { + global gdb_prompt + global hex + + set next "" + gdb_test_multiple "x/2i \$pc" $test { + -re "$hex .*:\[^\r\n\]+\r\n\[ \]+($hex).*\.\r\n$gdb_prompt $" { + set next $expect_out(1,string) + pass $test + } + } + + return $next +} + +# Test that the command skipped over the handler. + +proc test_skip_handler {prefix i} { + if {$i == "stepi" || $i == "nexti"} { + set next_pc [get_next_pc "$prefix; get next PC"] + gdb_test "$i" "dummy = 0.*" "$prefix; performing $i" + gdb_test "p /x \$pc" " = $next_pc" "$prefix; advanced" + } else { + gdb_test "$i" "done = 0.*" "$prefix; performing $i" + } +} # Try stepping when there's a signal pending but no breakpoints. # Should skip the handler advancing to the next line. @@ -295,13 +324,13 @@ proc skip_over_handler { i } { # Make the signal pending sleep 1 - - gdb_test "$i" "done = 0.*" "$prefix; performing $i" + + test_skip_handler $prefix $i } -skip_over_handler step -skip_over_handler next -skip_over_handler continue +foreach cmd {"stepi" "nexti" "step" "next" "continue"} { + skip_over_handler $cmd +} # Try stepping when there's a signal pending, a pre-existing # breakpoint at the current instruction, and a breakpoint in the @@ -385,7 +414,7 @@ breakpoint_to_handler_entry continue # Try stepping when there's a signal pending, and a pre-existing # breakpoint at the current instruction, and no breakpoint in the -# handler. Should advance to the next line. +# handler. Should advance to the next line/instruction. proc breakpoint_over_handler { i } { global gdb_prompt @@ -409,10 +438,10 @@ proc breakpoint_over_handler { i } { # Make the signal pending sleep 1 - gdb_test "$i" "done = 0.*" "$prefix; performing $i" + test_skip_handler $prefix $i gdb_test "clear $infinite_loop" ".*" "$prefix; clear infinite loop" } -breakpoint_over_handler step -breakpoint_over_handler next -breakpoint_over_handler continue +foreach cmd {"stepi" "nexti" "step" "next" "continue"} { + breakpoint_over_handler $cmd +} |