# Copyright (C) 2011-2014 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 . # Test that GDB doesn't inadvertently resume the stepped thread when a # signal arrives while stepping over the breakpoint that last caused a # stop, when the thread that hit that breakpoint is not the stepped # thread. standard_testfile set executable ${testfile} if [target_info exists gdb,nosignals] { verbose "Skipping ${testfile}.exp because of nosignals." return -1 } # Test uses host "kill". if { [is_remote target] } { return -1 } if {[gdb_compile_pthreads "${srcdir}/${subdir}/${srcfile}" "${binfile}" \ executable [list debug "incdir=${objdir}"]] != "" } { return -1 } proc get_value {var test} { global expect_out global gdb_prompt global decimal set value -1 gdb_test_multiple "print $var" "$test" { -re ".*= ($decimal).*\r\n$gdb_prompt $" { set value $expect_out(1,string) pass "$test" } } return ${value} } # Start with a fresh gdb. clean_restart $executable if ![runto_main] { return -1 } gdb_breakpoint [gdb_get_line_number "set wait-thread-2 breakpoint here"] gdb_continue_to_breakpoint "run to wait-thread-2 breakpoint" gdb_test "info threads" "" "info threads with thread 2" gdb_breakpoint [gdb_get_line_number "set wait-thread-3 breakpoint here"] gdb_continue_to_breakpoint "run to breakpoint" gdb_test "info threads" "" "info threads with thread 3" set testpid [get_value "pid" "get pid of inferior"] if { $testpid == -1 } { return -1 } gdb_test "set scheduler-locking on" gdb_breakpoint [gdb_get_line_number "set breakpoint child_two here"] gdb_breakpoint [gdb_get_line_number "set breakpoint child_one here"] gdb_test "thread 3" "" "switch to thread 3 to run to its breakpoint" gdb_continue_to_breakpoint "run to breakpoint in thread 3" gdb_test "thread 2" "" "switch to thread 2 to run to its breakpoint" gdb_continue_to_breakpoint "run to breakpoint in thread 2" delete_breakpoints gdb_test "b *\$pc" "" "set breakpoint to be stepped over" # Make sure the first loop breaks without hitting the breakpoint # again. gdb_test "p *myp = 0" " = 0" "force loop break in thread 2" # We want "print" to make sure the target reports the signal to the # core. gdb_test "handle SIGUSR1 print nostop pass" "" "" # Queue a signal in thread 2. remote_exec host "kill -SIGUSR1 ${testpid}" gdb_test "thread 3" "" "switch to thread 3 for stepping" set my_number [get_value "my_number" "get my_number"] set cnt_before [get_value "args\[$my_number\]" "get count before step"] gdb_test "set scheduler-locking off" # Make sure we're exercising the paths we want to. gdb_test "set debug infrun 1" gdb_test \ "step" \ ".*need to step-over.*resume \\(step=1.*signal arrived while stepping over breakpoint.*switching back to stepped thread.*stepped to a different line.*callme.*" \ "step" set cnt_after [get_value "args\[$my_number\]" "get count after step"] # Test that GDB doesn't inadvertently resume the stepped thread when a # signal arrives while stepping over a breakpoint in another thread. set test "stepped thread under control" if { $cnt_before + 1 == $cnt_after } { pass $test } else { fail $test }