blob: 5b4977529d8036f3a1086638e20c40a23df21630 [file] [log] [blame]
# Copyright 2025 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 casting some values to a type with dynamic properties.
load_lib dwarf.exp
require dwarf2_support
standard_testfile .c -dw.S
if { [prepare_for_testing "failed to prepare" ${testfile} ${srcfile}] } {
return
}
set asm_file [standard_output_file $srcfile2]
Dwarf::assemble $asm_file {
cu {version 5} {
DW_TAG_compile_unit {
DW_AT_language @DW_LANG_Fortran95
DW_AT_name FOO.F90
DW_AT_comp_dir /tmp
} {
set ptr_size [get_sizeof "void *" -1]
# int
declare_labels int_type_label
int_type_label: DW_TAG_base_type {
DW_AT_name int
DW_AT_byte_size 4 DW_FORM_udata
DW_AT_encoding @DW_ATE_signed
}
# char
declare_labels char_type_label
char_type_label: DW_TAG_base_type {
DW_AT_name char
DW_AT_byte_size 1 DW_FORM_udata
DW_AT_encoding @DW_ATE_signed
}
# uintptr_t
declare_labels uintptr_type_label
uintptr_type_label: DW_TAG_base_type {
DW_AT_name uintptr_t
DW_AT_byte_size $ptr_size DW_FORM_udata
DW_AT_encoding @DW_ATE_signed
}
# Array type.
declare_labels array_type_label
array_type_label: DW_TAG_array_type {
DW_AT_type :$uintptr_type_label
DW_AT_data_location {
DW_OP_push_object_address
DW_OP_deref
} SPECIAL_expr
DW_AT_associated {
DW_OP_push_object_address
DW_OP_deref
DW_OP_lit0
DW_OP_ne
} SPECIAL_expr
} {
DW_TAG_subrange_type {
DW_AT_type :$uintptr_type_label
DW_AT_count {
DW_OP_push_object_address
DW_OP_plus_uconst $ptr_size
DW_OP_deref
} SPECIAL_expr
DW_AT_lower_bound {
DW_OP_push_object_address
DW_OP_plus_uconst [expr {$ptr_size * 2}]
DW_OP_deref
} SPECIAL_expr
}
}
# Structure type with the array type as a member.
declare_labels outer_type_label
set sizeof_outer_type [expr {6 * $ptr_size}]
outer_type_label: DW_TAG_structure_type {
DW_AT_name outer_type
DW_AT_byte_size $sizeof_outer_type DW_FORM_udata
} {
DW_TAG_member {
DW_AT_name assoc
DW_AT_type :$array_type_label
DW_AT_data_member_location 0 DW_FORM_udata
}
DW_TAG_member {
DW_AT_name non_assoc
DW_AT_type :$array_type_label
DW_AT_data_member_location [expr {$ptr_size * 3}] DW_FORM_udata
}
}
# Structure instance.
DW_TAG_variable {
DW_AT_name g_outer
DW_AT_type :$outer_type_label
DW_AT_location {
DW_OP_addr [gdb_target_symbol g_outer_var]
} SPECIAL_expr
}
# char[sizeof(outer_type)]
declare_labels char_array_type_label
char_array_type_label: DW_TAG_array_type {
DW_AT_type :$char_type_label
} {
DW_TAG_subrange_type {
DW_AT_type :$int_type_label
DW_AT_count $sizeof_outer_type DW_FORM_udata
}
}
# A view over "g_outer", typed as a char buffer.
DW_TAG_variable {
DW_AT_name g_outer_as_char_array
DW_AT_type :$char_array_type_label
DW_AT_location {
DW_OP_addr [gdb_target_symbol g_outer_var]
} SPECIAL_expr
}
}
}
}
if { [prepare_for_testing "failed to prepare" ${testfile} \
[list $srcfile $asm_file] {nodebug}] } {
return
}
if {![runto_main]} {
return
}
# Generate a comma-separated sequence of numbers from LOW to HIGH (inclusive).
#
# For example
#
# make_num_sequence 10 12
#
# returns
#
# 10, 11, 12
proc make_num_sequence { low high } {
set nums {}
for {set x $low} {$x <= $high} {incr x} {
lappend nums $x
}
return [join $nums ", "]
}
set one_to_one_hundred [make_num_sequence 1 100]
# Set the language to Fortran, to enable support for dynamic array types.
gdb_test_no_output "set language fortran"
# Verify normal operation.
gdb_test "p g_outer" " = \\( assoc = \\($one_to_one_hundred\\), non_assoc = <not associated> \\)"
gdb_test "p g_outer%assoc" " = \\($one_to_one_hundred\\)"
gdb_test "p g_outer%non_assoc" " = <not associated>"
gdb_test "ptype g_outer%assoc" " = uintptr_t \\(6:105\\)"
gdb_test "ptype g_outer%non_assoc" " = uintptr_t \\(:\\)"
# Cast g_outer_var (a minimal symbol) to outer_type. This would trigger an
# assert.
gdb_test "p (outer_type) g_outer_var" " = \\( assoc = \\($one_to_one_hundred\\), non_assoc = <not associated> \\)"
gdb_test "p ((outer_type) g_outer_var)%assoc" "= \\($one_to_one_hundred\\)"
gdb_test "p ((outer_type) g_outer_var)%non_assoc" " = <not associated>"
gdb_test "ptype ((outer_type) g_outer_var)%assoc" " = uintptr_t \\(6:105\\)"
gdb_test "ptype ((outer_type) g_outer_var)%non_assoc" " = uintptr_t \\(:\\)"
# Cast to outer_type something with the same size as outer_type. The code path
# taken by this would miss resolving the dynamic type, causing an assert when
# trying to print the resulting value.
gdb_test "p (outer_type) g_outer_as_char_array" " = \\( assoc = \\($one_to_one_hundred\\), non_assoc = <not associated> \\)"
gdb_test "p ((outer_type) g_outer_as_char_array)%assoc" "= \\($one_to_one_hundred\\)"
gdb_test "p ((outer_type) g_outer_as_char_array)%non_assoc" " = <not associated>"
gdb_test "ptype ((outer_type) g_outer_as_char_array)%assoc" " = uintptr_t \\(6:105\\)"
gdb_test "ptype ((outer_type) g_outer_as_char_array)%non_assoc" " = uintptr_t \\(:\\)"