[gdb/testsuite] Fix gdb.opt/inline-small-func.exp with clang
[binutils-gdb.git] / gdb / linux-tdep.c
index 17ed9cac8a739b250872e0f32b804782dc39b5e5..8a83ed320cfeec937837a833c182f12634b4cbdf 100644 (file)
@@ -1,6 +1,6 @@
 /* Target-dependent code for GNU/Linux, architecture independent.
 
-   Copyright (C) 2009-2021 Free Software Foundation, Inc.
+   Copyright (C) 2009-2022 Free Software Foundation, Inc.
 
    This file is part of GDB.
 
 #include "inferior.h"
 #include "cli/cli-utils.h"
 #include "arch-utils.h"
-#include "gdb_obstack.h"
+#include "gdbsupport/gdb_obstack.h"
 #include "observable.h"
 #include "objfiles.h"
 #include "infcall.h"
 #include "gdbcmd.h"
-#include "gdb_regex.h"
+#include "gdbsupport/gdb_regex.h"
 #include "gdbsupport/enum-flags.h"
 #include "gdbsupport/gdb_optional.h"
 #include "gcore.h"
 #include "gcore-elf.h"
+#include "solib-svr4.h"
+#include "memtag.h"
 
 #include <ctype.h>
+#include <unordered_map>
 
 /* This enum represents the values that the user can choose when
    informing the Linux kernel about which memory mappings will be
@@ -88,8 +91,33 @@ struct smaps_vmflags
     /* Is this a MAP_SHARED mapping (VM_SHARED, "sh").  */
 
     unsigned int shared_mapping : 1;
+
+    /* Memory map has memory tagging enabled.  */
+
+    unsigned int memory_tagging : 1;
   };
 
+/* Data structure that holds the information contained in the
+   /proc/<pid>/smaps file.  */
+
+struct smaps_data
+{
+  ULONGEST start_address;
+  ULONGEST end_address;
+  std::string filename;
+  struct smaps_vmflags vmflags;
+  bool read;
+  bool write;
+  bool exec;
+  bool priv;
+  bool has_anonymous;
+  bool mapping_anon_p;
+  bool mapping_file_p;
+
+  ULONGEST inode;
+  ULONGEST offset;
+};
+
 /* Whether to take the /proc/PID/coredump_filter into account when
    generating a corefile.  */
 
@@ -354,6 +382,13 @@ linux_get_siginfo_type_with_fields (struct gdbarch *gdbarch,
   append_composite_type_field (type, "si_fd", int_type);
   append_composite_type_field (sifields_type, "_sigpoll", type);
 
+  /* _sigsys */
+  type = arch_composite_type (gdbarch, NULL, TYPE_CODE_STRUCT);
+  append_composite_type_field (type, "_call_addr", void_ptr_type);
+  append_composite_type_field (type, "_syscall", int_type);
+  append_composite_type_field (type, "_arch", uint_type);
+  append_composite_type_field (sifields_type, "_sigsys", type);
+
   /* struct siginfo */
   siginfo_type = arch_composite_type (gdbarch, NULL, TYPE_CODE_STRUCT);
   siginfo_type->set_name (xstrdup ("siginfo"));
@@ -385,9 +420,10 @@ int
 linux_is_uclinux (void)
 {
   CORE_ADDR dummy;
+  target_ops *target = current_inferior ()->top_target ();
 
-  return (target_auxv_search (current_top_target (), AT_NULL, &dummy) > 0
-         && target_auxv_search (current_top_target (), AT_PAGESZ, &dummy) == 0);
+  return (target_auxv_search (target, AT_NULL, &dummy) > 0
+         && target_auxv_search (target, AT_PAGESZ, &dummy) == 0);
 }
 
 static int
@@ -407,42 +443,55 @@ linux_core_pid_to_str (struct gdbarch *gdbarch, ptid_t ptid)
   return normal_pid_to_str (ptid);
 }
 
+/* Data from one mapping from /proc/PID/maps.  */
+
+struct mapping
+{
+  ULONGEST addr;
+  ULONGEST endaddr;
+  gdb::string_view permissions;
+  ULONGEST offset;
+  gdb::string_view device;
+  ULONGEST inode;
+
+  /* This field is guaranteed to be NULL-terminated, hence it is not a
+     gdb::string_view.  */
+  const char *filename;
+};
+
 /* Service function for corefiles and info proc.  */
 
