diff mbox series

[v3,07/14] xen/riscv: introduce exception context

Message ID 5aa05592497ba9c4d207185d81981442d43ba676.1675780434.git.oleksii.kurochko@gmail.com (mailing list archive)
State Superseded
Headers show
Series RISCV basic exception handling implementation | expand

Commit Message

Oleksii Kurochko Feb. 7, 2023, 2:46 p.m. UTC
The patch introduces a set of registers which should be saved to and
restored from a stack after an exception occurs and a set of defines
which will be used during exception context saving/restoring.

Originally <asm/processor.h> header was introduced in the patch series
from Bobby so partially it was
re-used and the following changes were done:
  - Move all RISCV_CPU_USER_REGS_* to asm/asm-offsets.c
  - Remove RISCV_CPU_USER_REGS_OFFSET & RISCV_CPU_USER_REGS_SIZE as
    there is no sense in them after RISCV_CPU_USER_REGS_* were moved to
    asm/asm-offsets.c
  - Remove RISCV_PCPUINFO_* as they aren't needed for current status of
    the RISC-V port
  - register_t renamed to unsigned long
  - rename wait_for_interrupt to wfi

Signed-off-by: Bobby Eshleman <bobby.eshleman@gmail.com>
Signed-off-by: Oleksii Kurochko <oleksii.kurochko@gmail.com>
---
Changes in V3:
  - update code style for die() function
---
Changes in V2:
  - All the changes were added to the commit message.
  - temporarily was added function die() to stop exectution it will be
    removed after panic() will be available.
---
 xen/arch/riscv/include/asm/processor.h | 83 ++++++++++++++++++++++++++
 xen/arch/riscv/riscv64/asm-offsets.c   | 53 ++++++++++++++++
 2 files changed, 136 insertions(+)
 create mode 100644 xen/arch/riscv/include/asm/processor.h

Comments

Alistair Francis Feb. 9, 2023, 1:06 a.m. UTC | #1
On Wed, Feb 8, 2023 at 12:47 AM Oleksii Kurochko
<oleksii.kurochko@gmail.com> wrote:
>
> The patch introduces a set of registers which should be saved to and
> restored from a stack after an exception occurs and a set of defines
> which will be used during exception context saving/restoring.
>
> Originally <asm/processor.h> header was introduced in the patch series
> from Bobby so partially it was
> re-used and the following changes were done:
>   - Move all RISCV_CPU_USER_REGS_* to asm/asm-offsets.c
>   - Remove RISCV_CPU_USER_REGS_OFFSET & RISCV_CPU_USER_REGS_SIZE as
>     there is no sense in them after RISCV_CPU_USER_REGS_* were moved to
>     asm/asm-offsets.c
>   - Remove RISCV_PCPUINFO_* as they aren't needed for current status of
>     the RISC-V port
>   - register_t renamed to unsigned long
>   - rename wait_for_interrupt to wfi
>
> Signed-off-by: Bobby Eshleman <bobby.eshleman@gmail.com>
> Signed-off-by: Oleksii Kurochko <oleksii.kurochko@gmail.com>

Reviewed-by: Alistair Francis <alistair.francis@wdc.com>

Alistair

