|  | # Copyright 2021-2025 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 stepping over a breakpoint installed on an instruction that | 
|  | # exits the thread. | 
|  |  | 
|  | standard_testfile .c | 
|  |  | 
|  | set syscalls_src $srcdir/lib/my-syscalls.S | 
|  |  | 
|  | if { [build_executable "failed to prepare" $testfile \ | 
|  | [list $srcfile $syscalls_src] {debug pthreads}] == -1 } { | 
|  | return | 
|  | } | 
|  |  | 
|  | # Test stepping/continuing at an exit syscall instruction. | 
|  | # | 
|  | # Each argument is a different testing axis. | 
|  | # | 
|  | # STEP_OVER_MODE can be one of: | 
|  | # | 
|  | #   - none: don't put a breakpoint on the exit syscall instruction. | 
|  | # | 
|  | #   - inline: put a breakpoint on the exit syscall instruction, and | 
|  | #     use in-line stepping to step over it (disable | 
|  | #     displaced-stepping). | 
|  | # | 
|  | #   - displaced: same, but use displaced stepping. | 
|  | # | 
|  | # SCHEDLOCK can be "on" or "off". | 
|  | # | 
|  | # CMD is the GDB command to run when at the exit syscall instruction. | 
|  | # | 
|  | # NS_STOP_ALL is only used if testing "set non-stop on", and indicates | 
|  | # whether to have GDB explicitly stop all threads before continuing to | 
|  | # thread exit. | 
|  | # | 
|  | proc test {step_over_mode non-stop target-non-stop schedlock cmd ns_stop_all} { | 
|  | if {${non-stop} == "off" && $ns_stop_all} { | 
|  | error "invalid arguments" | 
|  | } | 
|  |  | 
|  | save_vars ::GDBFLAGS { | 
|  | append ::GDBFLAGS " -ex \"maintenance set target-non-stop ${target-non-stop}\"" | 
|  | append ::GDBFLAGS " -ex \"set non-stop ${non-stop}\"" | 
|  | clean_restart $::testfile | 
|  | } | 
|  |  | 
|  | if { $step_over_mode == "none" } { | 
|  | # Nothing to do. | 
|  | } elseif { $step_over_mode == "inline" } { | 
|  | gdb_test_no_output "set displaced-stepping off" | 
|  | } elseif { $step_over_mode == "displaced" } { | 
|  | gdb_test_no_output "set displaced-stepping on" | 
|  | } else { | 
|  | error "Invalid step_over_mode value: $step_over_mode" | 
|  | } | 
|  |  | 
|  | if {$schedlock | 
|  | || (${non-stop} == "on" && $ns_stop_all)} { | 
|  |  | 
|  | gdb_test_no_output "set args 1" | 
|  |  | 
|  | if { ![runto my_exit_syscall] } { | 
|  | return | 
|  | } | 
|  |  | 
|  | if {${non-stop} == "on"} { | 
|  | # The test only spawns one thread at a time, so this just | 
|  | # stops the main thread.  IOW, we only need to wait for | 
|  | # one stop. | 
|  | gdb_test_multiple "interrupt -a" "" { | 
|  | -re "$::gdb_prompt " { | 
|  | gdb_test_multiple "" $gdb_test_name { | 
|  | -re "Thread 1 \[^\r\n\]*stopped." { | 
|  | pass $gdb_test_name | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | gdb_test "thread 2" "Switching to thread 2 .*" | 
|  | } | 
|  |  | 
|  | gdb_test_no_output "set scheduler-locking ${schedlock}" | 
|  |  | 
|  | # If testing a step-over is requested, leave the breakpoint at | 
|  | # the current instruction to force a step-over; otherwise, | 
|  | # remove it. | 
|  | if { $step_over_mode == "none" } { | 
|  | delete_breakpoints | 
|  | } | 
|  |  | 
|  | if {$cmd == "continue"} { | 
|  | gdb_test "continue" \ | 
|  | "No unwaited-for children left." \ | 
|  | "continue stops when thread exits" | 
|  | } else { | 
|  | gdb_test_multiple $cmd "command aborts when thread exits" { | 
|  | -re "Command aborted, thread exited\\.\r\n$::gdb_prompt " { | 
|  | pass $gdb_test_name | 
|  | } | 
|  | } | 
|  | } | 
|  | } else { | 
|  | # Schedlock is off here. | 
|  | # | 
|  | # With "continue" and no scheduler-locking, GDB doesn't stop | 
|  | # with "Command aborted, thread exited." when the thread | 
|  | # exits, it just lets the inferior continue running freely. | 
|  | # So we test that we can move past the thread exit, and that | 
|  | # other threads can be freely scheduled.  We do that by | 
|  | # spawning another thread as soon as the first exit.  We test | 
|  | # that a number of times.  This should also exercise GDB's | 
|  | # handling of inline or displaced step-overs, that GDB handles | 
|  | # the related resource accounting correctly when the stepping | 
|  | # thread exits, etc. | 
|  | # | 
|  | # With "continue" and $step_over_mode == "none" however, after | 
|  | # the first my_exit_syscall breakpoint hit, we will remove the | 
|  | # breakpoint, so no other thread would ever hit it again.  So | 
|  | # might as well just test one thread. | 
|  | # | 
|  | # With step/next, GDB aborts the execution command with | 
|  | # "Command aborted, thread exited." when the stepping thread | 
|  | # exits.  If we let the main spawn another thread as soon as | 
|  | # the first exits, it would be possible for that new thread to | 
|  | # hit the exit syscall insn breakpoint quickly enough that it | 
|  | # would be reported to be user before the first thread exit | 
|  | # would be, which would confuse testing.  To avoid that, we | 
|  | # only spawn one thread, too. | 
|  | # | 
|  | if {$cmd != "continue" || $step_over_mode == "none"} { | 
|  | set n_threads 1 | 
|  | } else { | 
|  | set n_threads 100 | 
|  | } | 
|  |  | 
|  | gdb_test_no_output "set args $n_threads" | 
|  |  | 
|  | if { ![runto_main] } { | 
|  | return | 
|  | } | 
|  |  | 
|  | gdb_breakpoint "my_exit_syscall" | 
|  |  | 
|  | gdb_test_no_output "set scheduler-locking ${schedlock}" | 
|  |  | 
|  | if {$cmd != "continue" || $step_over_mode == "none"} { | 
|  | set thread "<unknown>" | 
|  | gdb_test_multiple "continue" "" { | 
|  | -re -wrap "Thread ($::decimal) .*hit Breakpoint $::decimal.* my_exit_syscall .*" { | 
|  | set thread $expect_out(1,string) | 
|  | } | 
|  | } | 
|  | if {${non-stop}} { | 
|  | gdb_test -nopass "thread $thread" "Switching to thread .*" \ | 
|  | "switch to event thread" | 
|  | } | 
|  |  | 
|  | # If testing a step-over is requested, leave the breakpoint at | 
|  | # the current instruction to force a step-over; otherwise, | 
|  | # remove it. | 
|  | if { $step_over_mode == "none" } { | 
|  | delete_breakpoints | 
|  | } | 
|  |  | 
|  | if {$cmd == "continue"} { | 
|  | gdb_continue_to_end "continue to end" "continue" 1 | 
|  | } else { | 
|  | gdb_test_multiple $cmd "command aborts when thread exits" { | 
|  | -re "Command aborted, thread exited\\.\r\n$::gdb_prompt " { | 
|  | pass $gdb_test_name | 
|  | } | 
|  | } | 
|  | gdb_test "p \$_thread == $thread" "= 1" \ | 
|  | "selected thread didn't change" | 
|  | } | 
|  | } else { | 
|  | for { set i 0 } { $i < 100 } { incr i } { | 
|  | with_test_prefix "iter $i" { | 
|  | set ok 0 | 
|  | set thread "<unknown>" | 
|  | gdb_test_multiple "continue" "" -no-prompt-anchor { | 
|  | -re -wrap "Thread ($::decimal) .*hit Breakpoint $::decimal.* my_exit_syscall .*" { | 
|  | set thread $expect_out(1,string) | 
|  | set ok 1 | 
|  | } | 
|  | } | 
|  | if {!${ok}} { | 
|  | # Exit if there's a failure to avoid lengthy | 
|  | # timeouts. | 
|  | break | 
|  | } | 
|  |  | 
|  | if {${non-stop}} { | 
|  | gdb_test -nopass -no-prompt-anchor "thread $thread" \ | 
|  | "Switching to thread .*" \ | 
|  | "switch to event thread" | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | foreach_with_prefix step_over_mode {none inline displaced} { | 
|  | foreach_with_prefix non-stop {off on} { | 
|  | foreach_with_prefix target-non-stop {off on} { | 
|  | if {${non-stop} == "on" && ${target-non-stop} == "off"} { | 
|  | # Invalid combination. | 
|  | continue | 
|  | } | 
|  |  | 
|  | foreach_with_prefix schedlock {off on} { | 
|  | foreach_with_prefix cmd {"next" "continue"} { | 
|  | if {${non-stop} == "on"} { | 
|  | foreach_with_prefix ns_stop_all {0 1} { | 
|  | test ${step_over_mode} ${non-stop} ${target-non-stop} \ | 
|  | ${schedlock} ${cmd} ${ns_stop_all} | 
|  | } | 
|  | } else { | 
|  | test ${step_over_mode} ${non-stop} ${target-non-stop} ${schedlock} ${cmd} 0 | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  | } |