binutils-gdb/gdb/features/riscv/64bit-fpu.c
Andrew Burgess 4749b84b51 gdb/riscv: better support for fflags and frm registers
First, some background on the RISC-V registers fflags, frm, and fcsr.

These three registers all relate to the floating-point status and
control mechanism on RISC-V.  The fcsr is the floatint-point control
status register, and consists of two parts, the flags (bits 0 to 4)
and the rounding-mode (bits 5 to 7).

The fcsr register is just one of many control/status registers (or
CSRs) available on RISC-V.  The fflags and frm registers are also
CSRs.  These CSRs are aliases for the relevant parts of the fcsr
register.  So fflags is an alias for bits 0 to 4 of fcsr, and frm is
an alias for bits 5 to 7 of fcsr.

This means that a user can change the floating-point rounding mode
either, by writing a complete new value into fcsr, or by writing just
the rounding mode into frm.

How this impacts on GDB is like this: a target description could,
legitimately include all three registers, fcsr, fflags, and frm.  The
QEMU target currently does this, and this makes sense.  The target is
emulating the complete system, and has all three CSRs available, so
why not tell GDB about this.

In contrast, the RISC-V native Linux target only has access to the
fcsr.  This is because the ptrace data structure that the kernel uses
for reading and writing floating point state only contains a copy of
the fcsr, after all, this one field really contains both the fflags
and frm fields, so why carry around duplicate data.

So, we might expect that the target description for the RISC-V native
Linux GDB would only contain the fcsr register.  Unfortunately, this
is not the case.  The RISC-V native Linux target uses GDB's builtin
target descriptions by calling riscv_lookup_target_description, this
will then add an fpu feature from gdb/features/riscv, either
32bit-fpu.xml or 64bit-fpu.xml.  The problem, is that these features
include an entry for fcsr, fflags, and frm.  This means that GDB
expects the target to handle reading and writing these registers.  And
the RISC-V native Linux target currently doesn't.

In riscv_linux_nat_target::store_registers and
riscv_linux_nat_target::fetch_registers only the fcsr register is
handled, this means that, for RISC-V native Linux, the fflags and frm
registers always show up as <unavailable> - they are present in the
target description, but the target doesn't know how to access the
registers.

A final complication relating to these floating pointer CSRs is which
target description feature the registers appear in.

These registers are CSRs, so it would seem sensible that these
registers should appear in the CSR target description feature.

However, when I first added RISC-V target description support, I was
using a RISC-V simulator that didn't support any CSRs other than the
floating point related ones.  This simulator bundled all the float
related CSRs into the fpu target feature.  This didn't feel completely
unreasonable to me, and so I had GDB check for these registers in
either target feature.

In this commit I make some changes relating to how GDB handles the
three floating point CSR:

1. Remove fflags and frm from 32bit-fpu.xml and 64bit-fpu.xml.  This
means that the default RISC-V target description (which RISC-V native
FreeBSD), and the target descriptions created for RISC-V native Linux,
will not include these registers.  There's nothing stopping some other
target (e.g. QEMU) from continuing to include all three of these CSRs,
the code in riscv-tdep.c continues to check for all three of these
registers, and will handle them correctly if they are present.

2. If a target supplied fcsr, but does not supply fflags and/or frm,
then RISC-V GDB will now create two pseudo registers in order to
emulate the two missing CSRs.  These new pseudo-registers do the
obvious thing of just reading and writing the fcsr register.

3. With the new pseudo-registers we can no longer make use of the GDB
register numbers RISCV_CSR_FFLAGS_REGNUM and RISCV_CSR_FRM_REGNUM.
These will be the numbers used if the target supplies the registers in
its target description, but, if GDB falls back to using
pseudo-registers, then new, unique numbers will be used.  To handle
this I've added riscv_gdbarch_tdep::fflags_regnum and
riscv_gdbarch_tdep::frm_regnum, I've then updated the RISC-V code to
compare against these fields.

When adding the pseudo-register support, it is important that the
pseudo-register numbers are calculated after the call to
tdesc_use_registers.  This is because we don't know the total number
of physical registers until after this call, and the psuedo-register
numbers must follow on from the real (target supplied) registers.

I've updated some tests to include more testing of the fflags and frm
registers, as well as adding a new test.
2022-08-31 16:07:05 +01:00

