powerpc/64: treat low kernel text as irqs soft-masked

Treat code below __end_soft_masked as soft-masked for the purpose
of alternate return. 64s already mostly does this for scv entry.

This will be used to exit from interrupts without disabling MSR[EE].

Signed-off-by: Nicholas Piggin <npiggin@gmail.com>
Signed-off-by: Michael Ellerman <mpe@ellerman.id.au>
Link: https://lore.kernel.org/r/20210617155116.2167984-12-npiggin@gmail.com
This commit is contained in:
Nicholas Piggin 2021-06-18 01:51:10 +10:00 committed by Michael Ellerman
parent 862fa56352
commit 9d1988ca87
4 changed files with 35 additions and 13 deletions

View File

@ -146,8 +146,13 @@ static inline void interrupt_enter_prepare(struct pt_regs *regs, struct interrup
* CT_WARN_ON comes here via program_check_exception, * CT_WARN_ON comes here via program_check_exception,
* so avoid recursion. * so avoid recursion.
*/ */
if (TRAP(regs) != INTERRUPT_PROGRAM) if (TRAP(regs) != INTERRUPT_PROGRAM) {
CT_WARN_ON(ct_state() != CONTEXT_KERNEL); CT_WARN_ON(ct_state() != CONTEXT_KERNEL);
BUG_ON(regs->nip < (unsigned long)__end_soft_masked);
}
/* Move this under a debugging check */
if (arch_irq_disabled_regs(regs))
BUG_ON(search_kernel_restart_table(regs->nip));
} }
#endif #endif
@ -238,8 +243,8 @@ static inline void interrupt_nmi_enter_prepare(struct pt_regs *regs, struct inte
local_paca->irq_happened |= PACA_IRQ_HARD_DIS; local_paca->irq_happened |= PACA_IRQ_HARD_DIS;
if (IS_ENABLED(CONFIG_PPC_BOOK3S_64) && !(regs->msr & MSR_PR) && if (IS_ENABLED(CONFIG_PPC_BOOK3S_64) && !(regs->msr & MSR_PR) &&
regs->nip < (unsigned long)__end_interrupts) { regs->nip < (unsigned long)__end_soft_masked) {
// Kernel code running below __end_interrupts is // Kernel code running below __end_soft_masked is
// implicitly soft-masked. // implicitly soft-masked.
regs->softe = IRQS_ALL_DISABLED; regs->softe = IRQS_ALL_DISABLED;
} }

View File

@ -342,7 +342,17 @@ ret_from_mc_except:
#define PROLOG_ADDITION_MASKABLE_GEN(n) \ #define PROLOG_ADDITION_MASKABLE_GEN(n) \
lbz r10,PACAIRQSOFTMASK(r13); /* are irqs soft-masked? */ \ lbz r10,PACAIRQSOFTMASK(r13); /* are irqs soft-masked? */ \
andi. r10,r10,IRQS_DISABLED; /* yes -> go out of line */ \ andi. r10,r10,IRQS_DISABLED; /* yes -> go out of line */ \
bne masked_interrupt_book3e_##n bne masked_interrupt_book3e_##n; \
/* Kernel code below __end_soft_masked is implicitly masked */ \
andi. r10,r11,MSR_PR; \
bne 1f; /* user -> not masked */ \
std r14,PACA_EXGEN+EX_R14(r13); \
LOAD_REG_IMMEDIATE_SYM(r14, r10, __end_soft_masked); \
mfspr r10,SPRN_SRR0; \
cmpld r10,r14; \
ld r14,PACA_EXGEN+EX_R14(r13); \
blt masked_interrupt_book3e_##n; \
1:
/* /*
* Additional regs must be re-loaded from paca before EXCEPTION_COMMON* is * Additional regs must be re-loaded from paca before EXCEPTION_COMMON* is

View File

@ -430,10 +430,13 @@ DEFINE_FIXED_SYMBOL(\name\()_common_real)
andi. r10,r12,MSR_PR andi. r10,r12,MSR_PR
bne 2f bne 2f
/* Kernel code running below __end_interrupts is implicitly /*
* soft-masked */ * Kernel code running below __end_soft_masked is implicitly
LOAD_HANDLER(r10, __end_interrupts) * soft-masked
*/
LOAD_HANDLER(r10, __end_soft_masked)
cmpld r11,r10 cmpld r11,r10
li r10,IMASK li r10,IMASK
blt- 1f blt- 1f
@ -751,17 +754,17 @@ __start_interrupts:
* scv instructions enter the kernel without changing EE, RI, ME, or HV. * scv instructions enter the kernel without changing EE, RI, ME, or HV.
* In particular, this means we can take a maskable interrupt at any point * In particular, this means we can take a maskable interrupt at any point
* in the scv handler, which is unlike any other interrupt. This is solved * in the scv handler, which is unlike any other interrupt. This is solved
* by treating the instruction addresses below __end_interrupts as being * by treating the instruction addresses below __end_soft_masked as being
* soft-masked. * soft-masked.
* *
* AIL-0 mode scv exceptions go to 0x17000-0x17fff, but we set AIL-3 and * AIL-0 mode scv exceptions go to 0x17000-0x17fff, but we set AIL-3 and
* ensure scv is never executed with relocation off, which means AIL-0 * ensure scv is never executed with relocation off, which means AIL-0
* should never happen. * should never happen.
* *
* Before leaving the below __end_interrupts text, at least of the following * Before leaving the following inside-__end_soft_masked text, at least of the
* must be true: * following must be true:
* - MSR[PR]=1 (i.e., return to userspace) * - MSR[PR]=1 (i.e., return to userspace)
* - MSR_EE|MSR_RI is set (no reentrant exceptions) * - MSR_EE|MSR_RI is clear (no reentrant exceptions)
* - Standard kernel environment is set up (stack, paca, etc) * - Standard kernel environment is set up (stack, paca, etc)
* *
* Call convention: * Call convention:
@ -2957,7 +2960,7 @@ MASKED_INTERRUPT hsrr=1
USE_FIXED_SECTION(virt_trampolines) USE_FIXED_SECTION(virt_trampolines)
/* /*
* All code below __end_interrupts is treated as soft-masked. If * All code below __end_soft_masked is treated as soft-masked. If
* any code runs here with MSR[EE]=1, it must then cope with pending * any code runs here with MSR[EE]=1, it must then cope with pending
* soft interrupt being raised (i.e., by ensuring it is replayed). * soft interrupt being raised (i.e., by ensuring it is replayed).
* *

View File

@ -632,4 +632,8 @@ ALT_FTR_SECTION_END_IFCLR(CPU_FTR_STCX_CHECKS_ADDRESS)
interrupt_return_macro srr interrupt_return_macro srr
#ifdef CONFIG_PPC_BOOK3S #ifdef CONFIG_PPC_BOOK3S
interrupt_return_macro hsrr interrupt_return_macro hsrr
#endif #endif /* CONFIG_PPC_BOOK3S */
.globl __end_soft_masked
__end_soft_masked:
DEFINE_FIXED_SYMBOL(__end_soft_masked)