| # Copyright 2013-2021 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 solib-search-path, and in the case of solib-svr4.c whether l_addr_p |
| # is properly reset when the path is changed. |
| |
| if {[is_remote target] || [skip_shlib_tests]} { |
| untested "skipping remote target and shared library tests" |
| return -1 |
| } |
| |
| # This is required by gdb_compile_shlib. |
| if {[get_compiler_info]} { |
| warning "Could not get compiler info" |
| untested "no compiler info" |
| return 1 |
| } |
| |
| # Build "wrong" and "right" versions of the libraries in separate directories. |
| set wrong_lib_subdir "solib-search-wrong" |
| set right_lib_subdir "solib-search-right" |
| |
| # First library file. |
| set libname1 "solib-search-lib1" |
| set srcfile1_lib ${srcdir}/${subdir}/${libname1}.c |
| set wrong_binfile1_lib \ |
| [standard_output_file ${wrong_lib_subdir}/${libname1}.so] |
| set right_binfile1_lib \ |
| [standard_output_file ${right_lib_subdir}/${libname1}.so] |
| # Second library file. |
| set libname2 "solib-search-lib2" |
| set srcfile2_lib ${srcdir}/${subdir}/${libname2}.c |
| set wrong_binfile2_lib \ |
| [standard_output_file ${wrong_lib_subdir}/${libname2}.so] |
| set right_binfile2_lib \ |
| [standard_output_file ${right_lib_subdir}/${libname2}.so] |
| # Link with the library that lives here. |
| # This is so that we can replace what gdb sees with the wrong copy, |
| # and then tell gdb to use the right copy that lives someplace else. |
| set binfile1_lib [standard_output_file ${libname1}.so] |
| set binfile2_lib [standard_output_file ${libname2}.so] |
| |
| set lib_flags [list debug ldflags=-Wl,-Bsymbolic] |
| set wrong_lib_flags "$lib_flags additional_flags=-DARRAY_SIZE=1" |
| set right_lib_flags "$lib_flags additional_flags=-DARRAY_SIZE=8192 -DRIGHT" |
| |
| # Binary file. |
| standard_testfile .c |
| set bin_flags [list debug shlib=${binfile1_lib} shlib=${binfile2_lib}] |
| |
| remote_exec build "rm -rf [standard_output_file ${wrong_lib_subdir}]" |
| remote_exec build "rm -rf [standard_output_file ${right_lib_subdir}]" |
| remote_exec build "mkdir [standard_output_file ${wrong_lib_subdir}]" |
| remote_exec build "mkdir [standard_output_file ${right_lib_subdir}]" |
| |
| if { [gdb_compile_shlib ${srcfile1_lib} ${wrong_binfile1_lib} $wrong_lib_flags] != "" |
| || [gdb_compile_shlib ${srcfile2_lib} ${wrong_binfile2_lib} $wrong_lib_flags] != "" |
| || [gdb_compile_shlib ${srcfile1_lib} ${right_binfile1_lib} $right_lib_flags] != "" |
| || [gdb_compile_shlib ${srcfile2_lib} ${right_binfile2_lib} $right_lib_flags] != "" } { |
| untested "failed to compile shared library" |
| return -1 |
| } |
| |
| # Build the test binary using the right copies of the libraries. |
| remote_exec build "ln -sf ${right_lib_subdir}/${libname1}.so ${binfile1_lib}" |
| remote_exec build "ln -sf ${right_lib_subdir}/${libname2}.so ${binfile2_lib}" |
| if { [gdb_compile $srcdir/$subdir/${srcfile} ${binfile} \ |
| executable $bin_flags] != "" } { |
| untested "failed to compile" |
| return -1 |
| } |
| |
| clean_restart $testfile |
| |
| if { ![runto_main] } { |
| return |
| } |
| |
| gdb_breakpoint "break_here" |
| gdb_continue "break_here" |
| |
| set corefile [standard_output_file solib-search.core] |
| set core_supported [gdb_gcore_cmd "$corefile" "save a corefile"] |
| |
| if {!$core_supported} { |
| return -1 |
| } |
| |
| # Before we restart gdb, replace the libs with the "wrong" ones. |
| remote_exec build "ln -sf ${wrong_lib_subdir}/${libname1}.so ${binfile1_lib}" |
| remote_exec build "ln -sf ${wrong_lib_subdir}/${libname2}.so ${binfile2_lib}" |
| |
| clean_restart $testfile |
| |
| set core_loaded [gdb_core_cmd $corefile "re-load generated corefile"] |
| if { $core_loaded == -1 } { |
| # No use proceeding from here. |
| return |
| } |
| |
| proc test_backtrace { expect_fail } { |
| global gdb_prompt |
| |
| set count 0 |
| set total_expected 5 |
| |
| if { $expect_fail } { |
| set testname "backtrace (with wrong libs)" |
| } else { |
| set testname "backtrace (with right libs)" |
| } |
| # N.B. The order of the tests here is important. |
| # We need to count each function in the backtrace, and expect matches |
| # the first one it finds. |
| gdb_test_multiple "backtrace" $testname { |
| -re "\[^\r\n\]* in lib2_func4 \[^\r\n\]*" { |
| incr count |
| exp_continue |
| } |
| -re "\[^\r\n\]* in lib1_func3 \[^\r\n\]*" { |
| incr count |
| exp_continue |
| } |
| -re "\[^\r\n\]* in lib2_func2 \[^\r\n\]*" { |
| incr count |
| exp_continue |
| } |
| -re "\[^\r\n\]* in lib1_func1 \[^\r\n\]*" { |
| incr count |
| exp_continue |
| } |
| -re "\[^\r\n\]* in main \[^\r\n\]*" { |
| incr count |
| exp_continue |
| } |
| -re "\[\r\n\]$gdb_prompt $" { |
| pass "$testname (data collection)" |
| } |
| } |
| |
| set fail 0 |
| if { $expect_fail } { |
| # If the backtrace output is correct the test isn't sufficiently |
| # testing what it should. |
| if { $count == $total_expected } { |
| set fail 1 |
| } |
| } else { |
| if { $count != $total_expected } { |
| set fail 1 |
| } |
| } |
| if { $fail } { |
| fail $testname |
| } else { |
| pass $testname |
| } |
| } |
| |
| # Verify the backtrace is messed up. |
| test_backtrace 1 |
| |
| # Remove the copies gdb currently sees: the absolute path is encoded in |
| # the core file and we want gdb to find the right copies elsewhere. |
| remote_exec build "rm -f ${binfile1_lib}" |
| remote_exec build "rm -f ${binfile2_lib}" |
| # Set solib-search-path to use the correct copies of libraries. |
| gdb_test "set solib-search-path [standard_output_file ${right_lib_subdir}]" \ |
| "" \ |
| "set solib-search-path" |
| |
| # Verify gdb has properly updated the location of the libraries. |
| test_backtrace 0 |
| gdb_test "p lib1_size" " = 8192" |
| gdb_test "p lib2_size" " = 8192" |
| |