> ---
> Changes in V3:
>   - update code style for die() function
> ---
> Changes in V2:
>   - All the changes were added to the commit message.
>   - temporarily was added function die() to stop exectution it will be
>     removed after panic() will be available.
> ---
>  xen/arch/riscv/include/asm/processor.h | 83 ++++++++++++++++++++++++++
>  xen/arch/riscv/riscv64/asm-offsets.c   | 53 ++++++++++++++++
>  2 files changed, 136 insertions(+)
>  create mode 100644 xen/arch/riscv/include/asm/processor.h
>
> diff --git a/xen/arch/riscv/include/asm/processor.h b/xen/arch/riscv/include/asm/processor.h
> new file mode 100644
> index 0000000000..a71448e02e
> --- /dev/null
> +++ b/xen/arch/riscv/include/asm/processor.h
> @@ -0,0 +1,83 @@
> +/* SPDX-License-Identifier: MIT */
> +/******************************************************************************
> + *
> + * Copyright 2019 (C) Alistair Francis <alistair.francis@wdc.com>
> + * Copyright 2021 (C) Bobby Eshleman <bobby.eshleman@gmail.com>
> + * Copyright 2023 (C) Vates
> + *
> + */
> +
> +#ifndef _ASM_RISCV_PROCESSOR_H
> +#define _ASM_RISCV_PROCESSOR_H
> +
> +#ifndef __ASSEMBLY__
> +
> +/* On stack VCPU state */
> +struct cpu_user_regs
> +{
> +    unsigned long zero;
> +    unsigned long ra;
> +    unsigned long sp;
> +    unsigned long gp;
> +    unsigned long tp;
> +    unsigned long t0;
> +    unsigned long t1;
> +    unsigned long t2;
> +    unsigned long s0;
> +    unsigned long s1;
> +    unsigned long a0;
> +    unsigned long a1;
> +    unsigned long a2;
> +    unsigned long a3;
> +    unsigned long a4;
> +    unsigned long a5;
> +    unsigned long a6;
> +    unsigned long a7;
> +    unsigned long s2;
> +    unsigned long s3;
> +    unsigned long s4;
> +    unsigned long s5;
> +    unsigned long s6;
> +    unsigned long s7;
> +    unsigned long s8;
> +    unsigned long s9;
> +    unsigned long s10;
> +    unsigned long s11;
> +    unsigned long t3;
> +    unsigned long t4;
> +    unsigned long t5;
> +    unsigned long t6;
> +    unsigned long sepc;
> +    unsigned long sstatus;
> +    /* pointer to previous stack_cpu_regs */
> +    unsigned long pregs;
> +};
> +
> +static inline void wfi(void)
> +{
> +    __asm__ __volatile__ ("wfi");
> +}
> +
> +/*
> + * panic() isn't available at the moment so an infinite loop will be
> + * used temporarily.
> + * TODO: change it to panic()
> + */
> +static inline void die(void)
> +{
> +    for ( ;; )
> +        wfi();
> +}
> +
> +#endif /* __ASSEMBLY__ */
> +
> +#endif /* _ASM_RISCV_PROCESSOR_H */
> +
> +/*
> + * Local variables:
> + * mode: C
> + * c-file-style: "BSD"
> + * c-basic-offset: 4
> + * indent-tabs-mode: nil
> + * End:
> + */
> diff --git a/xen/arch/riscv/riscv64/asm-offsets.c b/xen/arch/riscv/riscv64/asm-offsets.c
> index e69de29bb2..d632b75c2a 100644
> --- a/xen/arch/riscv/riscv64/asm-offsets.c
> +++ b/xen/arch/riscv/riscv64/asm-offsets.c
> @@ -0,0 +1,53 @@
> +#define COMPILE_OFFSETS
> +
> +#include <asm/processor.h>
> +#include <xen/types.h>
> +
> +#define DEFINE(_sym, _val)                                                 \
> +    asm volatile ("\n.ascii\"==>#define " #_sym " %0 /* " #_val " */<==\"" \
> +                  : : "i" (_val) )
> +#define BLANK()                                                            \
> +    asm volatile ( "\n.ascii\"==><==\"" : : )
> +#define OFFSET(_sym, _str, _mem)                                           \
> +    DEFINE(_sym, offsetof(_str, _mem));
> +
> +void asm_offsets(void)
> +{
> +    BLANK();
> +    DEFINE(CPU_USER_REGS_SIZE, sizeof(struct cpu_user_regs));
> +    OFFSET(CPU_USER_REGS_ZERO, struct cpu_user_regs, zero);
> +    OFFSET(CPU_USER_REGS_RA, struct cpu_user_regs, ra);
> +    OFFSET(CPU_USER_REGS_SP, struct cpu_user_regs, sp);
> +    OFFSET(CPU_USER_REGS_GP, struct cpu_user_regs, gp);
> +    OFFSET(CPU_USER_REGS_TP, struct cpu_user_regs, tp);
> +    OFFSET(CPU_USER_REGS_T0, struct cpu_user_regs, t0);
> +    OFFSET(CPU_USER_REGS_T1, struct cpu_user_regs, t1);
> +    OFFSET(CPU_USER_REGS_T2, struct cpu_user_regs, t2);
> +    OFFSET(CPU_USER_REGS_S0, struct cpu_user_regs, s0);
> +    OFFSET(CPU_USER_REGS_S1, struct cpu_user_regs, s1);
> +    OFFSET(CPU_USER_REGS_A0, struct cpu_user_regs, a0);
> +    OFFSET(CPU_USER_REGS_A1, struct cpu_user_regs, a1);
> +    OFFSET(CPU_USER_REGS_A2, struct cpu_user_regs, a2);
> +    OFFSET(CPU_USER_REGS_A3, struct cpu_user_regs, a3);
> +    OFFSET(CPU_USER_REGS_A4, struct cpu_user_regs, a4);
> +    OFFSET(CPU_USER_REGS_A5, struct cpu_user_regs, a5);
> +    OFFSET(CPU_USER_REGS_A6, struct cpu_user_regs, a6);
> +    OFFSET(CPU_USER_REGS_A7, struct cpu_user_regs, a7);
> +    OFFSET(CPU_USER_REGS_S2, struct cpu_user_regs, s2);
> +    OFFSET(CPU_USER_REGS_S3, struct cpu_user_regs, s3);
> +    OFFSET(CPU_USER_REGS_S4, struct cpu_user_regs, s4);
> +    OFFSET(CPU_USER_REGS_S5, struct cpu_user_regs, s5);
> +    OFFSET(CPU_USER_REGS_S6, struct cpu_user_regs, s6);
> +    OFFSET(CPU_USER_REGS_S7, struct cpu_user_regs, s7);
> +    OFFSET(CPU_USER_REGS_S8, struct cpu_user_regs, s8);
> +    OFFSET(CPU_USER_REGS_S9, struct cpu_user_regs, s9);
> +    OFFSET(CPU_USER_REGS_S10, struct cpu_user_regs, s10);
> +    OFFSET(CPU_USER_REGS_S11, struct cpu_user_regs, s11);
> +    OFFSET(CPU_USER_REGS_T3, struct cpu_user_regs, t3);
> +    OFFSET(CPU_USER_REGS_T4, struct cpu_user_regs, t4);
> +    OFFSET(CPU_USER_REGS_T5, struct cpu_user_regs, t5);
> +    OFFSET(CPU_USER_REGS_T6, struct cpu_user_regs, t6);
> +    OFFSET(CPU_USER_REGS_SEPC, struct cpu_user_regs, sepc);
> +    OFFSET(CPU_USER_REGS_SSTATUS, struct cpu_user_regs, sstatus);
> +    OFFSET(CPU_USER_REGS_PREGS, struct cpu_user_regs, pregs);
> +}
> --
> 2.39.0
>
>
diff mbox series

