Add aarch64_get_hwcap functions for reading the HWCAP.
From this extract the PACA value and use this to enable pauth.
gdb/ChangeLog:
* aarch64-linux-nat.c
(aarch64_linux_nat_target::read_description): Read PACA hwcap.
* aarch64-linux-tdep.c
(aarch64_linux_core_read_description): Likewise.
(aarch64_linux_get_hwcap): New function.
* aarch64-linux-tdep.h (AARCH64_HWCAP_PACA): New define.
(aarch64_linux_get_hwcap): New declaration.
gdb/gdbserver/ChangeLog:
* linux-aarch64-low.c (AARCH64_HWCAP_PACA): New define.
(aarch64_get_hwcap): New function.
(aarch64_arch_setup): Read APIA hwcap.
+2019-03-22 Alan Hayward <alan.hayward@arm.com>
+ Jiong Wang <jiong.wang@arm.com>
+
+ * aarch64-linux-nat.c
+ (aarch64_linux_nat_target::read_description): Read PACA hwcap.
+ * aarch64-linux-tdep.c
+ (aarch64_linux_core_read_description): Likewise.
+ (aarch64_linux_get_hwcap): New function.
+ * aarch64-linux-tdep.h (AARCH64_HWCAP_PACA): New define.
+ (aarch64_linux_get_hwcap): New declaration.
+
2019-03-22 Alan Hayward <alan.hayward@arm.com>
Jiong Wang <jiong.wang@arm.com>
if (ret == 0)
return tdesc_arm_with_neon;
- /* pauth not yet supported. */
- return aarch64_read_description (aarch64_sve_get_vq (tid), false);
+ CORE_ADDR hwcap = 0;
+ bool pauth_p = aarch64_linux_get_hwcap (this, &hwcap)
+ && (hwcap & AARCH64_HWCAP_PACA);
+
+ return aarch64_read_description (aarch64_sve_get_vq (tid), pauth_p);
}
/* Convert a native/host siginfo object, into/from the siginfo in the
{
CORE_ADDR aarch64_hwcap = 0;
- if (target_auxv_search (target, AT_HWCAP, &aarch64_hwcap) != 1)
- return NULL;
+ if (!aarch64_linux_get_hwcap (target, &aarch64_hwcap))
+ return nullptr;
- /* pauth not yet supported. */
return aarch64_read_description (aarch64_linux_core_read_vq (gdbarch, abfd),
- false);
+ aarch64_hwcap & AARCH64_HWCAP_PACA);
}
/* Implementation of `gdbarch_stap_is_single_operand', as defined in
return NULL;
}
+/* See aarch64-linux-tdep.h. */
+
+bool
+aarch64_linux_get_hwcap (struct target_ops *target, CORE_ADDR *hwcap)
+{
+ *hwcap = 0;
+ return target_auxv_search (target, AT_HWCAP, hwcap) == 1;
+}
+
static void
aarch64_linux_init_abi (struct gdbarch_info info, struct gdbarch *gdbarch)
{
extern const struct regset aarch64_linux_gregset;
extern const struct regset aarch64_linux_fpregset;
+/* Matches HWCAP_PACA in kernel header arch/arm64/include/uapi/asm/hwcap.h. */
+#define AARCH64_HWCAP_PACA (1 << 30)
+
+/* Fetch the AT_HWCAP entry from the auxv vector for the given TARGET. */
+bool aarch64_linux_get_hwcap (struct target_ops *target, CORE_ADDR *hwcap);
+
#endif /* AARCH64_LINUX_TDEP_H */
+2019-03-22 Alan Hayward <alan.hayward@arm.com>
+ Jiong Wang <jiong.wang@arm.com>
+
+ * linux-aarch64-low.c (AARCH64_HWCAP_PACA): New define.
+ (aarch64_get_hwcap): New function.
+ (aarch64_arch_setup): Read APIA hwcap.
+
2019-03-22 Alan Hayward <alan.hayward@arm.com>
Jiong Wang <jiong.wang@arm.com>
*child->priv->arch_private = *parent->priv->arch_private;
}
+/* Matches HWCAP_PACA in kernel header arch/arm64/include/uapi/asm/hwcap.h. */
+#define AARCH64_HWCAP_PACA (1 << 30)
+
+/* Fetch the AT_HWCAP entry from the auxv vector. */
+
+static bool
+aarch64_get_hwcap (unsigned long *valp)
+{
+ unsigned char *data = (unsigned char *) alloca (16);
+ int offset = 0;
+
+ while ((*the_target->read_auxv) (offset, data, 16) == 16)
+ {
+ unsigned long *data_p = (unsigned long *)data;
+ if (data_p[0] == AT_HWCAP)
+ {
+ *valp = data_p[1];
+ return true;
+ }
+
+ offset += 16;
+ }
+
+ *valp = 0;
+ return false;
+}
+
/* Implementation of linux_target_ops method "arch_setup". */
static void
if (is_elf64)
{
uint64_t vq = aarch64_sve_get_vq (tid);
- /* pauth not yet supported. */
- current_process ()->tdesc = aarch64_linux_read_description (vq, false);
+ unsigned long hwcap = 0;
+ bool pauth_p = aarch64_get_hwcap (&hwcap) && (hwcap & AARCH64_HWCAP_PACA);
+
+ current_process ()->tdesc = aarch64_linux_read_description (vq, pauth_p);
}
else
current_process ()->tdesc = tdesc_arm_with_neon;