mirror of
https://github.com/bminor/binutils-gdb.git
synced 2025-12-25 08:47:28 +00:00
Currently the DWARF assembler implements manual parsing for location expressions. With a recent refactoring, this lead to the use of [subst] in a number of places. Following the same logic as the DW_AT_* change, this patch changes location expressions to simply be nested Tcl code. This avoids the need for subst and also allows more complex logic, should that ever be needed. Bug: https://sourceware.org/bugzilla/show_bug.cgi?id=33402
177 lines
5.0 KiB
Plaintext
177 lines
5.0 KiB
Plaintext
# Copyright (C) 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/>.
|
|
|
|
# Tests for gdb.printing.make_visualizer; specifically that the
|
|
# optimized-out and synthetic pointer cases work properly.
|
|
|
|
load_lib dwarf.exp
|
|
load_lib gdb-python.exp
|
|
|
|
require dwarf2_support
|
|
require allow_python_tests
|
|
|
|
# Use a simple plain-"main" source file.
|
|
standard_testfile py-progspace.c -dw.S
|
|
|
|
if {[prepare_for_testing "failed to prepare" $testfile $srcfile {debug}]} {
|
|
return
|
|
}
|
|
|
|
if {![runto_main]} {
|
|
return
|
|
}
|
|
|
|
# Get size to create fake DWARF for the test.
|
|
set ptr_size [get_sizeof "void *" 96]
|
|
|
|
set asm_file [standard_output_file ${srcfile2}]
|
|
Dwarf::assemble $asm_file {
|
|
cu {} {
|
|
compile_unit {} {
|
|
declare_labels i32_type i32_array \
|
|
struct_label variable_label pointer_label
|
|
|
|
i32_type: DW_TAG_base_type {
|
|
DW_AT_name "int32_t"
|
|
DW_AT_encoding @DW_ATE_signed
|
|
DW_AT_byte_size 4 DW_FORM_sdata
|
|
}
|
|
|
|
i32_array: DW_TAG_array_type {
|
|
DW_AT_name array_type
|
|
DW_AT_type :$i32_type
|
|
} {
|
|
DW_TAG_subrange_type {
|
|
DW_AT_type :$i32_type
|
|
DW_AT_lower_bound 0 DW_FORM_data1
|
|
DW_AT_upper_bound 3 DW_FORM_data1
|
|
}
|
|
}
|
|
|
|
# Used for testing optimized-out elements of an array.
|
|
DW_TAG_variable {
|
|
DW_AT_name i32_noptr
|
|
DW_AT_type :$i32_array
|
|
DW_AT_location {
|
|
DW_OP_constu 1779823878
|
|
DW_OP_stack_value
|
|
DW_OP_piece 4
|
|
} SPECIAL_expr
|
|
}
|
|
|
|
struct_label: DW_TAG_structure_type {
|
|
DW_AT_name i32_noptr
|
|
DW_AT_byte_size 4 DW_FORM_sdata
|
|
} {
|
|
DW_TAG_member {
|
|
DW_AT_name f
|
|
DW_AT_type :$i32_type
|
|
DW_AT_data_member_location 0 DW_FORM_data1
|
|
}
|
|
}
|
|
|
|
pointer_label: DW_TAG_pointer_type {
|
|
DW_AT_byte_size $::ptr_size DW_FORM_sdata
|
|
DW_AT_type :$struct_label
|
|
}
|
|
|
|
variable_label: DW_TAG_variable {
|
|
DW_AT_name v
|
|
DW_AT_location {
|
|
DW_OP_implicit_value 0x1 0x1 0x1 0x1
|
|
} SPECIAL_expr
|
|
DW_AT_type :$struct_label
|
|
}
|
|
|
|
# Used for testing synthetic pointers.
|
|
DW_TAG_variable {
|
|
DW_AT_name synthptr
|
|
DW_AT_location {
|
|
DW_OP_GNU_implicit_pointer $variable_label 0
|
|
} SPECIAL_expr
|
|
DW_AT_type :$pointer_label
|
|
}
|
|
|
|
# Optimized-out pointer.
|
|
DW_TAG_variable {
|
|
DW_AT_name optoutptr
|
|
DW_AT_location { } SPECIAL_expr
|
|
DW_AT_type :$pointer_label
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if {[prepare_for_testing "failed to prepare" ${testfile} \
|
|
[list $srcfile $asm_file] {nodebug}]} {
|
|
return
|
|
}
|
|
|
|
# Need a frame to evaluate a synthetic pointer.
|
|
if {![runto_main]} {
|
|
return
|
|
}
|
|
|
|
gdb_test_no_output "python import gdb"
|
|
gdb_test_no_output "python import gdb.printing"
|
|
|
|
gdb_test_no_output "python val = gdb.parse_and_eval('i32_noptr')" \
|
|
"fetch i32_noptr"
|
|
gdb_test_no_output "python vz = gdb.printing.make_visualizer(val)" \
|
|
"create i32_noptr visualizer"
|
|
gdb_test "python print(isinstance(vz, gdb.printing.NoOpArrayPrinter))" \
|
|
True \
|
|
"i32_noptr uses array printer"
|
|
|
|
gdb_test_no_output "python vz1 = gdb.printing.make_visualizer(val\[0\])" \
|
|
"create visualizer for valid element"
|
|
gdb_test "python print(isinstance(vz1, gdb.printing.NoOpScalarPrinter))" \
|
|
True \
|
|
"valid element uses scalar printer"
|
|
gdb_test "python print(vz1.to_string())" \
|
|
1779823878 \
|
|
"string form of valid element"
|
|
|
|
gdb_test_no_output "python vz2 = gdb.printing.make_visualizer(val\[1\])" \
|
|
"create visualizer for optimized-out element"
|
|
gdb_test "python print(isinstance(vz2, gdb.printing.NoOpScalarPrinter))" \
|
|
True \
|
|
"optimized-out element uses scalar printer"
|
|
gdb_test "python print(vz2.to_string())" \
|
|
"<optimized out>" \
|
|
"string form of optimized-out element"
|
|
|
|
gdb_test_no_output "python val2 = gdb.parse_and_eval('synthptr')" \
|
|
"fetch synthetic pointer"
|
|
gdb_test_no_output "python vzv2 = gdb.printing.make_visualizer(val2)" \
|
|
"create synthetic pointer visualizer"
|
|
gdb_test "python print(isinstance(vzv2, gdb.printing.NoOpPointerReferencePrinter))" \
|
|
True \
|
|
"synthetic pointer uses pointer printer"
|
|
gdb_test "python print(vzv2.child(0)\[1\])" \
|
|
"{f = 16843009}" \
|
|
"child of synthetic pointer"
|
|
|
|
gdb_test_no_output "python val3 = gdb.parse_and_eval('optoutptr')" \
|
|
"fetch optimized-out pointer"
|
|
gdb_test_no_output "python vzv3 = gdb.printing.make_visualizer(val3)" \
|
|
"create optimized-out pointer visualizer"
|
|
gdb_test "python print(isinstance(vzv3, gdb.printing.NoOpScalarPrinter))" \
|
|
True \
|
|
"optimized-out pointer uses scalar printer"
|
|
gdb_test "python print(vzv3.to_string())" \
|
|
"<optimized out>" \
|
|
"string representation of optimized-out pointer"
|