diff options
author | Pedro Alves <palves@redhat.com> | 2014-10-28 13:42:11 +0000 |
---|---|---|
committer | Pedro Alves <palves@redhat.com> | 2014-10-28 16:00:06 +0000 |
commit | 7f5ef60532b466ec7a83a943f36e93e32e30eafe (patch) | |
tree | 9608af6276beaeb374817057404649bf0b3ab5aa /gdb/infrun.c | |
parent | abbdbd03db7eea82cadbb418da733991cba91b15 (diff) | |
download | gdb-7f5ef60532b466ec7a83a943f36e93e32e30eafe.zip gdb-7f5ef60532b466ec7a83a943f36e93e32e30eafe.tar.gz gdb-7f5ef60532b466ec7a83a943f36e93e32e30eafe.tar.bz2 |
PR gdb/12623: non-stop crashes inferior, PC adjustment and 1-byte insns
TL;DR - if we step an instruction that is as long as
decr_pc_after_break (1-byte on x86) right after removing the
breakpoint at PC, in non-stop mode, adjust_pc_after_break adjusts the
PC, but it shouldn't.
In non-stop mode, when a breakpoint is removed, it is moved to the
"moribund locations" list. This is because other threads that are
running may have tripped on that breakpoint as well, and we haven't
heard about it. When a trap is reported, we check if perhaps it was
such a deleted breakpoint that caused the trap. If so, we also need
to adjust the PC (decr_pc_after_break).
Now, say that, on x86:
- a breakpoint was placed at an address where we have an instruction
of the same length as decr_pc_after_break on this arch (1 on x86).
- the breakpoint is removed, and thus put on the moribund locations
list.
- the thread is single-stepped.
As there's no breakpoint inserted at PC anymore, the single-step
actually executes the 1-byte instruction normally. GDB should _not_
adjust the PC for the resulting SIGTRAP. But, adjust_pc_after_break
confuses the step SIGTRAP reported for this single-step as being a
SIGTRAP for the moribund location of the breakpoint that used to be at
the previous PC, and so infrun applies the decr_pc_after_break
adjustment incorrectly.
The confusion comes from the special case mentioned in the comment:
static void
adjust_pc_after_break (struct execution_control_state *ecs)
{
...
As a special case, we could have hardware single-stepped a
software breakpoint. In this case (prev_pc == breakpoint_pc),
we also need to back up to the breakpoint address. */
if (thread_has_single_step_breakpoints_set (ecs->event_thread)
|| !ptid_equal (ecs->ptid, inferior_ptid)
|| !currently_stepping (ecs->event_thread)
|| (ecs->event_thread->stepped_breakpoint
&& ecs->event_thread->prev_pc == breakpoint_pc))
regcache_write_pc (regcache, breakpoint_pc);
The condition that incorrectly triggers is the
"ecs->event_thread->prev_pc == breakpoint_pc" one.
Afterwards, the next resume resume re-executes an instruction that had
already executed, which if you're lucky, results in the inferior
crashing. If you're unlucky, you'll get silent bad behavior...
The fix is to remember that we stepped a breakpoint. Turns out the
only case we step a breakpoint instruction today isn't covered by the
testsuite. It's the case of a 'handle nostop" signal arriving while a
step is in progress _and_ we have a software watchpoint, which forces
always single-stepping. This commit extends sigstep.exp to cover
that, and adds a new test for the adjust_pc_after_break issue.
Tested on x86_64 Fedora 20, native and gdbserver.
gdb/
2014-10-28 Pedro Alves <palves@redhat.com>
PR gdb/12623
* gdbthread.h (struct thread_info) <stepped_breakpoint>: New
field.
* infrun.c (resume) <stepping breakpoint instruction>: Set the
thread's stepped_breakpoint field. Skip if reverse debugging.
Add comment.
(init_thread_stepping_state, handle_signal_stop): Clear the
thread's stepped_breakpoint field.
gdb/testsuite/
2014-10-28 Pedro Alves <palves@redhat.com>
PR gdb/12623
* gdb.base/sigstep.c (no_handler): New global.
(main): If 'no_handler is true, set the signal handlers to
SIG_IGN.
* gdb.base/sigstep.exp (breakpoint_over_handler): Add
with_sw_watch and no_handler parameters. Handle them.
(top level) <stepping over handler when stopped at a breakpoint
test>: Add a test axis for testing with a software watchpoint, and
another for testing with the signal handler set to SIG_IGN.
* gdb.base/step-sw-breakpoint-adjust-pc.c: New file.
* gdb.base/step-sw-breakpoint-adjust-pc.exp: New file.
Diffstat (limited to 'gdb/infrun.c')
-rw-r--r-- | gdb/infrun.c | 34 |
1 files changed, 30 insertions, 4 deletions
diff --git a/gdb/infrun.c b/gdb/infrun.c index df053e2..7af5e0f 100644 --- a/gdb/infrun.c +++ b/gdb/infrun.c @@ -2213,12 +2213,35 @@ a command like `return' or `jump' to continue execution.")); resume_ptid = inferior_ptid; } - if (gdbarch_cannot_step_breakpoint (gdbarch)) - { + if (execution_direction != EXEC_REVERSE + && step && breakpoint_inserted_here_p (aspace, pc)) + { + /* The only case we currently need to step a breakpoint + instruction is when we have a signal to deliver. See + handle_signal_stop where we handle random signals that could + take out us out of the stepping range. Normally, in that + case we end up continuing (instead of stepping) over the + signal handler with a breakpoint at PC, but there are cases + where we should _always_ single-step, even if we have a + step-resume breakpoint, like when a software watchpoint is + set. Assuming single-stepping and delivering a signal at the + same time would takes us to the signal handler, then we could + have removed the breakpoint at PC to step over it. However, + some hardware step targets (like e.g., Mac OS) can't step + into signal handlers, and for those, we need to leave the + breakpoint at PC inserted, as otherwise if the handler + recurses and executes PC again, it'll miss the breakpoint. + So we leave the breakpoint inserted anyway, but we need to + record that we tried to step a breakpoint instruction, so + that adjust_pc_after_break doesn't end up confused. */ + gdb_assert (sig != GDB_SIGNAL_0); + + tp->stepped_breakpoint = 1; + /* Most targets can step a breakpoint instruction, thus executing it normally. But if this one cannot, just continue and we will hit it anyway. */ - if (step && breakpoint_inserted_here_p (aspace, pc)) + if (gdbarch_cannot_step_breakpoint (gdbarch)) step = 0; } @@ -3221,6 +3244,7 @@ set_step_info (struct frame_info *frame, struct symtab_and_line sal) void init_thread_stepping_state (struct thread_info *tss) { + tss->stepped_breakpoint = 0; tss->stepping_over_breakpoint = 0; tss->stepping_over_watchpoint = 0; tss->step_after_step_resume_breakpoint = 0; @@ -3385,7 +3409,8 @@ adjust_pc_after_break (struct execution_control_state *ecs) if (thread_has_single_step_breakpoints_set (ecs->event_thread) || !ptid_equal (ecs->ptid, inferior_ptid) || !currently_stepping (ecs->event_thread) - || ecs->event_thread->prev_pc == breakpoint_pc) + || (ecs->event_thread->stepped_breakpoint + && ecs->event_thread->prev_pc == breakpoint_pc)) regcache_write_pc (regcache, breakpoint_pc); do_cleanups (old_cleanups); @@ -4241,6 +4266,7 @@ handle_signal_stop (struct execution_control_state *ecs) return; } + ecs->event_thread->stepped_breakpoint = 0; ecs->event_thread->stepping_over_breakpoint = 0; ecs->event_thread->stepping_over_watchpoint = 0; bpstat_clear (&ecs->event_thread->control.stop_bpstat); |