57 lines
3.2 KiB
C

/* THIS FILE IS GENERATED. -*- buffer-read-only: t -*- vi:set ro:
Original: 64bit-fpu.xml */
#include "gdbsupport/tdesc.h"
static int
create_feature_riscv_64bit_fpu (struct target_desc *result, long regnum)
{
struct tdesc_feature *feature;
feature = tdesc_create_feature (result, "org.gnu.gdb.riscv.fpu");
tdesc_type_with_fields *type_with_fields;
type_with_fields = tdesc_create_union (feature, "riscv_double");
tdesc_type *field_type;
field_type = tdesc_named_type (feature, "ieee_single");
tdesc_add_field (type_with_fields, "float", field_type);
field_type = tdesc_named_type (feature, "ieee_double");
tdesc_add_field (type_with_fields, "double", field_type);
regnum = 33;
tdesc_create_reg (feature, "ft0", regnum++, 1, NULL, 64, "riscv_double");
tdesc_create_reg (feature, "ft1", regnum++, 1, NULL, 64, "riscv_double");
tdesc_create_reg (feature, "ft2", regnum++, 1, NULL, 64, "riscv_double");
tdesc_create_reg (feature, "ft3", regnum++, 1, NULL, 64, "riscv_double");
tdesc_create_reg (feature, "ft4", regnum++, 1, NULL, 64, "riscv_double");
tdesc_create_reg (feature, "ft5", regnum++, 1, NULL, 64, "riscv_double");
tdesc_create_reg (feature, "ft6", regnum++, 1, NULL, 64, "riscv_double");
tdesc_create_reg (feature, "ft7", regnum++, 1, NULL, 64, "riscv_double");
tdesc_create_reg (feature, "fs0", regnum++, 1, NULL, 64, "riscv_double");
tdesc_create_reg (feature, "fs1", regnum++, 1, NULL, 64, "riscv_double");
tdesc_create_reg (feature, "fa0", regnum++, 1, NULL, 64, "riscv_double");
tdesc_create_reg (feature, "fa1", regnum++, 1, NULL, 64, "riscv_double");
tdesc_create_reg (feature, "fa2", regnum++, 1, NULL, 64, "riscv_double");
tdesc_create_reg (feature, "fa3", regnum++, 1, NULL, 64, "riscv_double");
tdesc_create_reg (feature, "fa4", regnum++, 1, NULL, 64, "riscv_double");
tdesc_create_reg (feature, "fa5", regnum++, 1, NULL, 64, "riscv_double");
tdesc_create_reg (feature, "fa6", regnum++, 1, NULL, 64, "riscv_double");
tdesc_create_reg (feature, "fa7", regnum++, 1, NULL, 64, "riscv_double");
tdesc_create_reg (feature, "fs2", regnum++, 1, NULL, 64, "riscv_double");
tdesc_create_reg (feature, "fs3", regnum++, 1, NULL, 64, "riscv_double");
tdesc_create_reg (feature, "fs4", regnum++, 1, NULL, 64, "riscv_double");
tdesc_create_reg (feature, "fs5", regnum++, 1, NULL, 64, "riscv_double");
tdesc_create_reg (feature, "fs6", regnum++, 1, NULL, 64, "riscv_double");
tdesc_create_reg (feature, "fs7", regnum++, 1, NULL, 64, "riscv_double");
tdesc_create_reg (feature, "fs8", regnum++, 1, NULL, 64, "riscv_double");
tdesc_create_reg (feature, "fs9", regnum++, 1, NULL, 64, "riscv_double");
tdesc_create_reg (feature, "fs10", regnum++, 1, NULL, 64, "riscv_double");
tdesc_create_reg (feature, "fs11", regnum++, 1, NULL, 64, "riscv_double");
tdesc_create_reg (feature, "ft8", regnum++, 1, NULL, 64, "riscv_double");
tdesc_create_reg (feature, "ft9", regnum++, 1, NULL, 64, "riscv_double");
tdesc_create_reg (feature, "ft10", regnum++, 1, NULL, 64, "riscv_double");
tdesc_create_reg (feature, "ft11", regnum++, 1, NULL, 64, "riscv_double");
regnum = 68;
tdesc_create_reg (feature, "fcsr", regnum++, 1, NULL, 32, "int");
return regnum;
}