-static void
-read_mapping (const char *line,
-             ULONGEST *addr, ULONGEST *endaddr,
-             const char **permissions, size_t *permissions_len,
-             ULONGEST *offset,
-             const char **device, size_t *device_len,
-             ULONGEST *inode,
-             const char **filename)
+static mapping
+read_mapping (const char *line)
 {
+  struct mapping mapping;
   const char *p = line;
 
-  *addr = strtoulst (p, &p, 16);
+  mapping.addr = strtoulst (p, &p, 16);
   if (*p == '-')
     p++;
-  *endaddr = strtoulst (p, &p, 16);
+  mapping.endaddr = strtoulst (p, &p, 16);
 
   p = skip_spaces (p);
-  *permissions = p;
+  const char *permissions_start = p;
   while (*p && !isspace (*p))
     p++;
-  *permissions_len = p - *permissions;
+  mapping.permissions = {permissions_start, (size_t) (p - permissions_start)};
 
-  *offset = strtoulst (p, &p, 16);
+  mapping.offset = strtoulst (p, &p, 16);
 
   p = skip_spaces (p);
-  *device = p;
+  const char *device_start = p;
   while (*p && !isspace (*p))
     p++;
-  *device_len = p - *device;
+  mapping.device = {device_start, (size_t) (p - device_start)};
 
-  *inode = strtoulst (p, &p, 10);
+  mapping.inode = strtoulst (p, &p, 10);
 
   p = skip_spaces (p);
-  *filename = p;
+  mapping.filename = p;
+
+  return mapping;
 }
 
 /* Helper function to decode the "VmFlags" field in /proc/PID/smaps.
@@ -476,6 +525,8 @@ decode_vmflags (char *p, struct smaps_vmflags *v)
        v->exclude_coredump = 1;
       else if (strcmp (s, "sh") == 0)
        v->shared_mapping = 1;
+      else if (strcmp (s, "mt") == 0)
+       v->memory_tagging = 1;
     }
 }
 
@@ -788,7 +839,7 @@ linux_info_proc (struct gdbarch *gdbarch, const char *args,
   if (args && args[0])
     error (_("Too many parameters: %s"), args);
 
-  printf_filtered (_("process %ld\n"), pid);
+  gdb_printf (_("process %ld\n"), pid);
   if (cmdline_f)
     {
       xsnprintf (filename, sizeof filename, "/proc/%ld/cmdline", pid);
@@ -806,7 +857,7 @@ linux_info_proc (struct gdbarch *gdbarch, const char *args,
                buffer[pos] = ' ';
            }
          buffer[len - 1] = '\0';
-         printf_filtered ("cmdline = '%s'\n", buffer);
+         gdb_printf ("cmdline = '%s'\n", buffer);
        }
       else
        warning (_("unable to open /proc file '%s'"), filename);
@@ -817,7 +868,7 @@ linux_info_proc (struct gdbarch *gdbarch, const char *args,
       gdb::optional<std::string> contents
        = target_fileio_readlink (NULL, filename, &target_errno);
       if (contents.has_value ())
-       printf_filtered ("cwd = '%s'\n", contents->c_str ());
+       gdb_printf ("cwd = '%s'\n", contents->c_str ());
       else
        warning (_("unable to read link '%s'"), filename);
     }
@@ -827,7 +878,7 @@ linux_info_proc (struct gdbarch *gdbarch, const char *args,
       gdb::optional<std::string> contents
        = target_fileio_readlink (NULL, filename, &target_errno);
       if (contents.has_value ())
-       printf_filtered ("exe = '%s'\n", contents->c_str ());
+       gdb_printf ("exe = '%s'\n", contents->c_str ());
       else
        warning (_("unable to read link '%s'"), filename);
     }
@@ -840,20 +891,18 @@ linux_info_proc (struct gdbarch *gdbarch, const char *args,
        {
          char *line;
 
-         printf_filtered (_("Mapped address spaces:\n\n"));
+         gdb_printf (_("Mapped address spaces:\n\n"));
          if (gdbarch_addr_bit (gdbarch) == 32)
            {
-             printf_filtered ("\t%10s %10s %10s %10s %s\n",
-                          "Start Addr",
-                          "  End Addr",
-                          "      Size", "    Offset", "objfile");
+             gdb_printf ("\t%10s %10s %10s %10s  %s %s\n",
+                         "Start Addr", "  End Addr", "      Size",
+                         "    Offset", "Perms  ", "objfile");
            }
          else
            {
-             printf_filtered ("  %18s %18s %10s %10s %s\n",
-                          "Start Addr",
-                          "  End Addr",
-                          "      Size", "    Offset", "objfile");
+             gdb_printf ("  %18s %18s %10s %10s  %s %s\n",
+                         "Start Addr", "  End Addr", "      Size",
+                         "    Offset", "Perms ", "objfile");
            }
 
          char *saveptr;
@@ -861,32 +910,29 @@ linux_info_proc (struct gdbarch *gdbarch, const char *args,
               line;
               line = strtok_r (NULL, "\n", &saveptr))
            {
-             ULONGEST addr, endaddr, offset, inode;
-             const char *permissions, *device, *mapping_filename;
-             size_t permissions_len, device_len;
-
-             read_mapping (line, &addr, &endaddr,
-                           &permissions, &permissions_len,
-                           &offset, &device, &device_len,
-                           &inode, &mapping_filename);
+             struct mapping m = read_mapping (line);
 
              if (gdbarch_addr_bit (gdbarch) == 32)
                {
-                 printf_filtered ("\t%10s %10s %10s %10s %s\n",
-                                  paddress (gdbarch, addr),
-                                  paddress (gdbarch, endaddr),
-                                  hex_string (endaddr - addr),
-                                  hex_string (offset),
-                                  *mapping_filename ? mapping_filename : "");
+                 gdb_printf ("\t%10s %10s %10s %10s  %-5.*s  %s\n",
+                             paddress (gdbarch, m.addr),
+                             paddress (gdbarch, m.endaddr),
+                             hex_string (m.endaddr - m.addr),
+                             hex_string (m.offset),
+                             (int) m.permissions.size (),
+                             m.permissions.data (),
+                             m.filename);
                }
              else
                {
-                 printf_filtered ("  %18s %18s %10s %10s %s\n",
-                                  paddress (gdbarch, addr),
-                                  paddress (gdbarch, endaddr),
-                                  hex_string (endaddr - addr),
-                                  hex_string (offset),
-                                  *mapping_filename ? mapping_filename : "");
+                 gdb_printf ("  %18s %18s %10s %10s  %-5.*s  %s\n",
+                             paddress (gdbarch, m.addr),
+                             paddress (gdbarch, m.endaddr),
+                             hex_string (m.endaddr - m.addr),
+                             hex_string (m.offset),
+                             (int) m.permissions.size (),
+                             m.permissions.data (),
+                             m.filename);
                }
            }
        }
@@ -899,7 +945,7 @@ linux_info_proc (struct gdbarch *gdbarch, const char *args,
       gdb::unique_xmalloc_ptr<char> status
        = target_fileio_read_stralloc (NULL, filename);
       if (status)
-       puts_filtered (status.get ());
+       gdb_puts (status.get ());
       else
        warning (_("unable to open /proc file '%s'"), filename);
     }
@@ -912,8 +958,8 @@ linux_info_proc (struct gdbarch *gdbarch, const char *args,
        {
          const char *p = statstr.get ();
 
-         printf_filtered (_("Process: %s\n"),
-                          pulongest (strtoulst (p, &p, 10)));
+         gdb_printf (_("Process: %s\n"),
+                     pulongest (strtoulst (p, &p, 10)));
 
          p = skip_spaces (p);
          if (*p == '(')
@@ -923,117 +969,117 @@ linux_info_proc (struct gdbarch *gdbarch, const char *args,
              const char *ep = strrchr (p, ')');
              if (ep != NULL)
                {
-                 printf_filtered ("Exec file: %.*s\n",
-                                  (int) (ep - p - 1), p + 1);
+                 gdb_printf ("Exec file: %.*s\n",
+                             (int) (ep - p - 1), p + 1);
                  p = ep + 1;
                }
            }
 
          p = skip_spaces (p);
          if (*p)
-           printf_filtered (_("State: %c\n"), *p++);
+           gdb_printf (_("State: %c\n"), *p++);
 
          if (*p)
-           printf_filtered (_("Parent process: %s\n"),
-                            pulongest (strtoulst (p, &p, 10)));
+           gdb_printf (_("Parent process: %s\n"),
+                       pulongest (strtoulst (p, &p, 10)));
          if (*p)
-           printf_filtered (_("Process group: %s\n"),
-                            pulongest (strtoulst (p, &p, 10)));
+           gdb_printf (_("Process group: %s\n"),
+                       pulongest (strtoulst (p, &p, 10)));
          if (*p)
-           printf_filtered (_("Session id: %s\n"),
-                            pulongest (strtoulst (p, &p, 10)));
+           gdb_printf (_("Session id: %s\n"),
+                       pulongest (strtoulst (p, &p, 10)));
          if (*p)
-           printf_filtered (_("TTY: %s\n"),
-                            pulongest (strtoulst (p, &p, 10)));
+           gdb_printf (_("TTY: %s\n"),
+                       pulongest (strtoulst (p, &p, 10)));
          if (*p)
-           printf_filtered (_("TTY owner process group: %s\n"),
-                            pulongest (strtoulst (p, &p, 10)));
+           gdb_printf (_("TTY owner process group: %s\n"),
+                       pulongest (strtoulst (p, &p, 10)));
 
          if (*p)
-           printf_filtered (_("Flags: %s\n"),
-                            hex_string (strtoulst (p, &p, 10)));
+           gdb_printf (_("Flags: %s\n"),
+                       hex_string (strtoulst (p, &p, 10)));
          if (*p)
-           printf_filtered (_("Minor faults (no memory page): %s\n"),
-                            pulongest (strtoulst (p, &p, 10)));
+           gdb_printf (_("Minor faults (no memory page): %s\n"),
+                       pulongest (strtoulst (p, &p, 10)));
          if (*p)
-           printf_filtered (_("Minor faults, children: %s\n"),
-                            pulongest (strtoulst (p, &p, 10)));
+           gdb_printf (_("Minor faults, children: %s\n"),
+                       pulongest (strtoulst (p, &p, 10)));
          if (*p)
-           printf_filtered (_("Major faults (memory page faults): %s\n"),
-                            pulongest (strtoulst (p, &p, 10)));
+           gdb_printf (_("Major faults (memory page faults): %s\n"),
+                       pulongest (strtoulst (p, &p, 10)));
          if (*p)
-           printf_filtered (_("Major faults, children: %s\n"),
-                            pulongest (strtoulst (p, &p, 10)));
+           gdb_printf (_("Major faults, children: %s\n"),
+                       pulongest (strtoulst (p, &p, 10)));
          if (*p)
-           printf_filtered (_("utime: %s\n"),
-                            pulongest (strtoulst (p, &p, 10)));
+           gdb_printf (_("utime: %s\n"),
+                       pulongest (strtoulst (p, &p, 10)));
          if (*p)
-           printf_filtered (_("stime: %s\n"),
-                            pulongest (strtoulst (p, &p, 10)));
+           gdb_printf (_("stime: %s\n"),
+                       pulongest (strtoulst (p, &p, 10)));
          if (*p)
-           printf_filtered (_("utime, children: %s\n"),
-                            pulongest (strtoulst (p, &p, 10)));
+           gdb_printf (_("utime, children: %s\n"),
+                       pulongest (strtoulst (p, &p, 10)));
          if (*p)
-           printf_filtered (_("stime, children: %s\n"),
-                            pulongest (strtoulst (p, &p, 10)));
+           gdb_printf (_("stime, children: %s\n"),
+                       pulongest (strtoulst (p, &p, 10)));
          if (*p)
-           printf_filtered (_("jiffies remaining in current "
-                              "time slice: %s\n"),
-                            pulongest (strtoulst (p, &p, 10)));
+           gdb_printf (_("jiffies remaining in current "
+                         "time slice: %s\n"),
+                       pulongest (strtoulst (p, &p, 10)));
          if (*p)
-           printf_filtered (_("'nice' value: %s\n"),
-                            pulongest (strtoulst (p, &p, 10)));
+           gdb_printf (_("'nice' value: %s\n"),
+                       pulongest (strtoulst (p, &p, 10)));
          if (*p)
-           printf_filtered (_("jiffies until next timeout: %s\n"),
-                            pulongest (strtoulst (p, &p, 10)));
+           gdb_printf (_("jiffies until next timeout: %s\n"),
+                       pulongest (strtoulst (p, &p, 10)));
          if (*p)
-           printf_filtered (_("jiffies until next SIGALRM: %s\n"),
-                            pulongest (strtoulst (p, &p, 10)));
+           gdb_printf (_("jiffies until next SIGALRM: %s\n"),
+                       pulongest (strtoulst (p, &p, 10)));
          if (*p)
-           printf_filtered (_("start time (jiffies since "
-                              "system boot): %s\n"),
-                            pulongest (strtoulst (p, &p, 10)));
+           gdb_printf (_("start time (jiffies since "
+                         "system boot): %s\n"),
+                       pulongest (strtoulst (p, &p, 10)));
          if (*p)
-           printf_filtered (_("Virtual memory size: %s\n"),
-                            pulongest (strtoulst (p, &p, 10)));
+           gdb_printf (_("Virtual memory size: %s\n"),
+                       pulongest (strtoulst (p, &p, 10)));
          if (*p)
-           printf_filtered (_("Resident set size: %s\n"),
-                            pulongest (strtoulst (p, &p, 10)));
+           gdb_printf (_("Resident set size: %s\n"),
+                       pulongest (strtoulst (p, &p, 10)));
          if (*p)
-           printf_filtered (_("rlim: %s\n"),
-                            pulongest (strtoulst (p, &p, 10)));
+           gdb_printf (_("rlim: %s\n"),
+                       pulongest (strtoulst (p, &p, 10)));
          if (*p)
-           printf_filtered (_("Start of text: %s\n"),
-                            hex_string (strtoulst (p, &p, 10)));
+           gdb_printf (_("Start of text: %s\n"),
+                       hex_string (strtoulst (p, &p, 10)));
          if (*p)
-           printf_filtered (_("End of text: %s\n"),
-                            hex_string (strtoulst (p, &p, 10)));
+           gdb_printf (_("End of text: %s\n"),
+                       hex_string (strtoulst (p, &p, 10)));
          if (*p)
-           printf_filtered (_("Start of stack: %s\n"),
-                            hex_string (strtoulst (p, &p, 10)));
+           gdb_printf (_("Start of stack: %s\n"),
+                       hex_string (strtoulst (p, &p, 10)));
 #if 0  /* Don't know how architecture-dependent the rest is...
           Anyway the signal bitmap info is available from "status".  */
          if (*p)
