
We recently realized that symbol_needs_eval_fail.exp and symbol_needs_eval_timeout.exp invalidly dereference an int (4 bytes on x86_64) by reading 8 bytes (the size of a pointer). Here how it goes: In gdb/testsuite/gdb.dwarf2/symbol_needs_eval.c a global variable is defined: int exec_mask = 1; and later both tests build some DWARF using the assembler doing: set exec_mask_var [gdb_target_symbol exec_mask] ... DW_TAG_variable { {DW_AT_name a} {DW_AT_type :$int_type_label} {DW_AT_location { DW_OP_addr $exec_mask_var DW_OP_deref ... } } The definition of the DW_OP_deref (from Dwarf5 2.5.1.3 Stack Operations) says that "The size of the data retrieved from the dereferenced address is the size of an address on the target machine." On x86_64, the size of an int is 4 while the size of an address is 8. The result is that when evaluating this expression, the debugger reads outside of the `a` variable. Fix this by using `DW_OP_deref_size $int_size` instead. To achieve this, this patch adds the necessary steps so we can figure out what `sizeof(int)` evaluates to for the current target. While at it, also change the definition of the int type in the assembled DWARF information so we use the actual target's size for an int instead of the literal 4. Tested on x86_64 Linux. Approved-By: Tom Tromey <tom@tromey.com>
135 lines
3.3 KiB
Text
135 lines
3.3 KiB
Text
# Copyright 2017-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 <http://www.gnu.org/licenses/>.
|
|
|
|
# Test the symbol needs check mechanism if it assumes that faking
|
|
# reads from a target is a safe thing to do.
|
|
#
|
|
# In particular, the test uses a relative branch DWARF operation to
|
|
# potentially cause an infinite loop, if the target reads are indeed
|
|
# faked.
|
|
|
|
load_lib dwarf.exp
|
|
|
|
# This test can only be run on targets which support DWARF-2 and use gas.
|
|
require dwarf2_support
|
|
|
|
# Choose suitable integer registers for the test.
|
|
|
|
set dwarf_regnum 0
|
|
|
|
if { [is_aarch64_target] } {
|
|
set regname x0
|
|
} elseif { [is_aarch32_target]
|
|
|| [istarget "s390*-*-*" ]
|
|
|| [istarget "powerpc*-*-*"]
|
|
|| [istarget "rs6000*-*-aix*"] } {
|
|
set regname r0
|
|
} elseif { [is_x86_like_target] } {
|
|
set regname eax
|
|
} elseif { [is_amd64_regs_target] } {
|
|
set regname rax
|
|
} else {
|
|
verbose "Skipping ${gdb_test_file_name}."
|
|
return
|
|
}
|
|
|
|
standard_testfile symbol_needs_eval.c ${gdb_test_file_name}-dw.S
|
|
|
|
if [prepare_for_testing "failed to prepare" $testfile $srcfile {debug}] {
|
|
return
|
|
}
|
|
|
|
set int_size [get_sizeof "int" -1]
|
|
|
|
# Make some DWARF for the test.
|
|
|
|
set asm_file [standard_output_file $srcfile2]
|
|
Dwarf::assemble $asm_file {
|
|
global dwarf_regnum regname int_size
|
|
|
|
set exec_mask_var [gdb_target_symbol exec_mask]
|
|
|
|
cu {} {
|
|
DW_TAG_compile_unit {
|
|
{DW_AT_name symbol_needs_eval.c}
|
|
{DW_AT_comp_dir /tmp}
|
|
} {
|
|
declare_labels int_type_label
|
|
|
|
# define int type
|
|
int_type_label: DW_TAG_base_type {
|
|
{DW_AT_name "int"}
|
|
{DW_AT_encoding @DW_ATE_signed}
|
|
{DW_AT_byte_size $int_size DW_FORM_sdata}
|
|
}
|
|
|
|
# add info for variable exec_mask
|
|
DW_TAG_variable {
|
|
{DW_AT_name exec_mask}
|
|
{DW_AT_type :$int_type_label}
|
|
{DW_AT_location {
|
|
DW_OP_addr $exec_mask_var
|
|
} SPECIAL_expr}
|
|
{external 1 flag}
|
|
}
|
|
|
|
# add info for subprogram main
|
|
DW_TAG_subprogram {
|
|
{MACRO_AT_func { main }}
|
|
{DW_AT_frame_base {
|
|
DW_OP_regx $dwarf_regnum
|
|
} SPECIAL_expr}
|
|
} {
|
|
# define artificial variable a
|
|
DW_TAG_variable {
|
|
{DW_AT_name a}
|
|
{DW_AT_type :$int_type_label}
|
|
{DW_AT_location {
|
|
DW_OP_lit1
|
|
DW_OP_addr $exec_mask_var
|
|
DW_OP_deref_size $int_size
|
|
|
|
# jump to DW_OP_fbreg
|
|
DW_OP_skip 4
|
|
DW_OP_drop
|
|
DW_OP_fbreg 0
|
|
DW_OP_dup
|
|
DW_OP_lit0
|
|
DW_OP_eq
|
|
|
|
# conditional jump to DW_OP_drop
|
|
DW_OP_bra -9
|
|
DW_OP_stack_value
|
|
} SPECIAL_expr}
|
|
{external 1 flag}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if { [prepare_for_testing ${testfile}.exp ${testfile} \
|
|
[list $srcfile $asm_file] {nodebug}] } {
|
|
return -1
|
|
}
|
|
|
|
if ![runto_main] {
|
|
return -1
|
|
}
|
|
|
|
gdb_test_no_output "set var \$$regname = 2" "init reg to 2"
|
|
gdb_test_no_output "set var exec_mask = 0" "init exec_mask to 0"
|
|
|
|
gdb_test "print/d a" " = 2" "a == 2"
|