--- /dev/null
+/* This test program is part of GDB, the GNU debugger.
+
+   Copyright 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/>.  */
+
+/* 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;
+}
 
--- /dev/null
+# Copyright (C) 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 a binary that uses MTE and exercise various MTE-related scenarios.
+
+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]} {
+    unsupported "memory tagging unsupported"
+    return -1
+}
+
+gdb_breakpoint "access_memory"
+
+if [gdb_continue "access_memory"] {
+    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} {
+    unresolved "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} {
+    unresolved "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 "memory-tag print-allocation-tag ${untagged_ptr_addr}" $msg \
+            "memory-tag print-allocation-tag with an untagged address"
+
+    gdb_test "memory-tag set-allocation-tag ${untagged_ptr_addr} 1 00" $msg \
+            "memory-tag set-allocation-tag 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 "memory-tag print-logical-tag ${addr_tagged}" \
+                " = 0x${tag_hexnz}" \
+                "print-logical-tag with tag ${i}"
+
+       set tag_hexnn [get_tag_nn $i]
+       gdb_test "memory-tag with-logical-tag ${addr_tagged} ${tag_hexnn}" \
+                " = \\(void \\*\\) ${addr_tagged}" \
+                "with-logical-tag with tag ${i}"
+    }
+
+    set atag_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 "memory-tag set-allocation-tag ${tagged_ptr_addr} 1 ${tag_hexnn}" \
+                $atag_msg \
+                "set-allocation-tag with tag ${i}"
+
+       set tag_hexnz [get_hex_tag [expr $i % 16]]
+       gdb_test "memory-tag print-allocation-tag ${tagged_ptr_addr}" " = 0x${tag_hexnz}" \
+                "print-allocation-tag 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 "memory-tag set-allocation-tag ${tagged_ptr_addr} 1 ${tag_hexnn}" \
+                    $atag_msg \
+                    "set-allocation-tag 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 "memory-tag 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 "memory-tag 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 "memory-tag print-allocation-tag ${untagged_ptr_symbol}" $msg \
+            "memory-tag print-allocation-tag with an untagged address"
+
+    gdb_test "memory-tag set-allocation-tag ${untagged_ptr_symbol} 1 00" $msg \
+            "memory-tag set-allocation-tag 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 "memory-tag print-logical-tag ${tagged_ptr_symbol}" \
+                " = 0x${tag_hexnz}" \
+                "print-logical-tag with tag ${i}"
+
+       set tag_hexnn [get_tag_nn $i]
+       gdb_test "memory-tag with-logical-tag ${tagged_ptr_symbol} ${tag_hexnn}" \
+                " = \\(void \\*\\) ${addr_tagged}" \
+                "with-logical-tag 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 atag_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 "memory-tag set-allocation-tag ${tagged_ptr_symbol} 1 ${tag_hexnn}" \
+                $atag_msg \
+                "set-allocation-tag with tag ${i}"
+
+       set tag_hexnz [get_hex_tag [expr $i % 16]]
+       gdb_test "memory-tag print-allocation-tag ${tagged_ptr_symbol}" \
+                " = 0x${tag_hexnz}" \
+                "print-allocation-tag 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 "memory-tag set-allocation-tag ${tagged_ptr_symbol} 1 ${tag_hexnn}" \
+                    $atag_msg \
+                    "set-allocation-tag 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 "memory-tag 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 "memory-tag 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} {
+       unresolved "unexpected tag value"
+       return -1
+    }
+
+    set atag [expr [expr $ltag + 1] % 16]
+    set atag_hexnn [get_tag_nn $atag]
+
+    gdb_test "memory-tag set-allocation-tag ${tagged_ptr_symbol} 1 ${atag_hexnn}" \
+            $atag_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" \
+       "Logical 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 "memory-tag set-allocation-tag tagged_ptr 1 05" \
+            $atag_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"
+}
 
--- /dev/null
+# Copyright 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/>.
+
+# 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 {"with-logical-tag" "print-logical-tag" \
+                   "set-allocation-tag" "print-allocation-tag" "check"} {
+       gdb_test "memory-tag $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]} {
+    unsupported "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 memory-tag commands again.
+    gdb_test "memory-tag print-logical-tag" $msg
+    gdb_test "memory-tag print-allocation-tag" $msg
+    gdb_test "memory-tag with-logical-tag" \
+            "Argument required \\(<address> <tag>\\)\."
+    gdb_test "memory-tag set-allocation-tag" \
+            "Argument required \\(<starting address> <length> <tag bytes>\\)\."
+    gdb_test "memory-tag check" $msg
+}