| # 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 namespace aliases. |
| # PRs c++/7935, c++/10541 |
| |
| load_lib dwarf.exp |
| |
| if {![dwarf2_support]} { |
| return 0 |
| } |
| |
| if {[skip_cplus_tests]} { |
| continue |
| } |
| |
| standard_testfile .cc nsalias-dw.S |
| |
| # Make the DWARF used for the test. This is necessary to work |
| # around compiler issues. Not all versions of gcc output the |
| # correct debuginfo we need. |
| # |
| # This should create the equivalent DWARF to: |
| # |
| # namespace outer |
| # { |
| # namespace inner |
| # { |
| # namespace innermost |
| # { |
| # const int x = 2; |
| # int foo (void) { return x; } |
| # } |
| # |
| # namespace Innermost = innermost; |
| # |
| # const int x = 1; |
| # int foo (void) { return x + Innermost::foo (); } |
| # } |
| # |
| # namespace Inner = inner; |
| # |
| # const int x = 0; |
| # int foo (void) { return x + Inner::foo (); } |
| # } |
| # |
| # namespace Outer = outer; |
| # namespace oi = Outer::Inner; |
| |
| set asm_file [standard_output_file $srcfile2] |
| Dwarf::assemble $asm_file { |
| cu {} { |
| compile_unit {{language @DW_LANG_C_plus_plus}} { |
| declare_labels int_label outer_label inner_label innermost_label |
| declare_labels im_foo_label i_foo_label o_foo_label |
| declare_labels OuterInner_label oi1_label oi2_label |
| |
| int_label: base_type { |
| {name int} |
| {encoding @DW_ATE_signed} |
| {byte_size 4 DW_FORM_sdata} |
| } |
| |
| outer_label: DW_TAG_namespace { |
| {name outer} |
| } { |
| inner_label: DW_TAG_namespace { |
| {name inner} |
| } { |
| innermost_label: DW_TAG_namespace { |
| {name innermost} |
| } { |
| DW_TAG_variable { |
| {name x} |
| {type :$int_label} |
| {const_value 2 DW_FORM_data1} |
| } |
| |
| im_foo_label: DW_TAG_subprogram { |
| {name foo} |
| {external 1 flag_present} |
| {declaration 1 flag_present} |
| } |
| } |
| |
| imported_declaration { |
| {name Innermost} |
| {import :$innermost_label} |
| } |
| |
| DW_TAG_variable { |
| {name x} |
| {type :$int_label} |
| {const_value 1 DW_FORM_data1} |
| } |
| |
| i_foo_label: subprogram { |
| {name foo} |
| {external 1 flag_present} |
| {declaration 1 flag_present} |
| } |
| } |
| |
| OuterInner_label: imported_declaration { |
| {name Inner} |
| {import :$inner_label} |
| } |
| |
| DW_TAG_variable { |
| {name x} |
| {type :$int_label} |
| {const_value 0 DW_FORM_data1} |
| } |
| |
| o_foo_label: subprogram { |
| {name foo} |
| {external 1 flag_present} |
| {declaration 1 flag_present} |
| } |
| } |
| |
| imported_declaration { |
| {name Outer} |
| {import :$outer_label} |
| } |
| |
| oi1_label: imported_declaration { |
| {name oi1} |
| {import :$OuterInner_label} |
| } |
| |
| oi2_label: imported_declaration { |
| {name oi2} |
| {import :$oi1_label} |
| } |
| |
| imported_declaration { |
| {name oi3} |
| {import :$oi2_label} |
| } |
| |
| subprogram { |
| {specification :$im_foo_label} |
| {low_pc 0x4 DW_FORM_addr} |
| {high_pc 0x7 DW_FORM_addr} |
| } |
| |
| subprogram { |
| {specification :$i_foo_label} |
| {low_pc 0x8 DW_FORM_addr} |
| {high_pc 0xb DW_FORM_addr} |
| } |
| |
| subprogram { |
| {specification :$o_foo_label} |
| {low_pc 0xc DW_FORM_addr} |
| {high_pc 0xf DW_FORM_addr} |
| } |
| } |
| } |
| } |
| |
| if {[gdb_compile $srcdir/$subdir/$srcfile ${binfile}1.o \ |
| object {c++ debug}] != ""} { |
| return -1 |
| } |
| |
| if {[gdb_compile $asm_file ${binfile}2.o object {nodebug}] != ""} { |
| return -1 |
| } |
| |
| if {[gdb_compile [list ${binfile}1.o ${binfile}2.o] \ |
| $binfile executable {c++}] != ""} { |
| return -1 |
| } |
| |
| clean_restart $testfile |
| |
| # A procedure to run various tests on aliased namespaces. |
| proc do_alias_tests {ns {real ""} {x ""}} { |
| |
| # The "real" namespace is simply NS in all lowercase. |
| if {$real == ""} { |
| set real [string tolower $ns] |
| } |
| |
| # The value of `x' is the number of '::' in NS. |
| if {$x == ""} { |
| set x [expr {[llength [split $ns ":"]] / 2}] |
| } |
| |
| # Test "whatis" |
| gdb_test "whatis $ns" "type = $real" |
| |
| # Test "ptype" |
| gdb_test "ptype $ns" "type = namespace $real" |
| |
| # Print 'x' |
| send_log "expecting x = $x\n" |
| gdb_test "print ${ns}::x" " = $x" |
| |
| # Attempt to list the function. |
| gdb_test_no_output "list ${ns}::foo" |
| |
| # Attempt to break on the start of the function. |
| gdb_breakpoint "*${ns}::foo" |
| |
| # And then erase it |
| with_test_prefix "($ns)" { |
| gdb_test_no_output "delete \$bpnum" |
| } |
| } |
| |
| # This is a list of all the permutations to be tested. For troubleshooting |
| # purposes, this list is explicitly enumerated. |
| |
| set permutations {} |
| lappend permutations "outer" |
| lappend permutations "Outer" |
| lappend permutations "outer::inner" |
| lappend permutations "Outer::inner" |
| lappend permutations "outer::Inner" |
| lappend permutations "Outer::Inner" |
| lappend permutations "outer::inner::innermost" |
| lappend permutations "outer::inner::Innermost" |
| lappend permutations "outer::Inner::innermost" |
| lappend permutations "outer::Inner::Innermost" |
| lappend permutations "Outer::inner::innermost" |
| lappend permutations "Outer::inner::Innermost" |
| lappend permutations "Outer::Inner::innermost" |
| lappend permutations "Outer::Inner::Innermost" |
| |
| foreach p $permutations { |
| do_alias_tests $p |
| } |
| |
| # Test recursively imported aliases. |
| foreach ns {"oi1" "oi2" "oi3"} { |
| do_alias_tests $ns "outer::inner" 1 |
| do_alias_tests "${ns}::innermost" "outer::inner::innermost" 2 |
| do_alias_tests "${ns}::Innermost" "outer::inner::innermost" 2 |
| } |
| |
| # Generate another objfile with nested imported declarations. |
| |
| set imports { |
| declare_labels n0_label |
| |
| n0_label: DW_TAG_namespace { |
| {name n0} |
| } { |
| DW_TAG_variable { |
| {name x} |
| {type :$int_label} |
| {const_value 3 DW_FORM_data1} |
| } |
| } |
| |
| declare_labels n0_import |
| n0_import: imported_declaration { |
| {name N0} |
| {import :$n0_label} |
| } |
| } |
| |
| for {set i 1} {$i <= 100} {incr i} { |
| append imports [format " |
| declare_labels n%d_import |
| n%d_import: imported_declaration { |
| {name N%d} |
| {import :\$n%d_import} |
| }" $i $i $i [expr {$i - 1}]] |
| } |
| |
| standard_testfile .cc nsalias-r-dw.S |
| |
| set asm_file [standard_output_file $srcfile2] |
| set the_dwarf [format { |
| cu {} { |
| compile_unit {{language @DW_LANG_C_plus_plus}} { |
| declare_labels int_label n0_label |
| |
| int_label: base_type { |
| {name int} |
| {encoding @DW_ATE_signed} |
| {byte_size 4 DW_FORM_sdata} |
| } |
| |
| %s |
| } |
| } |
| } $imports] |
| |
| Dwarf::assemble $asm_file $the_dwarf |
| |
| if {[gdb_compile $asm_file ${binfile}3.o object {nodebug}] != ""} { |
| return -1 |
| } |
| |
| if {[gdb_compile [list ${binfile}1.o ${binfile}3.o] \ |
| ${binfile}-r executable {c++}] != ""} { |
| return -1 |
| } |
| |
| clean_restart |
| |
| # Set complaints before loading the file. Otherwise the complaint won't |
| # trigger for -readnow. |
| gdb_test_no_output "set complaints 1" |
| |
| gdb_load [standard_output_file ${testfile}-r] |
| |
| set readnow_p [readnow] |
| |
| set test "complaint for too many recursively imported declarations" |
| set re ".* has too many recursively imported declarations.*" |
| if { $readnow_p } { |
| global gdb_file_cmd_msg |
| gdb_assert {[regexp $re $gdb_file_cmd_msg]} $test |
| } else { |
| gdb_test "print N100::x" $re $test |
| } |