Commit 956895b9 authored by Andy Chiu's avatar Andy Chiu Committed by Palmer Dabbelt

riscv: vector: make Vector always available for softirq context

The goal of this patch is to provide full support of Vector in kernel
softirq context. So that some of the crypto alogrithms won't need scalar
fallbacks.

By disabling bottom halves in active kernel-mode Vector, softirq will
not be able to nest on top of any kernel-mode Vector. So, softirq
context is able to use Vector whenever it runs.

After this patch, Vector context cannot start with irqs disabled.
Otherwise local_bh_enable() may run in a wrong context.

Disabling bh is not enough for RT-kernel to prevent preeemption. So
we must disable preemption, which also implies disabling bh on RT.

Related-to: commit 696207d4 ("arm64/sve: Make kernel FPU protection RT friendly")
Related-to: commit 66c3ec5a ("arm64: neon: Forbid when irqs are disabled")
Signed-off-by: default avatarAndy Chiu <andy.chiu@sifive.com>
Reviewed-by: default avatarEric Biggers <ebiggers@google.com>
Tested-by: default avatarBjörn Töpel <bjorn@rivosinc.com>
Tested-by: default avatarLad Prabhakar <prabhakar.mahadev-lad.rj@bp.renesas.com>
Link: https://lore.kernel.org/r/20240115055929.4736-3-andy.chiu@sifive.comSigned-off-by: default avatarPalmer Dabbelt <palmer@rivosinc.com>
parent ecd2ada8
...@@ -78,7 +78,8 @@ struct pt_regs; ...@@ -78,7 +78,8 @@ struct pt_regs;
* following meaning: * following meaning:
* *
* - bit 0: indicates whether the in-kernel Vector context is active. The * - bit 0: indicates whether the in-kernel Vector context is active. The
* activation of this state disables the preemption. * activation of this state disables the preemption. On a non-RT kernel, it
* also disable bh.
*/ */
#define RISCV_KERNEL_MODE_V 0x1 #define RISCV_KERNEL_MODE_V 0x1
......
...@@ -28,8 +28,12 @@ static __must_check inline bool may_use_simd(void) ...@@ -28,8 +28,12 @@ static __must_check inline bool may_use_simd(void)
/* /*
* RISCV_KERNEL_MODE_V is only set while preemption is disabled, * RISCV_KERNEL_MODE_V is only set while preemption is disabled,
* and is clear whenever preemption is enabled. * and is clear whenever preemption is enabled.
*
* Kernel-mode Vector temporarily disables bh. So we must not return
* true on irq_disabled(). Otherwise we would fail the lockdep check
* calling local_bh_enable()
*/ */
return !in_hardirq() && !in_nmi() && !(riscv_v_flags() & RISCV_KERNEL_MODE_V); return !in_hardirq() && !in_nmi() && !irqs_disabled() && !(riscv_v_flags() & RISCV_KERNEL_MODE_V);
} }
#else /* ! CONFIG_RISCV_ISA_V */ #else /* ! CONFIG_RISCV_ISA_V */
......
...@@ -46,6 +46,13 @@ static inline void riscv_v_stop(u32 flags) ...@@ -46,6 +46,13 @@ static inline void riscv_v_stop(u32 flags)
*/ */
void get_cpu_vector_context(void) void get_cpu_vector_context(void)
{ {
/*
* disable softirqs so it is impossible for softirqs to nest
* get_cpu_vector_context() when kernel is actively using Vector.
*/
if (!IS_ENABLED(CONFIG_PREEMPT_RT))
local_bh_disable();
else
preempt_disable(); preempt_disable();
riscv_v_start(RISCV_KERNEL_MODE_V); riscv_v_start(RISCV_KERNEL_MODE_V);
...@@ -62,6 +69,9 @@ void put_cpu_vector_context(void) ...@@ -62,6 +69,9 @@ void put_cpu_vector_context(void)
{ {
riscv_v_stop(RISCV_KERNEL_MODE_V); riscv_v_stop(RISCV_KERNEL_MODE_V);
if (!IS_ENABLED(CONFIG_PREEMPT_RT))
local_bh_enable();
else
preempt_enable(); preempt_enable();
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment