# Copyright 2023-2024 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 . # Tests related to pending breakpoints in a multi-inferior environment. require allow_shlib_tests !use_gdb_stub standard_testfile set libname $testfile-lib set srcfile_lib $srcdir/$subdir/$libname.c set binfile_lib [standard_output_file $libname.so] if { [gdb_compile_shlib $srcfile_lib $binfile_lib {}] != "" } { untested "failed to compile shared library 1" return -1 } set binfile_lib_target [gdb_download_shlib $binfile_lib] if { [build_executable "failed to prepare" $testfile $srcfile \ [list debug \ additional_flags=-DSHLIB_NAME=\"$binfile_lib_target\" \ shlib_load]] } { return -1 } # Start two inferiors, both running the same test binary. The arguments # INF_1_STOP and INF_2_STOP are source code patterns that are passed to # gdb_get_line_number to figure out where each inferior should be stopped. # # This proc does a clean_restart and leaves inferior 2 selected. Also the # 'breakpoint pending' flag is enabled, so pending breakpoints can be created # without GDB prompting the user. proc do_test_setup { inf_1_stop inf_2_stop } { clean_restart ${::binfile} gdb_locate_shlib $::binfile_lib if {![runto_main]} { return false } gdb_breakpoint [gdb_get_line_number ${inf_1_stop}] temporary gdb_continue_to_breakpoint "move inferior 1 into position" gdb_test "add-inferior -exec ${::binfile}" \ "Added inferior 2.*" "add inferior 2" gdb_test "inferior 2" "Switching to inferior 2 .*" "switch to inferior 2" if {![runto_main]} { return false } gdb_breakpoint [gdb_get_line_number ${inf_2_stop}] temporary gdb_continue_to_breakpoint "move inferior 2 into position" gdb_test_no_output "set breakpoint pending on" return true } # Create a breakpoint on the function 'foo' in THREAD. It is expected # that the breakpoint created will be pending, this is checked by # running the 'info breakpoints' command. # # Returns the number for the newly created breakpoint. proc do_create_pending_foo_breakpoint { {thread "1.1"} } { gdb_test "break foo thread $thread" \ [multi_line \ "Function \"foo\" not defined\\." \ "Breakpoint $::decimal \\(foo\\) pending\."] \ "set pending thread-specific breakpoint" set bpnum [get_integer_valueof "\$bpnum" "*INVALID*" \ "get number for thread-specific breakpoint on foo"] gdb_test "info breakpoints $bpnum" \ [multi_line \ "$bpnum\\s+breakpoint\\s+keep\\s+y\\s+\\s+foo" \ "\\s+stop only in thread [string_to_regexp $thread]"] \ "check thread-specific breakpoint is initially pending" return $bpnum } # Create a breakpoint on the function 'foo' in THREAD. It is expected # that the breakpoint created will not be pending, this is checked by # running the 'info breakpoints' command. # # Returns the number for the newly created breakpoint. proc do_create_foo_breakpoint { {thread "1.1"} } { gdb_test "break foo thread $thread" \ "Breakpoint $::decimal at $::hex" \ "set thread-specific breakpoint" set bpnum [get_integer_valueof "\$bpnum" "*INVALID*" \ "get number for thread-specific breakpoint on foo"] gdb_test "info breakpoints $bpnum" \ [multi_line \ "$bpnum\\s+breakpoint\\s+keep\\s+y\\s+$::hex\\s+\]*> inf $::decimal" \ "\\s+stop only in thread [string_to_regexp $thread]"] \ "check thread-specific breakpoint is initially pending" return $bpnum } # Check that when a breakpoint is in the pending state, but that breakpoint # does have some locations (those locations themselves are pending), GDB # doesn't display the inferior list in the 'info breakpoints' output. proc_with_prefix test_no_inf_display {} { do_test_setup "Break before open" "Break before open" # Create a breakpoint on 'foo'. As the shared library (that # contains foo) has not been loaded into any inferior yet, then # there will be no locations and the breakpoint will be created # pending. Pass the 'allow-pending' flag so the gdb_breakpoint # correctly expects the new breakpoint to be pending. gdb_breakpoint "foo" allow-pending set bpnum [get_integer_valueof "\$bpnum" "*INVALID*" \ "get foo breakpoint number"] # Check the 'info breakpoints' output; the breakpoint is pending with # no 'inf X' appearing at the end of the line. gdb_test "info breakpoint $bpnum" \ "$bpnum\\s+breakpoint\\s+keep\\s+y\\s+\\s+foo" \ "check info bp before locations have been created" # Now select inferior 1 and allow the inferior to run forward to the # point where a breakpoint location for foo will have been created. gdb_test "inferior 1" "Switching to inferior 1 .*" gdb_breakpoint [gdb_get_line_number "Break after open"] temporary gdb_continue_to_breakpoint \ "move inferior 1 until a location has been added" # Check the 'info breakpoints' output. Notice we display the inferior # list at the end of the breakpoint line. gdb_test "info breakpoint $bpnum" \ "$bpnum\\s+breakpoint\\s+keep\\s+y\\s+$::hex\\s+\]*>\\s+inf 1" \ "check info breakpoints while breakpoint is inserted" # Continue inferior 1 until the shared library has been unloaded. The # breakpoint on 'foo' will return to the pending state. We will need to # 'continue' twice as the first time will hit the 'foo' breakpoint. gdb_breakpoint [gdb_get_line_number "Break after close"] temporary gdb_continue_to_breakpoint "hit the breakpoint in foo" gdb_continue_to_breakpoint "after close library" # Check the 'info breakpoints' output, check there is no 'inf 1' at the # end of the breakpoint line. gdb_test "info breakpoint $bpnum" \ [multi_line \ "$bpnum\\s+breakpoint\\s+keep\\s+y\\s+\\s+foo" \ "\\s+breakpoint already hit 1 time"] \ "check info breakpoints while breakpoint is pending" } # Setup two inferiors. In #1 the symbol 'foo' has not yet been # loaded, while in #2 the symbol 'foo' has been loaded. # # Create a thread-specific breakpoint on 'foo' tied to a thread in # inferior #1, the breakpoint should be pending -- 'foo' is not yet # loaded in #1. # # Now move inferior #1 forward until 'foo' is loaded, check the # breakpoint is no longer pending. # # Move inferior #1 forward more until 'foo' is unloaded, check that # the breakpoint returns to the pending state. proc_with_prefix test_pending_toggle { } { do_test_setup "Break before open" "Break before close" set bpnum [do_create_pending_foo_breakpoint] # Now return to inferior 1 and continue until the shared library is # loaded, the breakpoint should become non-pending. gdb_test "inferior 1" "Switching to inferior 1 .*" \ "switch back to inferior 1" gdb_continue_to_breakpoint "stop in foo in inferior 1" "foo \\(\\) .*" gdb_test "info breakpoint $bpnum" \ [multi_line \ "$bpnum\\s+breakpoint\\s+keep\\s+y\\s+$::hex \]*> inf 1" \ "\\s+stop only in thread 1\\.1" \ "\\s+breakpoint already hit 1 time"] \ "check thread-specific breakpoint is no longer pending" gdb_breakpoint [gdb_get_line_number "Break after close"] temporary gdb_continue_to_breakpoint "close library" gdb_test "info breakpoints $bpnum" \ [multi_line \ "$bpnum\\s+breakpoint\\s+keep\\s+y\\s+\\s+foo" \ "\\s+stop only in thread 1\\.1" \ "\\s+breakpoint already hit 1 time"] \ "check thread-specific breakpoint is pending again" } # Create a Python variable VAR and set it to the gdb.Breakpoint object # corresponding to the breakpoint numbered BPNUM. If THREAD is not # the empty string then THREAD should be an integer, check that # gdb.Breakpoint.thread is set to the value of THREAD. Otherwise, if # THREAD is the empty string, check that gdb.Breakpoint.thread is set # to None. proc py_find_breakpoint { var bpnum {thread ""} } { gdb_test_no_output \ "python ${var}=\[b for b in gdb.breakpoints() if b.number == $bpnum\]\[0\]" \ "find Python gdb.Breakpoint object" if { $thread ne "" } { gdb_test_no_output "python assert(${var}.thread == ${thread})" \ "check thread attribute is currently correct" } else { gdb_test_no_output "python assert(${var}.thread is None)" \ "check thread attribute is currently correct" } } # Setup two inferiors. In #1 the symbol 'foo' has not yet been # loaded, while in #2 the symbol 'foo' has been loaded. # # Create a thread-specific breakpoint on 'foo' tied to a thread in # inferior #1, the breakpoint should be pending -- 'foo' is not yet # loaded in #1. # # Use Python to change the thread of the thread-specific breakpoint to # a thread in inferior #2, at this point the thread should gain a # location and become non-pending. # # Set the thread back to a thread in inferior #1, the breakpoint # should return to the pending state. proc_with_prefix py_test_toggle_thread {} { do_test_setup "Break before open" "Break after open" set bpnum [do_create_pending_foo_breakpoint] py_find_breakpoint "bp" $bpnum 1 gdb_test_no_output "python bp.thread = 2" \ "change thread on thread-specific breakpoint" gdb_test "info breakpoint $bpnum" \ [multi_line \ "$bpnum\\s+breakpoint\\s+keep\\s+y\\s+$::hex \]*> inf 2" \ "\\s+stop only in thread 2\\.1"] \ "check thread-specific breakpoint now has a location" gdb_test_no_output "set call_count = 2" "set call_count in inferior 2" gdb_continue_to_breakpoint "stop at foo in inferior 2" "foo \\(\\) .*" gdb_test_no_output "python bp.thread = 1" \ "restore thread on thread-specific breakpoint" gdb_test "info breakpoints $bpnum" \ [multi_line \ "$bpnum\\s+breakpoint\\s+keep\\s+y\\s+\\s+foo" \ "\\s+stop only in thread 1\\.1" \ "\\s+breakpoint already hit 1 time"] \ "check thread-specific breakpoint has returned to pending" gdb_breakpoint [gdb_get_line_number "Break after close"] temporary gdb_continue_to_breakpoint "stop after close in inferior 2" \ ".* Break after close\\. .*" gdb_test "inferior 1" "Switching to inferior 1 .*" \ "switch to inferior 1" gdb_continue_to_breakpoint "stop at foo in inferior 1" "foo \\(\\) .*" } # Setup two inferiors. Both inferiors have the symbol 'foo' # available. # # Create a thread-specific breakpoint on 'foo' tied to a thread in # inferior #1, the breakpoint should not be pending, but will only # have a single location, the location in inferior #1. # # Use Python to change the thread of the thread-specific breakpoint to # None. At this point the breakpoint should gain a second location, a # location in inferior #2. proc_with_prefix py_test_clear_thread {} { do_test_setup "Break after open" "Break after open" set bpnum [do_create_foo_breakpoint] py_find_breakpoint "bp" $bpnum 1 gdb_test_no_output "python bp.thread = None" \ "clear thread on thread-specific breakpoint" gdb_test "info breakpoints $bpnum" \ [multi_line \ "${bpnum}\\s+breakpoint\\s+keep y\\s+\\s*" \ "${bpnum}\\.1\\s+y\\s+${::hex}\\s+\]*> inf $::decimal" \ "${bpnum}\\.2\\s+y\\s+${::hex}\\s+\]*> inf $::decimal"] \ "check for a location in both inferiors" gdb_continue_to_breakpoint "stop at foo in inferior 2" "foo \\(\\) .*" gdb_test_no_output "set call_count = 2" "set call_count in inferior 2" gdb_test "inferior 1" "Switching to inferior 1 .*" \ "switch to inferior 1" gdb_continue_to_breakpoint "stop at foo in inferior 1" "foo \\(\\) .*" gdb_test_no_output "set call_count = 2" "set call_count in inferior 1" gdb_test_no_output "python bp.thread = 2" gdb_test "info breakpoints $bpnum" \ [multi_line \ "${bpnum}\\s+breakpoint\\s+keep y\\s+${::hex}\\s+\]*> inf 2" \ "\\s+stop only in thread 2\\.1" \ "\\s+breakpoint already hit 2 times"] \ "check for a location only in inferior 2" gdb_breakpoint [gdb_get_line_number "Break after close"] temporary gdb_continue_to_breakpoint "stop after close in inferior 1" \ ".* Break after close\\. .*" gdb_test "inferior 2" "Switching to inferior 2 .*" \ "switch back to inferior 2" gdb_continue_to_breakpoint "stop at foo again in inferior 2" \ "foo \\(\\) .*" } # Run all the tests. test_no_inf_display test_pending_toggle py_test_toggle_thread py_test_clear_thread