# Copyright 2019-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 . # Import a CU into an "artificial" CU. For each DW_TAG DIE in the # artificial CU, use DW_AT_abstract_origin to refer to a DIE in the # imported CU. This DWARF file organization is frequently found in # programs compiled with -flto (and -g) using GCC. # # This test reproduces the bug described in BZ 25065 without relying # on specific compiler versions or use of optimization switches, in # this case -flto. if [skip_cplus_tests] { return } load_lib dwarf.exp # This test can only be run on targets which support DWARF-2 and use gas. if {![dwarf2_support]} { return 0 }; standard_testfile .c .S # ${testfile} is now "implref-struct". srcfile2 is "implref-struct.S". set executable ${testfile} set asm_file [standard_output_file ${srcfile2}] # We need to know the size of integer and address types in order # to write some of the debugging info we'd like to generate. if [prepare_for_testing "failed to prepare" ${testfile} ${srcfile} {debug c++}] { return -1 } # Create the DWARF. Dwarf::assemble $asm_file { declare_labels cu_label main_label doit_label int_label declare_labels Foo_label Foo_pointer_type doit_self_label declare_labels foo_label Foo_destructor_obj_pointer_label declare_labels Foo_constructor_obj_pointer_label set int_size [get_sizeof "int" 4] set addr_size [get_sizeof "void *" 8] cu {} { compile_unit { {language @DW_LANG_C_plus_plus} {name ""} } { imported_unit { {import %$cu_label} } subprogram { {abstract_origin %$main_label} {MACRO_AT_range {main}} } { subprogram { {abstract_origin %$doit_label} {MACRO_AT_range {doit}} } { formal_parameter { {abstract_origin %$doit_self_label} } } DW_TAG_variable { {abstract_origin %$foo_label} {location 4 data1} } } } } cu {} { cu_label: compile_unit { {language @DW_LANG_C_plus_plus} {name "imported_unit.c"} } { int_label: base_type { {byte_size $int_size sdata} {encoding @DW_ATE_signed} {name int} } main_label: subprogram { {name main} {type :$int_label} {external 1 flag} } { Foo_label: class_type { {name Foo} {byte_size 1 sdata} } { doit_label: subprogram { {name doit} {type :$int_label} {accessibility 1 DW_FORM_data1} } { doit_self_label: formal_parameter { {name this} {artificial 1 DW_FORM_flag_present} {type :$Foo_pointer_type} } } Foo_pointer_type: pointer_type { {byte_size $addr_size sdata} {type :$Foo_label} } } foo_label: DW_TAG_variable { {name foo} {type :$Foo_label} } } } } } if { [prepare_for_testing "failed to prepare" ${testfile} \ [list $srcfile $asm_file] {nodebug}] } { return -1 } gdb_test_no_output "set language c++" set psymtabs_p [psymtabs_p] # Verify that the partial symtab for CU "" does # not contain the static partial symbol int, which is defined in the # CU "imported_unit.c". Test-case for PR25646. set test "no static partial symbols in importing unit" if { $psymtabs_p } { gdb_test "main print psymbols" \ [multi_line \ " Depends on 0 other partial symtabs\." \ " Global partial symbols:" \ " `main', function, $hex" \ "" \ ".*"] \ $test } else { unsupported $test } # Verify that there's only one partial symtab for imported_unit.c. Test-case # for PR25700. set test "no duplicate psymtab for imported_unit.c" if { $psymtabs_p } { set line "Partial symtab for source file imported_unit.c" gdb_test_multiple "maint print psymbols" $test { -re -wrap "$line.*$line.*" { fail $gdb_test_name } -re -wrap "$line.*" { pass $gdb_test_name } } } else { unsupported $test } gdb_test "l imported_unit.c:1" \ "1\timported_unit.c: No such file or directory\." gdb_test "info source" "\r\nCurrent source file is imported_unit.c\r\n.*" \ "info source for imported_unit.c" # Sanity check gdb_test "ptype main" "= int \\(void\\)" # Each of these tests caused a segfault prior to fixing BZ 25065. gdb_test "ptype main::Foo" "= class Foo \{\\s+public:\\s+int doit\\(void\\);\\s+\}" gdb_test "ptype main::foo" "= class Foo \{\\s+public:\\s+int doit\\(void\\);\\s+\}"