-           printf_filtered (_("Kernel stack pointer: %s\n"),
-                            hex_string (strtoulst (p, &p, 10)));
+           gdb_printf (_("Kernel stack pointer: %s\n"),
+                       hex_string (strtoulst (p, &p, 10)));
          if (*p)
-           printf_filtered (_("Kernel instr pointer: %s\n"),
-                            hex_string (strtoulst (p, &p, 10)));
+           gdb_printf (_("Kernel instr pointer: %s\n"),
+                       hex_string (strtoulst (p, &p, 10)));
          if (*p)
-           printf_filtered (_("Pending signals bitmap: %s\n"),
-                            hex_string (strtoulst (p, &p, 10)));
+           gdb_printf (_("Pending signals bitmap: %s\n"),
+                       hex_string (strtoulst (p, &p, 10)));
          if (*p)
-           printf_filtered (_("Blocked signals bitmap: %s\n"),
-                            hex_string (strtoulst (p, &p, 10)));
+           gdb_printf (_("Blocked signals bitmap: %s\n"),
+                       hex_string (strtoulst (p, &p, 10)));
          if (*p)
-           printf_filtered (_("Ignored signals bitmap: %s\n"),
-                            hex_string (strtoulst (p, &p, 10)));
+           gdb_printf (_("Ignored signals bitmap: %s\n"),
+                       hex_string (strtoulst (p, &p, 10)));
          if (*p)