Patch

diff --git a/xen/arch/riscv/include/asm/processor.h b/xen/arch/riscv/include/asm/processor.h
new file mode 100644
index 0000000000..a71448e02e
--- /dev/null
+++ b/xen/arch/riscv/include/asm/processor.h
@@ -0,0 +1,83 @@ 
+/* SPDX-License-Identifier: MIT */
+/******************************************************************************
+ *
+ * Copyright 2019 (C) Alistair Francis <alistair.francis@wdc.com>
+ * Copyright 2021 (C) Bobby Eshleman <bobby.eshleman@gmail.com>
+ * Copyright 2023 (C) Vates
+ *
+ */
+
+#ifndef _ASM_RISCV_PROCESSOR_H
+#define _ASM_RISCV_PROCESSOR_H
+
+#ifndef __ASSEMBLY__
+
+/* On stack VCPU state */
+struct cpu_user_regs
+{
+    unsigned long zero;
+    unsigned long ra;
+    unsigned long sp;
+    unsigned long gp;
+    unsigned long tp;
+    unsigned long t0;
+    unsigned long t1;
+    unsigned long t2;
+    unsigned long s0;
+    unsigned long s1;
+    unsigned long a0;
+    unsigned long a1;
+    unsigned long a2;
+    unsigned long a3;
+    unsigned long a4;
+    unsigned long a5;
+    unsigned long a6;
+    unsigned long a7;
+    unsigned long s2;
+    unsigned long s3;
+    unsigned long s4;
+    unsigned long s5;
+    unsigned long s6;
+    unsigned long s7;
+    unsigned long s8;
+    unsigned long s9;
+    unsigned long s10;
+    unsigned long s11;
+    unsigned long t3;
+    unsigned long t4;
+    unsigned long t5;
+    unsigned long t6;
+    unsigned long sepc;
+    unsigned long sstatus;
+    /* pointer to previous stack_cpu_regs */
+    unsigned long pregs;
+};
+
+static inline void wfi(void)
+{
+    __asm__ __volatile__ ("wfi");
+}
+
+/*
+ * panic() isn't available at the moment so an infinite loop will be
+ * used temporarily.
+ * TODO: change it to panic()
+ */
+static inline void die(void)
+{
+    for ( ;; )
+        wfi();
+}
+
+#endif /* __ASSEMBLY__ */
+
+#endif /* _ASM_RISCV_PROCESSOR_H */
+
+/*
+ * Local variables:
+ * mode: C
+ * c-file-style: "BSD"
+ * c-basic-offset: 4
+ * indent-tabs-mode: nil
+ * End:
+ */
diff --git a/xen/arch/riscv/riscv64/asm-offsets.c b/xen/arch/riscv/riscv64/asm-offsets.c
index e69de29bb2..d632b75c2a 100644
--- a/xen/arch/riscv/riscv64/asm-offsets.c
+++ b/xen/arch/riscv/riscv64/asm-offsets.c
@@ -0,0 +1,53 @@ 
+#define COMPILE_OFFSETS
+
+#include <asm/processor.h>
+#include <xen/types.h>
+
+#define DEFINE(_sym, _val)                                                 \
+    asm volatile ("\n.ascii\"==>#define " #_sym " %0 /* " #_val " */<==\"" \
+                  : : "i" (_val) )
+#define BLANK()                                                            \
+    asm volatile ( "\n.ascii\"==><==\"" : : )
+#define OFFSET(_sym, _str, _mem)                                           \
+    DEFINE(_sym, offsetof(_str, _mem));
+
+void asm_offsets(void)
+{
+    BLANK();
+    DEFINE(CPU_USER_REGS_SIZE, sizeof(struct cpu_user_regs));
+    OFFSET(CPU_USER_REGS_ZERO, struct cpu_user_regs, zero);
+    OFFSET(CPU_USER_REGS_RA, struct cpu_user_regs, ra);
+    OFFSET(CPU_USER_REGS_SP, struct cpu_user_regs, sp);
+    OFFSET(CPU_USER_REGS_GP, struct cpu_user_regs, gp);
+    OFFSET(CPU_USER_REGS_TP, struct cpu_user_regs, tp);
+    OFFSET(CPU_USER_REGS_T0, struct cpu_user_regs, t0);
+    OFFSET(CPU_USER_REGS_T1, struct cpu_user_regs, t1);
+    OFFSET(CPU_USER_REGS_T2, struct cpu_user_regs, t2);
+    OFFSET(CPU_USER_REGS_S0, struct cpu_user_regs, s0);
+    OFFSET(CPU_USER_REGS_S1, struct cpu_user_regs, s1);
+    OFFSET(CPU_USER_REGS_A0, struct cpu_user_regs, a0);
+    OFFSET(CPU_USER_REGS_A1, struct cpu_user_regs, a1);
+    OFFSET(CPU_USER_REGS_A2, struct cpu_user_regs, a2);
+    OFFSET(CPU_USER_REGS_A3, struct cpu_user_regs, a3);
+    OFFSET(CPU_USER_REGS_A4, struct cpu_user_regs, a4);
+    OFFSET(CPU_USER_REGS_A5, struct cpu_user_regs, a5);
+    OFFSET(CPU_USER_REGS_A6, struct cpu_user_regs, a6);
+    OFFSET(CPU_USER_REGS_A7, struct cpu_user_regs, a7);
+    OFFSET(CPU_USER_REGS_S2, struct cpu_user_regs, s2);
+    OFFSET(CPU_USER_REGS_S3, struct cpu_user_regs, s3);
+    OFFSET(CPU_USER_REGS_S4, struct cpu_user_regs, s4);
+    OFFSET(CPU_USER_REGS_S5, struct cpu_user_regs, s5);
+    OFFSET(CPU_USER_REGS_S6, struct cpu_user_regs, s6);
+    OFFSET(CPU_USER_REGS_S7, struct cpu_user_regs, s7);
+    OFFSET(CPU_USER_REGS_S8, struct cpu_user_regs, s8);
+    OFFSET(CPU_USER_REGS_S9, struct cpu_user_regs, s9);
+    OFFSET(CPU_USER_REGS_S10, struct cpu_user_regs, s10);
+    OFFSET(CPU_USER_REGS_S11, struct cpu_user_regs, s11);
+    OFFSET(CPU_USER_REGS_T3, struct cpu_user_regs, t3);
+    OFFSET(CPU_USER_REGS_T4, struct cpu_user_regs, t4);
+    OFFSET(CPU_USER_REGS_T5, struct cpu_user_regs, t5);
+    OFFSET(CPU_USER_REGS_T6, struct cpu_user_regs, t6);
+    OFFSET(CPU_USER_REGS_SEPC, struct cpu_user_regs, sepc);
+    OFFSET(CPU_USER_REGS_SSTATUS, struct cpu_user_regs, sstatus);
+    OFFSET(CPU_USER_REGS_PREGS, struct cpu_user_regs, pregs);
+}