Home | History | Annotate | Line # | Download | only in gdb.arch
      1 # Copyright (C) 2023-2024 Free Software Foundation, Inc.
      2 
      3 # This program is free software; you can redistribute it and/or modify
      4 # it under the terms of the GNU General Public License as published by
      5 # the Free Software Foundation; either version 3 of the License, or
      6 # (at your option) any later version.
      7 #
      8 # This program is distributed in the hope that it will be useful,
      9 # but WITHOUT ANY WARRANTY; without even the implied warranty of
     10 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
     11 # GNU General Public License for more details.
     12 #
     13 # You should have received a copy of the GNU General Public License
     14 # along with this program.  If not, see <http://www.gnu.org/licenses/>.
     15 
     16 # Exercise restoring SME/TPIDR2 state from a signal frame.
     17 
     18 load_lib aarch64-scalable.exp
     19 
     20 require is_aarch64_target
     21 require allow_aarch64_sve_tests
     22 require allow_aarch64_sme_tests
     23 
     24 # Remote targets can't communicate vector length (vl or svl) changes
     25 # to GDB via the RSP.
     26 require !gdb_protocol_is_remote
     27 
     28 #
     29 # Validate the state of registers in the signal frame for various states.
     30 #
     31 proc test_sme_registers_sigframe { id_start id_end } {
     32 
     33     set compile_flags {"debug" "macros"}
     34     lappend compile_flags "additional_flags=-march=armv8.5-a+sve"
     35     lappend compile_flags "additional_flags=-DID_START=${id_start}"
     36     lappend compile_flags "additional_flags=-DID_END=${id_end}"
     37 
     38     standard_testfile ${::srcdir}/${::subdir}/aarch64-sme-regs-sigframe.c
     39     set executable "${::testfile}-${id_start}-${id_end}"
     40     if {[prepare_for_testing "failed to prepare" ${executable} ${::srcfile} ${compile_flags}]} {
     41 	return -1
     42     }
     43     set binfile [standard_output_file ${executable}]
     44 
     45     if ![runto_main] {
     46 	untested "could not run to main"
     47 	return -1
     48     }
     49 
     50     set sigill_breakpoint "stop before SIGILL"
     51     set handler_breakpoint "handler"
     52     gdb_breakpoint [gdb_get_line_number $sigill_breakpoint]
     53     gdb_breakpoint [gdb_get_line_number $handler_breakpoint]
     54 
     55     for {set id $id_start} {$id <= $id_end} {incr id} {
     56 	set state [test_id_to_state $id]
     57 	set vl [test_id_to_vl $id]
     58 	set svl [test_id_to_svl $id]
     59 
     60 	set skip_unsupported 0
     61 	if {![aarch64_supports_sve_vl $vl]
     62 	    || ![aarch64_supports_sme_svl $svl]} {
     63 	    # We have a vector length or streaming vector length that
     64 	    # is not supported by this target.  Skip to the next iteration
     65 	    # since it is no use running tests for an unsupported vector
     66 	    # length.
     67 	    if {![aarch64_supports_sve_vl $vl]} {
     68 		verbose -log "SVE vector length $vl not supported."
     69 	    } elseif {![aarch64_supports_sme_svl $svl]} {
     70 		verbose -log "SME streaming vector length $svl not supported."
     71 	    }
     72 	    verbose -log "Skipping test."
     73 	    set skip_unsupported 1
     74 	}
     75 
     76 	with_test_prefix "state=${state} vl=${vl} svl=${svl}" {
     77 
     78 	    # If the SVE or SME vector length is not supported, just skip
     79 	    # these next tests.
     80 	    if {$skip_unsupported} {
     81 		untested "unsupported configuration on target"
     82 		continue
     83 	    }
     84 
     85 	    # Run the program until it has adjusted the svl.
     86 	    if [gdb_continue_to_breakpoint $sigill_breakpoint] {
     87 		return -1
     88 	    }
     89 
     90 	    # Check SVG to make sure it is correct
     91 	    set expected_svg [expr $svl / 8]
     92 	    gdb_test "print \$svg" "= ${expected_svg}"
     93 
     94 	    # Check the size of ZA.
     95 	    set expected_za_size [expr $svl * $svl]
     96 	    gdb_test "print sizeof \$za" " = $expected_za_size"
     97 
     98 	    # Check the value of SVCR.
     99 	    gdb_test "print \$svcr" [get_svcr_value $state] "svcr before signal"
    100 
    101 	    # Handle SME ZA and SME2 initialization and state.
    102 	    set byte 0
    103 	    set sme2_byte 0
    104 	    if { $state == "za" || $state == "za_ssve" } {
    105 		set byte 170
    106 		set sme2_byte 255
    107 	    }
    108 
    109 	    # Set the expected ZA pattern.
    110 	    set za_pattern [string_to_regexp [2d_array_value_pattern $byte $svl $svl]]
    111 
    112 	    # Handle SVE/SSVE initialization and state.
    113 	    set sve_vl $svl
    114 	    if { $state == "ssve" || $state == "za_ssve" } {
    115 		# SVE state comes from SSVE.
    116 		set sve_vl $svl
    117 	    } else {
    118 		# SVE state comes from regular SVE.
    119 		set sve_vl $vl
    120 	    }
    121 
    122 	    # Initialize the SVE state.
    123 	    set sve_pattern [string_to_regexp [sve_value_pattern $state $sve_vl 85 255]]
    124 	    for {set row 0} {$row < 32} {incr row} {
    125 		set register_name "\$z${row}\.b\.u"
    126 		gdb_test "print sizeof $register_name" " = $sve_vl" "size of $register_name"
    127 		gdb_test "print $register_name" $sve_pattern "read back from $register_name"
    128 	    }
    129 
    130 	    # Print ZA to check its value.
    131 	    gdb_test "print \$za" $za_pattern "read back from za"
    132 
    133 	    # Test TPIDR2 restore from signal frame as well.
    134 	    gdb_test_no_output "set \$tpidr2=0x0102030405060708"
    135 
    136 	    # Run to the illegal instruction.
    137 	    if [gdb_test "continue" "Continuing\.\r\n\r\nProgram received signal SIGILL, Illegal instruction\..*in main.*"] {
    138 		return
    139 	    }
    140 
    141 	    # Skip the illegal instruction.  The signal handler will be called after we continue.
    142 	    gdb_test_no_output "set \$pc=\$pc+4"
    143 	    # Continue to the signal handler.
    144 	    if [gdb_continue_to_breakpoint $handler_breakpoint] {
    145 		return -1
    146 	    }
    147 
    148 	    # Modify TPIDR2 so it is different from its value past the signal
    149 	    # frame.
    150 	    gdb_test_no_output "set \$tpidr2 = 0x0"
    151 
    152 	    # Select the frame that contains "main".
    153 	    gdb_test "frame 2" "#2.* main \\\(.*\\\) at.*"
    154 
    155 	    for {set row 0} {$row < 32} {incr row} {
    156 		set register_name "\$z${row}\.b\.u"
    157 		gdb_test "print sizeof $register_name" " = $sve_vl" "size of $register_name in the signal frame"
    158 		gdb_test "print $register_name" $sve_pattern "$register_name contents from signal frame"
    159 	    }
    160 
    161 	    # Check the size of ZA in the signal frame.
    162 	    set expected_za_size [expr $svl * $svl]
    163 	    gdb_test "print sizeof \$za" " = $expected_za_size" "size of za in signal frame"
    164 
    165 	    # Check the value of SVCR in the signal frame.
    166 	    gdb_test "print \$svcr" [get_svcr_value $state] "svcr from signal frame"
    167 
    168 	    # Check the value of ZA in the signal frame.
    169 	    gdb_test "print \$za" $za_pattern "za contents from signal frame"
    170 
    171 	    # Check the value of TPIDR2 in the signal frame.
    172 	    gdb_test "print/x \$tpidr2" " = 0x102030405060708" "tpidr2 contents from signal frame"
    173 
    174 	    # Check the value of SME2 ZT0 in the signal frame.
    175 	    if [is_sme2_available] {
    176 		# The target supports SME2.
    177 		set zt_size 64
    178 		gdb_test "print sizeof \$zt0" " = $zt_size"
    179 		set zt_pattern [string_to_regexp [1d_array_value_pattern $sme2_byte $zt_size]]
    180 		gdb_test "print \$zt0" " = $zt_pattern" "zt contents from signal frame"
    181 	    }
    182 	}
    183     }
    184 }
    185 
    186 test_sme_registers_sigframe $id_start $id_end
    187