-           printf_filtered (_("Catched signals bitmap: %s\n"),
-                            hex_string (strtoulst (p, &p, 10)));
+           gdb_printf (_("Catched signals bitmap: %s\n"),
+                       hex_string (strtoulst (p, &p, 10)));
          if (*p)
-           printf_filtered (_("wchan (system call): %s\n"),
-                            hex_string (strtoulst (p, &p, 10)));
+           gdb_printf (_("wchan (system call): %s\n"),
+                       hex_string (strtoulst (p, &p, 10)));
 #endif
        }
       else
@@ -1068,16 +1114,11 @@ linux_info_proc (struct gdbarch *gdbarch, const char *args,
    for each mapping.  */
 
 static void
-linux_read_core_file_mappings (struct gdbarch *gdbarch,
-                              struct bfd *cbfd,
-                              gdb::function_view<void (ULONGEST count)>
-                                pre_loop_cb,
-                              gdb::function_view<void (int num,
-                                                       ULONGEST start,
-                                                       ULONGEST end,
-                                                       ULONGEST file_ofs,
-                                                       const char *filename)>
-                                loop_cb)
+linux_read_core_file_mappings
+  (struct gdbarch *gdbarch,
+   struct bfd *cbfd,
+   read_core_file_mappings_pre_loop_ftype pre_loop_cb,
+   read_core_file_mappings_loop_ftype  loop_cb)
 {
   /* Ensure that ULONGEST is big enough for reading 64-bit core files.  */
   gdb_static_assert (sizeof (ULONGEST) >= 8);
@@ -1146,6 +1187,22 @@ linux_read_core_file_mappings (struct gdbarch *gdbarch,
   if (f != descend)
     warning (_("malformed note - filename area is too big"));
 
+  const bfd_build_id *orig_build_id = cbfd->build_id;
+  std::unordered_map<ULONGEST, const bfd_build_id *> vma_map;
+
+  /* Search for solib build-ids in the core file.  Each time one is found,
+     map the start vma of the corresponding elf header to the build-id.  */
+  for (bfd_section *sec = cbfd->sections; sec != nullptr; sec = sec->next)
+    {
+      cbfd->build_id = nullptr;
+
+      if (sec->flags & SEC_LOAD
+         && (get_elf_backend_data (cbfd)->elf_backend_core_find_build_id
+              (cbfd, (bfd_vma) sec->filepos)))
+       vma_map[sec->vma] = cbfd->build_id;
+    }
+
+  cbfd->build_id = orig_build_id;
   pre_loop_cb (count);
 
   for (int i = 0; i < count; i++)
@@ -1159,8 +1216,13 @@ linux_read_core_file_mappings (struct gdbarch *gdbarch,
       descdata += addr_size;
       char * filename = filenames;
       filenames += strlen ((char *) filenames) + 1;
+      const bfd_build_id *build_id = nullptr;
+      auto vma_map_it = vma_map.find (start);
 
-      loop_cb (i, start, end, file_ofs, filename);
+      if (vma_map_it != vma_map.end ())
+       build_id = vma_map_it->second;
+
+      loop_cb (i, start, end, file_ofs, filename, build_id);
     }
 }
 
@@ -1172,39 +1234,39 @@ linux_core_info_proc_mappings (struct gdbarch *gdbarch, const char *args)
   linux_read_core_file_mappings (gdbarch, core_bfd,
     [=] (ULONGEST count)
       {
-       printf_filtered (_("Mapped address spaces:\n\n"));
+       gdb_printf (_("Mapped address spaces:\n\n"));
        if (gdbarch_addr_bit (gdbarch) == 32)
          {
-           printf_filtered ("\t%10s %10s %10s %10s %s\n",
-                            "Start Addr",
-                            "  End Addr",
-                            "      Size", "    Offset", "objfile");
+           gdb_printf ("\t%10s %10s %10s %10s %s\n",
+                       "Start Addr",
+                       "  End Addr",
+                       "      Size", "    Offset", "objfile");
          }
        else
          {
-           printf_filtered ("  %18s %18s %10s %10s %s\n",
-                            "Start Addr",
-                            "  End Addr",
-                            "      Size", "    Offset", "objfile");
+           gdb_printf ("  %18s %18s %10s %10s %s\n",
+                       "Start Addr",
+                       "  End Addr",
+                       "      Size", "    Offset", "objfile");
          }
       },
     [=] (int num, ULONGEST start, ULONGEST end, ULONGEST file_ofs,
-        const char *filename)
+        const char *filename, const bfd_build_id *build_id)
       {
        if (gdbarch_addr_bit (gdbarch) == 32)
-         printf_filtered ("\t%10s %10s %10s %10s %s\n",
-                          paddress (gdbarch, start),
-                          paddress (gdbarch, end),
-                          hex_string (end - start),
-                          hex_string (file_ofs),
-                          filename);
+         gdb_printf ("\t%10s %10s %10s %10s %s\n",
+                     paddress (gdbarch, start),
+                     paddress (gdbarch, end),
+                     hex_string (end - start),
+                     hex_string (file_ofs),
+                     filename);
        else
-         printf_filtered ("  %18s %18s %10s %10s %s\n",
-                          paddress (gdbarch, start),
-                          paddress (gdbarch, end),
-                          hex_string (end - start),
-                          hex_string (file_ofs),
-                          filename);
+         gdb_printf ("  %18s %18s %10s %10s %s\n",
+                     paddress (gdbarch, start),
+                     paddress (gdbarch, end),
+                     hex_string (end - start),
+                     hex_string (file_ofs),
+                     filename);
       });
 }
 
