[PATCH v3 24/24] Add memory tagging testcases
David Spickett
david.spickett@linaro.org
Mon Nov 16 15:47:05 GMT 2020
(I've just skimmed through so not a review in any real way)
In gdb/testsuite/gdb.arch/aarch64-mte.exp:
+# Test a binary with address signing works regardless of whether the target
+# supports pauth instructions. On non pauth systems, all pauth instructions
+# are treated as nops.
I assume this was left over from the original test file you adapted,
it doesn't apply to MTE.
On Mon, 9 Nov 2020 at 17:05, Luis Machado <luis.machado@linaro.org> wrote:
>
> Add an AArch64-specific test and a more generic memory tagging test that
> other architectures can run.
>
> Even though architectures not supporting memory tagging can run the memory
> tagging tests, the runtime check will make the tests bail out early, as it
> would make no sense to proceed without proper support.
>
> It is also tricky to do any further runtime tests for memory tagging, given
> we'd need to deal with tags, and those are arch-specific. Therefore the
> test in gdb.base is more of a smoke test.
>
> If an architecture wants to implement memory tagging, then it makes sense to
> have tests within gdb.arch instead.
>
> gdb/testsuite/ChangeLog:
>
> YYYY-MM-DD Luis Machado <luis.machado@linaro.org>
>
> * gdb.arch/aarch64-mte.c: New file.
> * gdb.arch/aarch64-mte.exp: New test.
> * gdb.base/memtag.c: New file.
> * gdb.base/memtag.exp: New test.
> * lib/gdb.exp (supports_memtag): New function.
> ---
> gdb/testsuite/gdb.arch/aarch64-mte.c | 107 +++++++
> gdb/testsuite/gdb.arch/aarch64-mte.exp | 371 +++++++++++++++++++++++++
> gdb/testsuite/gdb.base/memtag.c | 22 ++
> gdb/testsuite/gdb.base/memtag.exp | 64 +++++
> gdb/testsuite/lib/gdb.exp | 16 ++
> 5 files changed, 580 insertions(+)
> create mode 100644 gdb/testsuite/gdb.arch/aarch64-mte.c
> create mode 100644 gdb/testsuite/gdb.arch/aarch64-mte.exp
> create mode 100644 gdb/testsuite/gdb.base/memtag.c
> create mode 100644 gdb/testsuite/gdb.base/memtag.exp
>
> diff --git a/gdb/testsuite/gdb.arch/aarch64-mte.c b/gdb/testsuite/gdb.arch/aarch64-mte.c
> new file mode 100644
> index 0000000000..f6fb4cca67
> --- /dev/null
> +++ b/gdb/testsuite/gdb.arch/aarch64-mte.c
> @@ -0,0 +1,107 @@
> +/* This test program is part of GDB, the GNU debugger.
> +
> + Copyright 2020 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 AArch64's Memory Tagging Extension with tagged pointers. */
> +
> +/* This test was based on the documentation for the AArch64 Memory Tagging
> + Extension from the Linux Kernel, found in the sources in
> + Documentation/arm64/memory-tagging-extension.rst. */
> +
> +#include <errno.h>
> +#include <stdio.h>
> +#include <stdlib.h>
> +#include <unistd.h>
> +#include <sys/auxv.h>
> +#include <sys/mman.h>
> +#include <sys/prctl.h>
> +
> +/* From arch/arm64/include/uapi/asm/hwcap.h */
> +#define HWCAP2_MTE (1 << 18)
> +
> +/* From arch/arm64/include/uapi/asm/mman.h */
> +#define PROT_MTE 0x20
> +
> +/* From include/uapi/linux/prctl.h */
> +#define PR_SET_TAGGED_ADDR_CTRL 55
> +#define PR_GET_TAGGED_ADDR_CTRL 56
> +#define PR_TAGGED_ADDR_ENABLE (1UL << 0)
> +#define PR_MTE_TCF_SHIFT 1
> +#define PR_MTE_TCF_NONE (0UL << PR_MTE_TCF_SHIFT)
> +#define PR_MTE_TCF_SYNC (1UL << PR_MTE_TCF_SHIFT)
> +#define PR_MTE_TCF_ASYNC (2UL << PR_MTE_TCF_SHIFT)
> +#define PR_MTE_TCF_MASK (3UL << PR_MTE_TCF_SHIFT)
> +#define PR_MTE_TAG_SHIFT 3
> +#define PR_MTE_TAG_MASK (0xffffUL << PR_MTE_TAG_SHIFT)
> +
> +void
> +access_memory (unsigned char *tagged_ptr, unsigned char *untagged_ptr)
> +{
> + tagged_ptr[0] = 'a';
> +}
> +
> +int
> +main (int argc, char **argv)
> +{
> + unsigned char *tagged_ptr;
> + unsigned char *untagged_ptr;
> + unsigned long page_sz = sysconf (_SC_PAGESIZE);
> + unsigned long hwcap2 = getauxval(AT_HWCAP2);
> +
> + /* Bail out if MTE is not supported. */
> + if (!(hwcap2 & HWCAP2_MTE))
> + return 1;
> +
> + /* Enable the tagged address ABI, synchronous MTE tag check faults and
> + allow all non-zero tags in the randomly generated set. */
> + if (prctl (PR_SET_TAGGED_ADDR_CTRL,
> + PR_TAGGED_ADDR_ENABLE | PR_MTE_TCF_SYNC
> + | (0xfffe << PR_MTE_TAG_SHIFT),
> + 0, 0, 0))
> + {
> + perror ("prctl () failed");
> + return 1;
> + }
> +
> + /* Create a mapping that will have PROT_MTE set. */
> + tagged_ptr = mmap (0, page_sz, PROT_READ | PROT_WRITE,
> + MAP_PRIVATE | MAP_ANONYMOUS, -1, 0);
> + if (tagged_ptr == MAP_FAILED)
> + {
> + perror ("mmap () failed");
> + return 1;
> + }
> +
> + /* Create another mapping that won't have PROT_MTE set. */
> + untagged_ptr = mmap (0, page_sz, PROT_READ | PROT_WRITE,
> + MAP_PRIVATE | MAP_ANONYMOUS, -1, 0);
> + if (untagged_ptr == MAP_FAILED)
> + {
> + perror ("mmap () failed");
> + return 1;
> + }
> +
> + /* Enable MTE on the above anonymous mmap. */
> + if (mprotect (tagged_ptr, page_sz, PROT_READ | PROT_WRITE | PROT_MTE))
> + {
> + perror ("mprotect () failed");
> + return 1;
> + }
> +
> + access_memory (tagged_ptr, untagged_ptr);
> +
> + return 0;
> +}
> diff --git a/gdb/testsuite/gdb.arch/aarch64-mte.exp b/gdb/testsuite/gdb.arch/aarch64-mte.exp
> new file mode 100644
> index 0000000000..39dba493fb
> --- /dev/null
> +++ b/gdb/testsuite/gdb.arch/aarch64-mte.exp
> @@ -0,0 +1,371 @@
> +# Copyright (C) 2020 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 a binary with address signing works regardless of whether the target
> +# supports pauth instructions. On non pauth systems, all pauth instructions
> +# are treated as nops.
> +
> +global hex
> +global decimal
> +
> +# Return TAG in hex format with no leading zeroes.
> +proc get_hex_tag { tag } {
> + return [format "%x" $tag]
> +}
> +
> +# Return TAG in the NN format where N is 4 bits of the byte.
> +proc get_tag_nn { tag } {
> + return [format "%02x" $tag]
> +}
> +
> +# Return the address of PTR with a tag of TAG.
> +proc get_tagged_ptr { tag ptr } {
> + set addr [get_hexadecimal_valueof $ptr -1]
> + return [get_valueof "/x" \
> + "${addr} & (0xf0ffffffffffffff) | ((unsigned long) ${tag} << 56)" \
> + "0" "fetch pointer ${ptr} with tag ${tag}"]
> +}
> +
> +# Return the logical TAG from PTR.
> +proc get_ltag_from_ptr { ptr } {
> + set addr [get_hexadecimal_valueof $ptr -1]
> + return [get_valueof "/x" "${addr} >> 56 & 0xf" -1 \
> + "fetch tag from pointer ${ptr}"]
> +}
> +
> +if {![is_aarch64_target]} {
> + verbose "Skipping ${gdb_test_file_name}."
> + return
> +}
> +
> +standard_testfile
> +if { [prepare_for_testing "failed to prepare" ${testfile} ${srcfile}] } {
> + return -1
> +}
> +
> +if ![runto_main] {
> + untested "could not run to main"
> + return -1
> +}
> +
> +# Targets that don't support memory tagging should not execute the
> +# runtime memory tagging tests.
> +if {![supports_memtag]} {
> + untested "memory tagging unsupported"
> + return -1
> +}
> +
> +gdb_breakpoint "access_memory"
> +
> +if [gdb_continue "access_memory"] {
> + fail "could not run to tagged memory test function"
> + return -1
> +}
> +
> +# Fetch a known pointer to an area mapped with PROT_MTE.
> +set tagged_ptr_symbol "tagged_ptr"
> +set tagged_ptr_addr [get_hexadecimal_valueof $tagged_ptr_symbol -1]
> +
> +if {$tagged_ptr_addr == -1} {
> + untested "unexpected pointer or tag value"
> + return -1
> +}
> +
> +# Fetch a known pointer to an area not mapped with PROT_MTE.
> +set untagged_ptr_symbol "untagged_ptr"
> +set untagged_ptr_addr [get_hexadecimal_valueof $untagged_ptr_symbol -1]
> +
> +if {$untagged_ptr_addr == -1} {
> + untested "unexpected pointer or tag value"
> + return -1
> +}
> +
> +with_test_prefix "literals" {
> + # Test inspecting an allocation tag from a pointer to a memory area that
> + # is not mapped with PROT_MTE.
> + set msg "Address ${untagged_ptr_addr} not in a region mapped with a memory tagging flag\."
> + gdb_test "mtag showatag ${untagged_ptr_addr}" $msg \
> + "mtag showatag with an untagged address"
> +
> + gdb_test "mtag setatag ${untagged_ptr_addr} 0 00" $msg \
> + "mtag setatag with an untagged address"
> +
> + set addr_tagged 0
> + set addr_tagged_valid 0
> +
> + # Test setting and showing the logical tags for a literal address.
> + for {set i 0} {$i < 32} {incr i} {
> + with_test_prefix "tag ${i}" {
> + set addr_tagged [get_tagged_ptr $i ${tagged_ptr_addr}]
> + }
> +
> + set tag_hexnz [get_hex_tag [expr $i % 16]]
> + gdb_test "mtag showltag ${addr_tagged}" \
> + " = 0x${tag_hexnz}" \
> + "showltag with tag ${i}"
> +
> + set tag_hexnn [get_tag_nn $i]
> + gdb_test "mtag withltag ${addr_tagged} ${tag_hexnn}" \
> + " = \\(void \\*\\) ${addr_tagged}" \
> + "withltag with tag ${i}"
> + }
> +
> + set setatag_msg "Allocation tag\\(s\\) updated successfully\."
> + # Test setting and showing the allocation tags.
> + for {set i 0} {$i < 32} {incr i} {
> +
> + set tag_hexnn [get_tag_nn $i]
> + gdb_test "mtag setatag ${tagged_ptr_addr} 0 ${tag_hexnn}" \
> + $setatag_msg \
> + "setatag with tag ${i}"
> +
> + set tag_hexnz [get_hex_tag [expr $i % 16]]
> + gdb_test "mtag showatag ${tagged_ptr_addr}" " = 0x${tag_hexnz}" \
> + "showatag with tag ${i}"
> + }
> +
> + # Test tag mismatches.
> + with_test_prefix "tag mismatches" {
> + for {set i 0} {$i < 32} {incr i} {
> +
> + # Set the allocation tag to a known value.
> + set tag_hexnn [get_tag_nn $i]
> + gdb_test "mtag setatag ${tagged_ptr_addr} 0 ${tag_hexnn}" \
> + $setatag_msg \
> + "setatag with tag ${i}"
> +
> + set atag_hexnz [get_hex_tag [expr $i % 16]]
> +
> + # Validate that the logical tag matches the allocation tag.
> + with_test_prefix "tag ${i}" {
> + set addr_tagged [get_tagged_ptr $i ${tagged_ptr_addr}]
> + }
> +
> + gdb_test "mtag check ${addr_tagged}" \
> + "Memory tags for address $hex match \\(0x${atag_hexnz}\\)\." \
> + "check match with tag ${i}"
> +
> + # Get a pointer with the logical tag that does not match the
> + # allocation tag.
> + set ltag [expr $i + 1]
> + with_test_prefix "fetch mismatch tag ${i}" {
> + set addr_tagged [get_tagged_ptr $ltag ${tagged_ptr_addr}]
> + }
> +
> + # Validate that the logical tag does not match the allocation
> + # tag.
> + set ltag_hexnz [get_hex_tag [expr [expr $i + 1]% 16]]
> + gdb_test "mtag check ${addr_tagged}" \
> + "Logical tag \\(0x${ltag_hexnz}\\) does not match the allocation tag \\(0x${atag_hexnz}\\) for address $hex\." \
> + "check mismatch with tag ${i}"
> + }
> + }
> +}
> +
> +with_test_prefix "symbolic" {
> + # Test inspecting an allocation tag from a pointer to a memory area that
> + # is not mapped with PROT_MTE.
> + set msg "Address ${untagged_ptr_addr} not in a region mapped with a memory tagging flag\."
> + gdb_test "mtag showatag ${untagged_ptr_symbol}" $msg \
> + "mtag showatag with an untagged address"
> +
> + gdb_test "mtag setatag ${untagged_ptr_symbol} 0 00" $msg \
> + "mtag setatag with an untagged address"
> +
> + # Test setting and showing the logical tags for a literal address.
> + for {set i 0} {$i < 32} {incr i} {
> + set addr_tagged 0
> +
> + with_test_prefix "tag ${i}" {
> + set addr_tagged [get_tagged_ptr $i ${tagged_ptr_addr}]
> + gdb_test_no_output "set variable ${tagged_ptr_symbol} = ${addr_tagged}" \
> + "update value of symbol ${tagged_ptr_symbol}"
> + }
> +
> + set tag_hexnz [get_hex_tag [expr $i % 16]]
> + gdb_test "mtag showltag ${tagged_ptr_symbol}" \
> + " = 0x${tag_hexnz}" \
> + "showltag with tag ${i}"
> +
> + set tag_hexnn [get_tag_nn $i]
> + gdb_test "mtag withltag ${tagged_ptr_symbol} ${tag_hexnn}" \
> + " = \\(void \\*\\) ${addr_tagged}" \
> + "withltag with tag ${i}"
> + }
> +
> + # Reset the tagged ptr to its original value
> + gdb_test_no_output "set variable ${tagged_ptr_symbol} = ${tagged_ptr_addr}" \
> + "reset ${tagged_ptr_symbol} to ${tagged_ptr_addr}"
> +
> + set setatag_msg "Allocation tag\\(s\\) updated successfully\."
> + # Test setting and showing the allocation tags.
> + for {set i 0} {$i < 32} {incr i} {
> +
> + set tag_hexnn [get_tag_nn $i]
> + gdb_test "mtag setatag ${tagged_ptr_symbol} 0 ${tag_hexnn}" \
> + $setatag_msg \
> + "setatag with tag ${i}"
> +
> + set tag_hexnz [get_hex_tag [expr $i % 16]]
> + gdb_test "mtag showatag ${tagged_ptr_symbol}" " = 0x${tag_hexnz}" \
> + "showatag with tag ${i}"
> + }
> +
> + # Test tag mismatches.
> + with_test_prefix "tag mismatches" {
> + for {set i 0} {$i < 32} {incr i} {
> +
> + # Set the allocation tag to a known value (0).
> + set tag_hexnn [get_tag_nn $i]
> + gdb_test "mtag setatag ${tagged_ptr_symbol} 0 ${tag_hexnn}" \
> + $setatag_msg \
> + "setatag with tag ${i}"
> +
> + set atag_hexnz [get_hex_tag [expr $i % 16]]
> +
> + # Validate that the logical tag matches the allocation tag.
> + with_test_prefix "tag ${i}" {
> + set addr_tagged [get_tagged_ptr $i ${tagged_ptr_addr}]
> + }
> +
> + with_test_prefix "tag ${i}" {
> + gdb_test_no_output "set variable ${tagged_ptr_symbol} = ${addr_tagged}" \
> + "set ${tagged_ptr_symbol} to a matching logical tag"
> + }
> +
> + gdb_test "mtag check ${tagged_ptr_symbol}" \
> + "Memory tags for address $hex match \\(0x${atag_hexnz}\\)\." \
> + "check match with tag ${i}"
> +
> + # Get a pointer with the logical tag that does not match the
> + # allocation tag.
> + set ltag [expr $i + 1]
> + with_test_prefix "fetch mismatch tag ${i}" {
> + set addr_tagged [get_tagged_ptr $ltag ${tagged_ptr_addr}]
> + }
> +
> + with_test_prefix "tag ${i}" {
> + gdb_test_no_output "set variable ${tagged_ptr_symbol} = ${addr_tagged}" \
> + "set ${tagged_ptr_symbol} to a mismatching logical tag"
> + }
> +
> + # Validate that the logical tag does not match the allocation
> + # tag.
> + set ltag_hexnz [get_hex_tag [expr [expr $i + 1]% 16]]
> + gdb_test "mtag check ${tagged_ptr_symbol}" \
> + "Logical tag \\(0x${ltag_hexnz}\\) does not match the allocation tag \\(0x${atag_hexnz}\\) for address $hex\." \
> + "check mismatch with tag ${i}"
> + }
> + # Reset the tagged ptr to its original value
> + gdb_test_no_output "set variable ${tagged_ptr_symbol} = ${tagged_ptr_addr}" \
> + "reset ${tagged_ptr_symbol} to ${tagged_ptr_addr}"
> + }
> +}
> +
> +# Test the memory tagging extensions for the "print" command.
> +with_test_prefix "print command" {
> + set untagged_ptr [get_tagged_ptr 0 ${tagged_ptr_addr}]
> +
> + with_test_prefix "fetch ltag" {
> + set ltag [get_ltag_from_ptr ${tagged_ptr_addr}]
> + }
> +
> + if {$ltag == -1} {
> + untested "unexpected tag value"
> + return -1
> + }
> +
> + set atag [expr [expr $ltag + 1] % 16]
> + set atag_hexnn [get_tag_nn $atag]
> +
> + gdb_test "mtag setatag ${tagged_ptr_symbol} 0 ${atag_hexnn}" \
> + $setatag_msg \
> + "make atag and ltag different"
> +
> + set atag_hexnz [get_hex_tag $atag]
> + gdb_test "p/x ${tagged_ptr_symbol}" \
> + [multi_line \
> + "Logical tag \\(${ltag}\\) does not match the allocation tag \\(0x${atag_hexnz}\\)\." \
> + "\\\$\[0-9\]+ = ${untagged_ptr}"] \
> + "show tag mismatch"
> +}
> +
> +# Test the memory tagging extensions for the "x" command.
> +with_test_prefix "x command" {
> +
> + # Check if the allocation tags match what we expect.
> + gdb_test "x/gxm ${tagged_ptr_symbol}" \
> + [multi_line \
> + "<Allocation Tag $hex for range \\\[$hex,$hex\\)>" \
> + "$hex:\[ \t\]+$hex"] \
> + "outputs tag information"
> +
> + # Also make sure no tag information is output for memory areas without
> + # PROT_MTE mappings.
> + gdb_test "x/gxm ${untagged_ptr_symbol}" \
> + "$hex:\[ \t\]+$hex" \
> + "does not output tag information"
> +}
> +
> +# Validate the presence of the MTE registers.
> +foreach reg {"tag_ctl" } {
> + gdb_test "info registers $reg" \
> + "$reg\[ \t\]+$hex\[ \t\]+$decimal" \
> + "register $reg available"
> +}
> +
> +# Run until a crash and confirm GDB displays memory tag violation
> +# information.
> +gdb_test "continue" \
> + [multi_line \
> + "Program received signal SIGSEGV, Segmentation fault" \
> + "Memory tag violation while accessing address $hex" \
> + "Allocation tag $hex\." \
> + "$hex in access_memory \\(.*\\) at .*" \
> + ".*tagged_ptr\\\[0\\\] = 'a';"] \
> + "display tag violation information"
> +
> +# Restart to execute the async tag fault test.
> +with_test_prefix "async" {
> + if ![runto_main] {
> + untested "could not run to main"
> + return -1
> + }
> +
> + gdb_breakpoint "access_memory"
> +
> + if [gdb_continue "access_memory"] {
> + fail "could not run to tagged memory test function"
> + return -1
> + }
> +
> + # Force a tag fault.
> + gdb_test "mtag setatag tagged_ptr 0 05" \
> + $setatag_msg \
> + "make atag and ltag different"
> +
> + # Force the tag fault to be async.
> + gdb_test_no_output "set \$tag_ctl=0x7fff5" "set tag_ctl to async"
> +
> + # Run until a crash and confirm GDB displays memory tag violation
> + # information for async mode
> + gdb_test "continue" \
> + [multi_line \
> + "Program received signal SIGSEGV, Segmentation fault" \
> + "Memory tag violation" \
> + "Fault address unavailable\." \
> + "$hex in .* \\(.*\\) .*"] \
> + "display tag violation information"
> +}
> diff --git a/gdb/testsuite/gdb.base/memtag.c b/gdb/testsuite/gdb.base/memtag.c
> new file mode 100644
> index 0000000000..63a42ae278
> --- /dev/null
> +++ b/gdb/testsuite/gdb.base/memtag.c
> @@ -0,0 +1,22 @@
> +/* This test program is part of GDB, the GNU debugger.
> +
> + Copyright 2020 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/>. */
> +
> +int
> +main (int argc, char **argv)
> +{
> + return 0;
> +}
> diff --git a/gdb/testsuite/gdb.base/memtag.exp b/gdb/testsuite/gdb.base/memtag.exp
> new file mode 100644
> index 0000000000..484a78e828
> --- /dev/null
> +++ b/gdb/testsuite/gdb.base/memtag.exp
> @@ -0,0 +1,64 @@
> +# Copyright 2020 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/>.
> +
> +# Smoke testing for the various memory tagging commands in GDB.
> +
> +set u_msg "Memory tagging not supported or disabled by the current architecture\."
> +
> +standard_testfile
> +if {[prepare_for_testing "failed to prepare" ${testfile} ${srcfile}]} {
> + return -1
> +}
> +
> +if {[target_info gdb_protocol] == "extended-remote"} {
> + # Make sure we're disconnected, in case we're testing with an
> + # extended-remote board, therefore already connected.
> + gdb_test "disconnect" ".*"
> +}
> +
> +# Test commands without running the program.
> +with_test_prefix "before program execution" {
> + # These commands should all fails without a running program.
> + foreach subcmd {"withltag" "showltag" "setatag" "showatag" "check"} {
> + gdb_test "mtag $subcmd" $u_msg
> + }
> +}
> +
> +clean_restart $testfile
> +
> +if ![runto_main] {
> + untested "could not run to main"
> + return -1
> +}
> +
> +# Targets that don't support memory tagging should not execute the
> +# runtime memory tagging tests.
> +if {![supports_memtag]} {
> + untested "memory tagging unsupported"
> + return -1
> +}
> +
> +# With the program running, try to use the memory tagging commands.
> +with_test_prefix "during program execution" {
> + set msg "Argument required \\(address or pointer\\)\."
> +
> + # Test the various mtag commands again.
> + gdb_test "mtag showltag" $msg
> + gdb_test "mtag showatag" $msg
> + gdb_test "mtag withltag" "Argument required \\(<address> <tag>\\)\."
> + gdb_test "mtag setatag" \
> + "Argument required \\(<starting address> <length> <tag bytes>\\)\."
> + gdb_test "mtag check" $msg
> +}
> diff --git a/gdb/testsuite/lib/gdb.exp b/gdb/testsuite/lib/gdb.exp
> index 55154db6a5..347704ce0a 100644
> --- a/gdb/testsuite/lib/gdb.exp
> +++ b/gdb/testsuite/lib/gdb.exp
> @@ -2679,6 +2679,22 @@ proc supports_get_siginfo_type {} {
> }
> }
>
> +# Return 1 if memory tagging is supported at runtime, otherwise return 0.
> +
> +proc supports_memtag {} {
> + global gdb_prompt
> +
> + gdb_test_multiple "mtag check" "" {
> + -re "Memory tagging not supported or disabled by the current architecture\..*$gdb_prompt $" {
> + return 0
> + }
> + -re "Argument required \\(address or pointer\\).*$gdb_prompt $" {
> + return 1
> + }
> + }
> + return 0
> +}
> +
> # Return 1 if the target supports hardware single stepping.
>
> proc can_hardware_single_step {} {
> --
> 2.17.1
>
More information about the Gdb-patches
mailing list