Files
binutils-gdb/gdb/testsuite/gdb.python/make-visualizer.exp
Tom Tromey c44edec047 Make location expressions be code in DWARF assembler
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
2025-10-06 09:30:00 -06:00

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"