@@ -1223,7 +1285,7 @@ linux_core_info_proc (struct gdbarch *gdbarch, const char *args,
 
       exe = bfd_core_file_failing_command (core_bfd);
       if (exe != NULL)
-       printf_filtered ("exe = '%s'\n", exe);
+       gdb_printf ("exe = '%s'\n", exe);
       else
        warning (_("unable to find command name in core file"));
     }
@@ -1259,6 +1321,7 @@ typedef int linux_find_memory_region_ftype (ULONGEST vaddr, ULONGEST size,
                                            ULONGEST offset, ULONGEST inode,
                                            int read, int write,
                                            int exec, int modified,
+                                           bool memory_tagged,
                                            const char *filename,
                                            void *data);
 
@@ -1271,6 +1334,204 @@ typedef int linux_dump_mapping_p_ftype (filter_flags filterflags,
                                        ULONGEST addr,
                                        ULONGEST offset);
 
+/* Helper function to parse the contents of /proc/<pid>/smaps into a data
+   structure, for easy access.
+
+   DATA is the contents of the smaps file.  The parsed contents are stored
+   into the SMAPS vector.  */
+
+static std::vector<struct smaps_data>
+parse_smaps_data (const char *data,
+                 const std::string maps_filename)
+{
+  char *line, *t;
+
+  gdb_assert (data != nullptr);
+
+  line = strtok_r ((char *) data, "\n", &t);
+
+  std::vector<struct smaps_data> smaps;
+
+  while (line != NULL)
+    {
+      struct smaps_vmflags v;
+      int read, write, exec, priv;
+      int has_anonymous = 0;
+      int mapping_anon_p;
+      int mapping_file_p;
+
+      memset (&v, 0, sizeof (v));
+      struct mapping m = read_mapping (line);
+      mapping_anon_p = mapping_is_anonymous_p (m.filename);
+      /* If the mapping is not anonymous, then we can consider it
+        to be file-backed.  These two states (anonymous or
+        file-backed) seem to be exclusive, but they can actually
+        coexist.  For example, if a file-backed mapping has
+        "Anonymous:" pages (see more below), then the Linux
+        kernel will dump this mapping when the user specified
+        that she only wants anonymous mappings in the corefile
+        (*even* when she explicitly disabled the dumping of
+        file-backed mappings).  */
+      mapping_file_p = !mapping_anon_p;
+
+      /* Decode permissions.  */
+      auto has_perm = [&m] (char c)
+       { return m.permissions.find (c) != gdb::string_view::npos; };
+      read = has_perm ('r');
+      write = has_perm ('w');
+      exec = has_perm ('x');
+
+      /* 'private' here actually means VM_MAYSHARE, and not
+        VM_SHARED.  In order to know if a mapping is really
+        private or not, we must check the flag "sh" in the
+        VmFlags field.  This is done by decode_vmflags.  However,
+        if we are using a Linux kernel released before the commit
+        834f82e2aa9a8ede94b17b656329f850c1471514 (3.10), we will
+        not have the VmFlags there.  In this case, there is
+        really no way to know if we are dealing with VM_SHARED,
+        so we just assume that VM_MAYSHARE is enough.  */
+      priv = has_perm ('p');
+
+      /* Try to detect if region should be dumped by parsing smaps
+        counters.  */
+      for (line = strtok_r (NULL, "\n", &t);
+          line != NULL && line[0] >= 'A' && line[0] <= 'Z';
+          line = strtok_r (NULL, "\n", &t))
+       {
+         char keyword[64 + 1];
+
+         if (sscanf (line, "%64s", keyword) != 1)
+           {
+             warning (_("Error parsing {s,}maps file '%s'"),
+                      maps_filename.c_str ());
+             break;
+           }
+
+         if (strcmp (keyword, "Anonymous:") == 0)
+           {
+             /* Older Linux kernels did not support the
+                "Anonymous:" counter.  Check it here.  */
+             has_anonymous = 1;
+           }
+         else if (strcmp (keyword, "VmFlags:") == 0)
+           decode_vmflags (line, &v);
+
+         if (strcmp (keyword, "AnonHugePages:") == 0
+             || strcmp (keyword, "Anonymous:") == 0)
+           {
+             unsigned long number;
+
+             if (sscanf (line, "%*s%lu", &number) != 1)
+               {
+                 warning (_("Error parsing {s,}maps file '%s' number"),
+                          maps_filename.c_str ());
+                 break;
+               }
+             if (number > 0)
+               {
+                 /* Even if we are dealing with a file-backed
+                    mapping, if it contains anonymous pages we
+                    consider it to be *also* an anonymous
+                    mapping, because this is what the Linux
+                    kernel does:
+
+                    // Dump segments that have been written to.
+                    if (vma->anon_vma && FILTER(ANON_PRIVATE))
+                      goto whole;
+
+                   Note that if the mapping is already marked as
+                   file-backed (i.e., mapping_file_p is
+                   non-zero), then this is a special case, and
+                   this mapping will be dumped either when the
+                   user wants to dump file-backed *or* anonymous
+                   mappings.  */
+                 mapping_anon_p = 1;
+               }
+           }
+       }
+      /* Save the smaps entry to the vector.  */
+       struct smaps_data map;
+
+       map.start_address = m.addr;
+       map.end_address = m.endaddr;
+       map.filename = m.filename;
+       map.vmflags = v;
+       map.read = read? true : false;
+       map.write = write? true : false;
+       map.exec = exec? true : false;
+       map.priv = priv? true : false;
+       map.has_anonymous = has_anonymous;
+       map.mapping_anon_p = mapping_anon_p? true : false;
+       map.mapping_file_p = mapping_file_p? true : false;
+       map.offset = m.offset;
+       map.inode = m.inode;
+
+       smaps.emplace_back (map);
+    }
+
+  return smaps;
+}
+
+/* Helper that checks if an address is in a memory tag page for a live
+   process.  */
+
+static bool
+linux_process_address_in_memtag_page (CORE_ADDR address)
+{
+  if (current_inferior ()->fake_pid_p)
+    return false;
+
+  pid_t pid = current_inferior ()->pid;
+
+  std::string smaps_file = string_printf ("/proc/%d/smaps", pid);
+
+  gdb::unique_xmalloc_ptr<char> data
+    = target_fileio_read_stralloc (NULL, smaps_file.c_str ());
+
+  if (data == nullptr)
+    return false;
+
+  /* Parse the contents of smaps into a vector.  */
+  std::vector<struct smaps_data> smaps
+    = parse_smaps_data (data.get (), smaps_file);
+
+  for (const smaps_data &map : smaps)
+    {
+      /* Is the address within [start_address, end_address) in a page
+        mapped with memory tagging?  */
+      if (address >= map.start_address
+         && address < map.end_address
+         && map.vmflags.memory_tagging)
+       return true;
+    }
+
+  return false;
+}
+
+/* Helper that checks if an address is in a memory tag page for a core file
+   process.  */
+
+static bool
+linux_core_file_address_in_memtag_page (CORE_ADDR address)
+{
+  if (core_bfd == nullptr)
+    return false;
+
+  memtag_section_info info;
+  return get_next_core_memtag_section (core_bfd, nullptr, address, info);
+}
+
+/* See linux-tdep.h.  */
+
+bool
+linux_address_in_memtag_page (CORE_ADDR address)
+{
+  if (!target_has_execution ())
+    return linux_core_file_address_in_memtag_page (address);
+
+  return linux_process_address_in_memtag_page (address);
+}
+
 /* List memory regions in the inferior for a corefile.  */
 
 static int
