# Copyright 2009-2023 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 . # # Contributed by Paul Pluzhnikov # # This test case verifies that if a display is active on a variable # which belongs in a shared library, and that shared library is # reloaded (e.g. due to re-execution of the program), GDB will continue # to display it (gdb-6.8 crashed under this scenario). # Also test that a display of variable which is currently present in # a shared library, but disappears before re-run, doesn't cause GDB # difficulties, and that it continues to display other variables. # Finally, test that displays which refer to main executable # (and thus aren't affected by shared library unloading) are not # disabled prematurely. if { [skip_shlib_tests] } { return 0 } # This test is currently not supported for stub targets, because it uses the # start command (through gdb_start_cmd). In theory, it could be changed to # use something else (kill + gdb_run_cmd with a manual breakpoint at main). # However, when we try that with the native-gdbserver board, we see that the # test fails and gdb outputs this upon connection: # # warning: Unable to display "a_global": No symbol "a_global" in current context. # warning: Unable to display "b_global": No symbol "b_global" in current context. # warning: Unable to display "c_global": No symbol "c_global" in current context. # # This is because the initial stop is done before the shared libraries are # loaded. if { [use_gdb_stub] } { return 0 } # Library file. set libname "solib-display-lib" set srcfile_lib ${srcdir}/${subdir}/${libname}.c set binfile_lib [standard_output_file ${libname}.so] set lib_flags {} # Binary file. set testfile "solib-display-main" set srcfile ${srcdir}/${subdir}/${testfile}.c set executable ${testfile} set binfile [standard_output_file ${executable}] set bin_flags [list debug shlib=${binfile_lib}] # SEP must be last for the possible `unsupported' error path. foreach libsepdebug {NO IN SEP} { with_test_prefix "$libsepdebug" { set sep_lib_flags $lib_flags if {$libsepdebug != "NO"} { lappend sep_lib_flags {debug} } if { [gdb_compile_shlib ${srcfile_lib} ${binfile_lib} $sep_lib_flags] != "" || [gdb_compile ${srcfile} ${binfile} executable $bin_flags] != "" } { untested "failed to compile" return -1 } if {$libsepdebug == "SEP"} { if {[gdb_gnu_strip_debug $binfile_lib] != 0} { unsupported "could not split debug of $binfile_lib." return } else { pass "split solib" } } clean_restart $executable if {![runto_main]} { return 0 } gdb_test "display (int) a_global" "1: \\(int\\) a_global = 41" gdb_test "display (int) b_global" "2: \\(int\\) b_global = 42" gdb_test "display (int) c_global" "3: \\(int\\) c_global = 43" if { [gdb_start_cmd] < 0 } { fail "can't run to main (2)" continue } gdb_test "" [multi_line \ "1: \\(int\\) a_global = 41" \ "2: \\(int\\) b_global = 42" \ "3: \\(int\\) c_global = 43" \ ] "after rerun" # Now rebuild the library without b_global if { [gdb_compile_shlib ${srcfile_lib} ${binfile_lib} \ "$sep_lib_flags additional_flags=-DNO_B_GLOBAL"] != ""} { fail "can't rebuild $binfile_lib" } if {$libsepdebug == "SEP"} { set test "split solib second time" if {[gdb_gnu_strip_debug $binfile_lib] != 0} { fail $test continue } else { pass $test } } if { [gdb_start_cmd] < 0 } { fail "can't run to main (3)" continue } gdb_test "" [multi_line \ "1: \\(int\\) a_global = 41" \ "warning: .*b_global.*" \ "3: \\(int\\) c_global = 43" \ ] "after rerun (2)" # Now verify that displays which are not in the shared library # are not cleared permaturely. gdb_test "break [gdb_get_line_number "break here" ${testfile}.c]" \ ".*Breakpoint.* at .*" gdb_test "continue" gdb_test "display main_global" "4: main_global = 44" gdb_test "display a_local" "5: a_local = 45" gdb_test "display a_static" "6: a_static = 46" if { [gdb_start_cmd] < 0 } { fail "can't run to main (4)" continue } gdb_test "" "6: a_static = 46\\r\\n4: main_global = 44\\r\\n.*" gdb_test "continue" [multi_line \ "4: main_global = 44" \ "5: a_local = 45" \ "6: a_static = 46" \ ] "continue two" }}