blob: 734007ea680af4270950a784e5fc99c332ac064f [file] [log] [blame]
# Copyright 2017-2021 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 DWARF operation that allow adding byte and bit offset to any
# location description.
#
# In particular, the test uses memory and register location
# descriptions (both as standalone and parts of the composite
# location), and applies different byte and bit offsets to them.
load_lib dwarf.exp
# This test can only be run on targets which support DWARF-2 and use gas.
if {![dwarf2_support]} {
return 0
}
# 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 var-access.c ${gdb_test_file_name}-dw.S
# Make some DWARF for the test.
set asm_file [standard_output_file $srcfile2]
Dwarf::assemble $asm_file {
global dwarf_regnum regname srcdir subdir srcfile
set buf_var [gdb_target_symbol buf]
set main_result [function_range main ${srcdir}/${subdir}/${srcfile}]
set main_start [lindex $main_result 0]
set main_length [lindex $main_result 1]
cu {} {
DW_TAG_compile_unit {
{DW_AT_name var-access.c}
{DW_AT_comp_dir /tmp}
} {
declare_labels char_type_label int_type_label
declare_labels array_size_4_type_label array_size_8_type_label
# define char type.
char_type_label: DW_TAG_base_type {
{DW_AT_name "char"}
{DW_AT_encoding @DW_ATE_signed}
{DW_AT_byte_size 1 DW_FORM_sdata}
}
# define int type.
int_type_label: DW_TAG_base_type {
{DW_AT_name "int"}
{DW_AT_encoding @DW_ATE_signed}
{DW_AT_byte_size 4 DW_FORM_sdata}
}
# define 4 byte size array type.
array_size_4_type_label: DW_TAG_array_type {
{DW_AT_type :$char_type_label}
} {
DW_TAG_subrange_type {
{DW_AT_type :$int_type_label}
{DW_AT_upper_bound 3 DW_FORM_udata}
}
}
# define 8 byte size array type.
array_size_8_type_label: DW_TAG_array_type {
{DW_AT_type :$char_type_label}
} {
DW_TAG_subrange_type {
{DW_AT_type :$int_type_label}
{DW_AT_upper_bound 7 DW_FORM_udata}
}
}
DW_TAG_subprogram {
{DW_AT_name main}
{DW_AT_low_pc $main_start addr}
{DW_AT_high_pc $main_length data8}
} {
# define original buf variable.
DW_TAG_variable {
{DW_AT_name buf}
{DW_AT_type :$array_size_4_type_label}
{DW_AT_location {
DW_OP_addr $buf_var
} SPECIAL_expr}
}
# defined a variable located in
# a third byte of the buf variable.
DW_TAG_variable {
{DW_AT_name buf_byte_3}
{DW_AT_type :$char_type_label}
{DW_AT_location {
DW_OP_addr $buf_var
DW_OP_LLVM_offset_constu 2
} SPECIAL_expr}
{external 1 flag}
}
# defined a variable located in a second byte
# of the buf variable with a bit offset of one.
DW_TAG_variable {
{DW_AT_name buf_byte_2_bit_1}
{DW_AT_type :$char_type_label}
{DW_AT_location {
DW_OP_addr $buf_var
DW_OP_lit9
DW_OP_LLVM_bit_offset
} SPECIAL_expr}
{external 1 flag}
}
# defined a variable located in a
# third byte of the REGNAME register.
DW_TAG_variable {
{DW_AT_name reg_byte_3}
{DW_AT_type :$char_type_label}
{DW_AT_location {
DW_OP_regx $dwarf_regnum
DW_OP_lit2
DW_OP_LLVM_offset
} SPECIAL_expr}
{external 1 flag}
}
# defined a variable located in a second byte of
# the REGNAME register with a bit offset of one.
DW_TAG_variable {
{DW_AT_name reg_byte_2_bit_1}
{DW_AT_type :$char_type_label}
{DW_AT_location {
DW_OP_regx $dwarf_regnum
DW_OP_lit1
DW_OP_LLVM_offset
DW_OP_lit1
DW_OP_LLVM_bit_offset
} SPECIAL_expr}
{external 1 flag}
}
# Define an array variable spread in different
# pieces of buf variable and REGNAME register.
DW_TAG_variable {
{DW_AT_name mix_array}
{DW_AT_type :$array_size_8_type_label}
{DW_AT_location {
# a byte piece located in a
# fourth byte of the buf variable.
DW_OP_addr $buf_var
DW_OP_LLVM_offset_constu 3
DW_OP_piece 0x1
# a byte piece located in a
# third byte of the buf variable.
DW_OP_addr $buf_var
DW_OP_lit2
DW_OP_LLVM_offset
DW_OP_piece 0x1
# a byte piece located in a second byte of
# the buf variable with a bit offset of one.
DW_OP_addr $buf_var
DW_OP_lit1
DW_OP_LLVM_offset
DW_OP_lit1
DW_OP_LLVM_bit_offset
DW_OP_piece 0x1
# a four bit piece located in a first byte
# of the buf variable with a bit offset of one.
DW_OP_addr $buf_var
DW_OP_LLVM_offset_constu 0
DW_OP_bit_piece 0x4 0x1
# a four bit piece located in a first byte of
# the buf variable with a bit offset of eight.
DW_OP_addr $buf_var
DW_OP_lit1
DW_OP_LLVM_bit_offset
DW_OP_LLVM_offset_constu 0
DW_OP_bit_piece 0x4 0x7
# a byte piece located in a fourth
# byte of the REGNAME register.
DW_OP_regx $dwarf_regnum
DW_OP_LLVM_offset_constu 3
DW_OP_piece 0x1
# a byte piece located in a third
# byte of the REGNAME register.
DW_OP_regx $dwarf_regnum
DW_OP_lit2
DW_OP_LLVM_offset
DW_OP_piece 0x1
# a byte piece located in a second byte of the
# REGNAME register with a bit offset of one.
DW_OP_regx $dwarf_regnum
DW_OP_lit1
DW_OP_LLVM_offset
DW_OP_lit1
DW_OP_LLVM_bit_offset
DW_OP_piece 0x1
# a four bit piece located in a first byte of
# the REGNAME register with a bit offset of one.
DW_OP_regx $dwarf_regnum
DW_OP_LLVM_offset_constu 0
DW_OP_bit_piece 0x4 0x1
# a four bit piece located in a first byte of the
# REGNAME register with a bit offset of eight.
DW_OP_regx $dwarf_regnum
DW_OP_lit1
DW_OP_LLVM_bit_offset
DW_OP_LLVM_offset_constu 0
DW_OP_bit_piece 0x4 0x7
} SPECIAL_expr}
{external 1 flag}
}
}
}
}
}
if { [prepare_for_testing ${testfile}.exp ${testfile} \
[list $srcfile $asm_file] {nodebug}] } {
return -1
}
if ![runto_main] {
return -1
}
# Determine byte order.
set endian [get_endianness]
if { $endian != "little" } then {
verbose "Skipping ${gdb_test_file_name}."
return
}
gdb_test_no_output "set var \$$regname = 0x04030201" "init reg"
gdb_test_no_output "set var *\(\(unsigned int *\) buf\) = 0x04030201" \
"init buf"
gdb_test "print/x buf_byte_3" " = 0x3" "buf_byte_3 == 0x3"
gdb_test "print/x buf_byte_2_bit_1" " = 0x81" \
"print buf_byte_2_bit_1"
gdb_test "print/x reg_byte_3" " = 0x3" "reg_byte_3 == 0x3"
gdb_test "print/x reg_byte_2_bit_1" " = 0x81" \
"print reg_byte_2_bit_1"
gdb_test_no_output "set var buf_byte_3 = 0x4" "init buf_byte_3 to 0x4"
gdb_test "print/x buf_byte_3" " = 0x4" "buf_byte_3 == 0x4"
gdb_test_no_output "set var buf_byte_2_bit_1 = 0x4" \
"init buf_byte_2_bit_1 to 0x4"
gdb_test "print/x buf_byte_2_bit_1" " = 0x4" "buf_byte_2_bit_1 == 0x4"
gdb_test "print/x buf" " = \\{0x1, 0x8, 0x4, 0x4\\}" "buf print"
gdb_test_no_output "set var reg_byte_3 = 0x4" "init reg_byte_3 to 0x4"
gdb_test "print/x reg_byte_3" " = 0x4" "reg_byte_3 == 0x4"
gdb_test_no_output "set var reg_byte_2_bit_1 = 0x4" \
"init reg_byte_2_bit_1 to 0x4"
gdb_test "print/x reg_byte_2_bit_1" " = 0x4" "reg_byte_2_bit_1 == 0x4"
gdb_test "print/x \$$regname" " = 0x4040801" "\$$regname print"
gdb_test_no_output "set var \$$regname = 0x04030201" "reset reg"
gdb_test_no_output "set var *\(\(unsigned int *\) buf\) = 0x04030201" \
"reset buf"
gdb_test "print/x mix_array" \
" = \\{0x4, 0x3, 0x81, 0x20, 0x4, 0x3, 0x81, 0x20\\}" \
"mix_array print"
gdb_test_no_output "set var mix_array\[1\] = 0x4" \
"set mix_array second byte"
gdb_test_no_output "set var mix_array\[2\] = 0x4" \
"set mix_array third byte"
gdb_test_no_output "set var mix_array\[5\] = 0x4" \
"set mix_array fifth byte"
gdb_test_no_output "set var mix_array\[6\] = 0x4" \
"set mix_array sixth byte"
gdb_test "print/x mix_array" \
" = \\{0x4, 0x4, 0x4, 0x80, 0x4, 0x4, 0x4, 0x80\\}" \
"mix_array second print"
gdb_test "print/x buf" " = \\{0x1, 0x8, 0x4, 0x4\\}" "buf second print"
gdb_test "print/x \$$regname" " = 0x4040801" "\$$regname second print"