@@ -1321,137 +1582,50 @@ linux_find_memory_regions_full (struct gdbarch *gdbarch,
       /* Older Linux kernels did not support /proc/PID/smaps.  */
       maps_filename = string_printf ("/proc/%d/maps", pid);
       data = target_fileio_read_stralloc (NULL, maps_filename.c_str ());
+
+      if (data == nullptr)
+       return 1;
     }
 
-  if (data != NULL)
+  /* Parse the contents of smaps into a vector.  */
+  std::vector<struct smaps_data> smaps
+    = parse_smaps_data (data.get (), maps_filename.c_str ());
+
+  for (const struct smaps_data &map : smaps)
     {
-      char *line, *t;
+      int should_dump_p = 0;
 
-      line = strtok_r (data.get (), "\n", &t);
-      while (line != NULL)
+      if (map.has_anonymous)
        {
-         ULONGEST addr, endaddr, offset, inode;
-         const char *permissions, *device, *filename;
-         struct smaps_vmflags v;
-         size_t permissions_len, device_len;
-         int read, write, exec, priv;
-         int has_anonymous = 0;
-         int should_dump_p = 0;
-         int mapping_anon_p;
-         int mapping_file_p;
-
-         memset (&v, 0, sizeof (v));
-         read_mapping (line, &addr, &endaddr, &permissions, &permissions_len,
-                       &offset, &device, &device_len, &inode, &filename);
-         mapping_anon_p = mapping_is_anonymous_p (filename);
-         /* If the mapping is not anonymous, then we can consider it
-            to be file-backed.  These two states (anonymous or
-            file-backed) seem to be exclusive, but they can actually
-            coexist.  For example, if a file-backed mapping has
-            "Anonymous:" pages (see more below), then the Linux
-            kernel will dump this mapping when the user specified
-            that she only wants anonymous mappings in the corefile
-            (*even* when she explicitly disabled the dumping of
-            file-backed mappings).  */
-         mapping_file_p = !mapping_anon_p;
-
-         /* Decode permissions.  */
-         read = (memchr (permissions, 'r', permissions_len) != 0);
-         write = (memchr (permissions, 'w', permissions_len) != 0);
-         exec = (memchr (permissions, 'x', permissions_len) != 0);
-         /* 'private' here actually means VM_MAYSHARE, and not
-            VM_SHARED.  In order to know if a mapping is really
-            private or not, we must check the flag "sh" in the
-            VmFlags field.  This is done by decode_vmflags.  However,
-            if we are using a Linux kernel released before the commit
-            834f82e2aa9a8ede94b17b656329f850c1471514 (3.10), we will
-            not have the VmFlags there.  In this case, there is
-            really no way to know if we are dealing with VM_SHARED,
-            so we just assume that VM_MAYSHARE is enough.  */
-         priv = memchr (permissions, 'p', permissions_len) != 0;
-
-         /* Try to detect if region should be dumped by parsing smaps
-            counters.  */
-         for (line = strtok_r (NULL, "\n", &t);
-              line != NULL && line[0] >= 'A' && line[0] <= 'Z';
-              line = strtok_r (NULL, "\n", &t))
-           {
-             char keyword[64 + 1];
-
-             if (sscanf (line, "%64s", keyword) != 1)
-               {
-                 warning (_("Error parsing {s,}maps file '%s'"),
-                          maps_filename.c_str ());
-                 break;
-               }
-
-             if (strcmp (keyword, "Anonymous:") == 0)
-               {
-                 /* Older Linux kernels did not support the
-                    "Anonymous:" counter.  Check it here.  */
-                 has_anonymous = 1;
-               }
-             else if (strcmp (keyword, "VmFlags:") == 0)
-               decode_vmflags (line, &v);
-
-             if (strcmp (keyword, "AnonHugePages:") == 0
-                 || strcmp (keyword, "Anonymous:") == 0)
-               {
-                 unsigned long number;
-
-                 if (sscanf (line, "%*s%lu", &number) != 1)
-                   {
-                     warning (_("Error parsing {s,}maps file '%s' number"),
-                              maps_filename.c_str ());
-                     break;
-                   }
-                 if (number > 0)
-                   {
-                     /* Even if we are dealing with a file-backed
-                        mapping, if it contains anonymous pages we
-                        consider it to be *also* an anonymous
-                        mapping, because this is what the Linux
-                        kernel does:
-
-                        // Dump segments that have been written to.
-                        if (vma->anon_vma && FILTER(ANON_PRIVATE))
-                               goto whole;
-
-                        Note that if the mapping is already marked as
-                        file-backed (i.e., mapping_file_p is
-                        non-zero), then this is a special case, and
-                        this mapping will be dumped either when the
-                        user wants to dump file-backed *or* anonymous
-                        mappings.  */
-                     mapping_anon_p = 1;
-                   }
-               }
-           }
-
-         if (has_anonymous)
-           should_dump_p = should_dump_mapping_p (filterflags, &v, priv,
-                                                  mapping_anon_p,
-                                                  mapping_file_p,
-                                                  filename, addr, offset);
-         else
-           {
-             /* Older Linux kernels did not support the "Anonymous:" counter.
-                If it is missing, we can't be sure - dump all the pages.  */
-             should_dump_p = 1;
-           }
-
-         /* Invoke the callback function to create the corefile segment.  */
-         if (should_dump_p)
-           func (addr, endaddr - addr, offset, inode,
-                 read, write, exec, 1, /* MODIFIED is true because we
-                                          want to dump the mapping.  */
-                 filename, obfd);
+         should_dump_p
+           = should_dump_mapping_p (filterflags, &map.vmflags,
+                                    map.priv,
+                                    map.mapping_anon_p,
+                                    map.mapping_file_p,
+                                    map.filename.c_str (),
+                                    map.start_address,
+                                    map.offset);
+       }
+      else
+       {
+         /* Older Linux kernels did not support the "Anonymous:" counter.
+            If it is missing, we can't be sure - dump all the pages.  */
+         should_dump_p = 1;
        }
 
-      return 0;
+      /* Invoke the callback function to create the corefile segment.  */
+      if (should_dump_p)
+       {
+         func (map.start_address, map.end_address - map.start_address,
+               map.offset, map.inode, map.read, map.write, map.exec,
+               1, /* MODIFIED is true because we want to dump
+                     the mapping.  */
+               map.vmflags.memory_tagging != 0,
+               map.filename.c_str (), obfd);
+       }
     }
 
