
This is similar to the previous patch, but for gdb_protocol_is_remote. gdb_is_target_remote and its MI cousin mi_is_target_remote, use "maint print target-stack", which is unnecessary when checking whether gdb_protocol is "remote" or "extended-remote" would do. Checking gdb_protocol is more efficient, and can be done before starting GDB and running to main, unlike gdb_is_target_remote/mi_is_target_remote. This adds a new gdb_protocol_is_remote procedure, and uses it in place of gdb_is_target_remote/mi_is_target_remote throughout. There are no uses of gdb_is_target_remote/mi_is_target_remote left after this. Those will be eliminated in a following patch. In some spots, we no longer need to defer the check until after starting GDB, so the patch adjusts accordingly. Change-Id: I90267c132f942f63426f46dbca0b77dbfdf9d2ef Approved-By: Tom Tromey <tom@tromey.com>
186 lines
6.3 KiB
Tcl
186 lines
6.3 KiB
Tcl
# Copyright (C) 2023-2024 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/>.
|
|
|
|
# Exercise restoring SME/TPIDR2 state from a signal frame.
|
|
|
|
load_lib aarch64-scalable.exp
|
|
|
|
require is_aarch64_target
|
|
require allow_aarch64_sve_tests
|
|
require allow_aarch64_sme_tests
|
|
|
|
# Remote targets can't communicate vector length (vl or svl) changes
|
|
# to GDB via the RSP.
|
|
require !gdb_protocol_is_remote
|
|
|
|
#
|
|
# Validate the state of registers in the signal frame for various states.
|
|
#
|
|
proc test_sme_registers_sigframe { id_start id_end } {
|
|
|
|
set compile_flags {"debug" "macros"}
|
|
lappend compile_flags "additional_flags=-march=armv8.5-a+sve"
|
|
lappend compile_flags "additional_flags=-DID_START=${id_start}"
|
|
lappend compile_flags "additional_flags=-DID_END=${id_end}"
|
|
|
|
standard_testfile ${::srcdir}/${::subdir}/aarch64-sme-regs-sigframe.c
|
|
set executable "${::testfile}-${id_start}-${id_end}"
|
|
if {[prepare_for_testing "failed to prepare" ${executable} ${::srcfile} ${compile_flags}]} {
|
|
return -1
|
|
}
|
|
set binfile [standard_output_file ${executable}]
|
|
|
|
if ![runto_main] {
|
|
untested "could not run to main"
|
|
return -1
|
|
}
|
|
|
|
set sigill_breakpoint "stop before SIGILL"
|
|
set handler_breakpoint "handler"
|
|
gdb_breakpoint [gdb_get_line_number $sigill_breakpoint]
|
|
gdb_breakpoint [gdb_get_line_number $handler_breakpoint]
|
|
|
|
for {set id $id_start} {$id <= $id_end} {incr id} {
|
|
set state [test_id_to_state $id]
|
|
set vl [test_id_to_vl $id]
|
|
set svl [test_id_to_svl $id]
|
|
|
|
set skip_unsupported 0
|
|
if {![aarch64_supports_sve_vl $vl]
|
|
|| ![aarch64_supports_sme_svl $svl]} {
|
|
# We have a vector length or streaming vector length that
|
|
# is not supported by this target. Skip to the next iteration
|
|
# since it is no use running tests for an unsupported vector
|
|
# length.
|
|
if {![aarch64_supports_sve_vl $vl]} {
|
|
verbose -log "SVE vector length $vl not supported."
|
|
} elseif {![aarch64_supports_sme_svl $svl]} {
|
|
verbose -log "SME streaming vector length $svl not supported."
|
|
}
|
|
verbose -log "Skipping test."
|
|
set skip_unsupported 1
|
|
}
|
|
|
|
with_test_prefix "state=${state} vl=${vl} svl=${svl}" {
|
|
|
|
# If the SVE or SME vector length is not supported, just skip
|
|
# these next tests.
|
|
if {$skip_unsupported} {
|
|
untested "unsupported configuration on target"
|
|
continue
|
|
}
|
|
|
|
# Run the program until it has adjusted the svl.
|
|
if [gdb_continue_to_breakpoint $sigill_breakpoint] {
|
|
return -1
|
|
}
|
|
|
|
# Check SVG to make sure it is correct
|
|
set expected_svg [expr $svl / 8]
|
|
gdb_test "print \$svg" "= ${expected_svg}"
|
|
|
|
# Check the size of ZA.
|
|
set expected_za_size [expr $svl * $svl]
|
|
gdb_test "print sizeof \$za" " = $expected_za_size"
|
|
|
|
# Check the value of SVCR.
|
|
gdb_test "print \$svcr" [get_svcr_value $state] "svcr before signal"
|
|
|
|
# Handle SME ZA and SME2 initialization and state.
|
|
set byte 0
|
|
set sme2_byte 0
|
|
if { $state == "za" || $state == "za_ssve" } {
|
|
set byte 170
|
|
set sme2_byte 255
|
|
}
|
|
|
|
# Set the expected ZA pattern.
|
|
set za_pattern [string_to_regexp [2d_array_value_pattern $byte $svl $svl]]
|
|
|
|
# Handle SVE/SSVE initialization and state.
|
|
set sve_vl $svl
|
|
if { $state == "ssve" || $state == "za_ssve" } {
|
|
# SVE state comes from SSVE.
|
|
set sve_vl $svl
|
|
} else {
|
|
# SVE state comes from regular SVE.
|
|
set sve_vl $vl
|
|
}
|
|
|
|
# Initialize the SVE state.
|
|
set sve_pattern [string_to_regexp [sve_value_pattern $state $sve_vl 85 255]]
|
|
for {set row 0} {$row < 32} {incr row} {
|
|
set register_name "\$z${row}\.b\.u"
|
|
gdb_test "print sizeof $register_name" " = $sve_vl" "size of $register_name"
|
|
gdb_test "print $register_name" $sve_pattern "read back from $register_name"
|
|
}
|
|
|
|
# Print ZA to check its value.
|
|
gdb_test "print \$za" $za_pattern "read back from za"
|
|
|
|
# Test TPIDR2 restore from signal frame as well.
|
|
gdb_test_no_output "set \$tpidr2=0x0102030405060708"
|
|
|
|
# Run to the illegal instruction.
|
|
if [gdb_test "continue" "Continuing\.\r\n\r\nProgram received signal SIGILL, Illegal instruction\..*in main.*"] {
|
|
return
|
|
}
|
|
|
|
# Skip the illegal instruction. The signal handler will be called after we continue.
|
|
gdb_test_no_output "set \$pc=\$pc+4"
|
|
# Continue to the signal handler.
|
|
if [gdb_continue_to_breakpoint $handler_breakpoint] {
|
|
return -1
|
|
}
|
|
|
|
# Modify TPIDR2 so it is different from its value past the signal
|
|
# frame.
|
|
gdb_test_no_output "set \$tpidr2 = 0x0"
|
|
|
|
# Select the frame that contains "main".
|
|
gdb_test "frame 2" "#2.* main \\\(.*\\\) at.*"
|
|
|
|
for {set row 0} {$row < 32} {incr row} {
|
|
set register_name "\$z${row}\.b\.u"
|
|
gdb_test "print sizeof $register_name" " = $sve_vl" "size of $register_name in the signal frame"
|
|
gdb_test "print $register_name" $sve_pattern "$register_name contents from signal frame"
|
|
}
|
|
|
|
# Check the size of ZA in the signal frame.
|
|
set expected_za_size [expr $svl * $svl]
|
|
gdb_test "print sizeof \$za" " = $expected_za_size" "size of za in signal frame"
|
|
|
|
# Check the value of SVCR in the signal frame.
|
|
gdb_test "print \$svcr" [get_svcr_value $state] "svcr from signal frame"
|
|
|
|
# Check the value of ZA in the signal frame.
|
|
gdb_test "print \$za" $za_pattern "za contents from signal frame"
|
|
|
|
# Check the value of TPIDR2 in the signal frame.
|
|
gdb_test "print/x \$tpidr2" " = 0x102030405060708" "tpidr2 contents from signal frame"
|
|
|
|
# Check the value of SME2 ZT0 in the signal frame.
|
|
if [is_sme2_available] {
|
|
# The target supports SME2.
|
|
set zt_size 64
|
|
gdb_test "print sizeof \$zt0" " = $zt_size"
|
|
set zt_pattern [string_to_regexp [1d_array_value_pattern $sme2_byte $zt_size]]
|
|
gdb_test "print \$zt0" " = $zt_pattern" "zt contents from signal frame"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
test_sme_registers_sigframe $id_start $id_end
|