-  return 1;
+  return 0;
 }
 
 /* A structure for passing information through
@@ -1475,12 +1649,14 @@ static int
 linux_find_memory_regions_thunk (ULONGEST vaddr, ULONGEST size,
                                 ULONGEST offset, ULONGEST inode,
                                 int read, int write, int exec, int modified,
+                                bool memory_tagged,
                                 const char *filename, void *arg)
 {
   struct linux_find_memory_regions_data *data
     = (struct linux_find_memory_regions_data *) arg;
 
-  return data->func (vaddr, size, read, write, exec, modified, data->obfd);
+  return data->func (vaddr, size, read, write, exec, modified, memory_tagged,
+                    data->obfd);
 }
 
 /* A variant of linux_find_memory_regions_full that is suitable as the
@@ -1529,6 +1705,7 @@ static int
 linux_make_mappings_callback (ULONGEST vaddr, ULONGEST size,
                              ULONGEST offset, ULONGEST inode,
                              int read, int write, int exec, int modified,
+                             bool memory_tagged,
                              const char *filename, void *data)
 {
   struct linux_make_mappings_data *map_data
@@ -1622,7 +1799,8 @@ linux_get_siginfo_data (thread_info *thread, struct gdbarch *gdbarch)
 
   gdb::byte_vector buf (TYPE_LENGTH (siginfo_type));
 
-  bytes_read = target_read (current_top_target (), TARGET_OBJECT_SIGNAL_INFO, NULL,
+  bytes_read = target_read (current_inferior ()->top_target (),
+                           TARGET_OBJECT_SIGNAL_INFO, NULL,
                            buf.data (), 0, TYPE_LENGTH (siginfo_type));
   if (bytes_read != TYPE_LENGTH (siginfo_type))
     buf.clear ();
@@ -1689,7 +1867,6 @@ linux_fill_prpsinfo (struct elf_internal_linux_prpsinfo *p)
   char filename[100];
   /* The basename of the executable.  */
   const char *basename;
-  const char *infargs;
   /* Temporary buffer.  */
   char *tmpstr;
   /* The valid states of a process, according to the Linux kernel.  */
@@ -1733,12 +1910,12 @@ linux_fill_prpsinfo (struct elf_internal_linux_prpsinfo *p)
   strncpy (p->pr_fname, basename, sizeof (p->pr_fname) - 1);
   p->pr_fname[sizeof (p->pr_fname) - 1] = '\0';
 
-  infargs = get_inferior_args ();
+  const std::string &infargs = current_inferior ()->args ();
 
   /* The arguments of the program.  */
   std::string psargs = fname.get ();
-  if (infargs != NULL)
-    psargs = psargs + " " + infargs;
+  if (!infargs.empty ())
+    psargs += ' ' + infargs;
 
   strncpy (p->pr_psargs, psargs.c_str (), sizeof (p->pr_psargs) - 1);
   p->pr_psargs[sizeof (p->pr_psargs) - 1] = '\0';
@@ -1902,7 +2079,7 @@ linux_make_corefile_notes (struct gdbarch *gdbarch, bfd *obfd, int *note_size)
   thread_info *signalled_thr = gcore_find_signalled_thread ();
   gdb_signal stop_signal;
   if (signalled_thr != nullptr)
-    stop_signal = signalled_thr->suspend.stop_signal;
+    stop_signal = signalled_thr->stop_signal ();
   else
     stop_signal = GDB_SIGNAL_0;
 
@@ -1924,7 +2101,8 @@ linux_make_corefile_notes (struct gdbarch *gdbarch, bfd *obfd, int *note_size)
 
   /* Auxillary vector.  */
   gdb::optional<gdb::byte_vector> auxv =
-    target_read_alloc (current_top_target (), TARGET_OBJECT_AUXV, NULL);
+    target_read_alloc (current_inferior ()->top_target (),
+                      TARGET_OBJECT_AUXV, NULL);
   if (auxv && !auxv->empty ())
     {
       note_data.reset (elfcore_write_note (obfd, note_data.release (),
@@ -2204,7 +2382,8 @@ linux_vsyscall_range_raw (struct gdbarch *gdbarch, struct mem_range *range)
   char filename[100];
   long pid;
 
-  if (target_auxv_search (current_top_target (), AT_SYSINFO_EHDR, &range->start) <= 0)
+  if (target_auxv_search (current_inferior ()->top_target (),
+                         AT_SYSINFO_EHDR, &range->start) <= 0)
     return 0;
 
   /* It doesn't make sense to access the host's /proc when debugging a
@@ -2394,14 +2573,15 @@ linux_displaced_step_location (struct gdbarch *gdbarch)
      local-store address and is thus not usable as displaced stepping
      location.  The auxiliary vector gets us the PowerPC-side entry
      point address instead.  */
-  if (target_auxv_search (current_top_target (), AT_ENTRY, &addr) <= 0)
+  if (target_auxv_search (current_inferior ()->top_target (),
+                         AT_ENTRY, &addr) <= 0)
     throw_error (NOT_SUPPORTED_ERROR,
                 _("Cannot find AT_ENTRY auxiliary vector entry."));
 
   /* Make certain that the address points at real code, and not a
      function descriptor.  */
-  addr = gdbarch_convert_from_func_ptr_addr (gdbarch, addr,
-                                            current_top_target ());
+  addr = gdbarch_convert_from_func_ptr_addr
+    (gdbarch, addr, current_inferior ()->top_target ());
 
   /* Inferior calls also use the entry point as a breakpoint location.
      We don't want displaced stepping to interfere with those
@@ -2510,8 +2690,8 @@ static void
 show_use_coredump_filter (struct ui_file *file, int from_tty,
                          struct cmd_list_element *c, const char *value)
 {
-  fprintf_filtered (file, _("Use of /proc/PID/coredump_filter file to generate"
-                           " corefiles is %s.\n"), value);
+  gdb_printf (file, _("Use of /proc/PID/coredump_filter file to generate"
+                     " corefiles is %s.\n"), value);
 }
 
 /* Display whether the gcore command is dumping mappings marked with
@@ -2521,8 +2701,8 @@ static void
 show_dump_excluded_mappings (struct ui_file *file, int from_tty,
                             struct cmd_list_element *c, const char *value)
 {
-  fprintf_filtered (file, _("Dumping of mappings marked with the VM_DONTDUMP"
-                           " flag is %s.\n"), value);
+  gdb_printf (file, _("Dumping of mappings marked with the VM_DONTDUMP"
+                     " flag is %s.\n"), value);
 }
 
 /* To be called from the various GDB_OSABI_LINUX handlers for the
@@ -2576,9 +2756,12 @@ _initialize_linux_tdep ()
     gdbarch_data_register_pre_init (init_linux_gdbarch_data);
 
   /* Observers used to invalidate the cache when needed.  */
-  gdb::observers::inferior_exit.attach (invalidate_linux_cache_inf);
-  gdb::observers::inferior_appeared.attach (invalidate_linux_cache_inf);
-  gdb::observers::inferior_execd.attach (invalidate_linux_cache_inf);
+  gdb::observers::inferior_exit.attach (invalidate_linux_cache_inf,
+                                       "linux-tdep");
+  gdb::observers::inferior_appeared.attach (invalidate_linux_cache_inf,
+                                           "linux-tdep");
+  gdb::observers::inferior_execd.attach (invalidate_linux_cache_inf,
+                                        "linux-tdep");
 
   add_setshow_boolean_cmd ("use-coredump-filter", class_files,
                           &use_coredump_filter, _("\
@@ -2604,3 +2787,62 @@ more information about this file, refer to the manpage of proc(5) and core(5).")
                           NULL, show_dump_excluded_mappings,
                           &setlist, &showlist);
 }
+
+/* Fetch (and possibly build) an appropriate `link_map_offsets' for
+   ILP32/LP64 Linux systems which don't have the r_ldsomap field.  */
+
+link_map_offsets *
+linux_ilp32_fetch_link_map_offsets ()
+{
+  static link_map_offsets lmo;
+  static link_map_offsets *lmp = nullptr;
+
+  if (lmp == nullptr)
+    {
+      lmp = &lmo;
+
+      lmo.r_version_offset = 0;
+      lmo.r_version_size = 4;
+      lmo.r_map_offset = 4;
+      lmo.r_brk_offset = 8;
+      lmo.r_ldsomap_offset = -1;
+
+      /* Everything we need is in the first 20 bytes.  */
+      lmo.link_map_size = 20;
+      lmo.l_addr_offset = 0;
+      lmo.l_name_offset = 4;
+      lmo.l_ld_offset = 8;
+      lmo.l_next_offset = 12;
+      lmo.l_prev_offset = 16;
+    }
+
+  return lmp;
+}
+
+link_map_offsets *
+linux_lp64_fetch_link_map_offsets ()
+{
+  static link_map_offsets lmo;
+  static link_map_offsets *lmp = nullptr;
+
+  if (lmp == nullptr)
+    {
+      lmp = &lmo;
+
+      lmo.r_version_offset = 0;
+      lmo.r_version_size = 4;
+      lmo.r_map_offset = 8;
+      lmo.r_brk_offset = 16;
+      lmo.r_ldsomap_offset = -1;
+
+      /* Everything we need is in the first 40 bytes.  */
+      lmo.link_map_size = 40;
+      lmo.l_addr_offset = 0;
+      lmo.l_name_offset = 8;
+      lmo.l_ld_offset = 16;
+      lmo.l_next_offset = 24;
+      lmo.l_prev_offset = 32;
+    }
+
+  return lmp;
+}