2021-01-30 21:08:36 +08:00
|
|
|
/* SPDX-License-Identifier: GPL-2.0-or-later */
|
|
|
|
#ifndef _ASM_POWERPC_INTERRUPT_H
|
|
|
|
#define _ASM_POWERPC_INTERRUPT_H
|
|
|
|
|
|
|
|
#include <linux/context_tracking.h>
|
2021-01-30 21:08:39 +08:00
|
|
|
#include <linux/hardirq.h>
|
2021-01-30 21:08:48 +08:00
|
|
|
#include <asm/cputime.h>
|
2021-01-30 21:08:36 +08:00
|
|
|
#include <asm/ftrace.h>
|
2021-02-11 14:36:36 +08:00
|
|
|
#include <asm/kprobes.h>
|
2021-01-30 21:08:51 +08:00
|
|
|
#include <asm/runlatch.h>
|
2021-01-30 21:08:36 +08:00
|
|
|
|
2021-04-06 10:55:08 +08:00
|
|
|
static inline void nap_adjust_return(struct pt_regs *regs)
|
|
|
|
{
|
|
|
|
#ifdef CONFIG_PPC_970_NAP
|
|
|
|
if (unlikely(test_thread_local_flags(_TLF_NAPPING))) {
|
|
|
|
/* Can avoid a test-and-clear because NMIs do not call this */
|
|
|
|
clear_thread_local_flags(_TLF_NAPPING);
|
|
|
|
regs->nip = (unsigned long)power4_idle_nap_return;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2021-01-30 21:08:37 +08:00
|
|
|
struct interrupt_state {
|
2021-01-30 21:08:45 +08:00
|
|
|
#ifdef CONFIG_PPC_BOOK3E_64
|
2021-01-30 21:08:42 +08:00
|
|
|
enum ctx_state ctx_state;
|
|
|
|
#endif
|
2021-01-30 21:08:37 +08:00
|
|
|
};
|
|
|
|
|
2021-02-10 03:29:28 +08:00
|
|
|
static inline void booke_restore_dbcr0(void)
|
|
|
|
{
|
|
|
|
#ifdef CONFIG_PPC_ADV_DEBUG_REGS
|
|
|
|
unsigned long dbcr0 = current->thread.debug.dbcr0;
|
|
|
|
|
|
|
|
if (IS_ENABLED(CONFIG_PPC32) && unlikely(dbcr0 & DBCR0_IDM)) {
|
|
|
|
mtspr(SPRN_DBSR, -1);
|
|
|
|
mtspr(SPRN_DBCR0, global_dbcr0[smp_processor_id()]);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2021-01-30 21:08:37 +08:00
|
|
|
static inline void interrupt_enter_prepare(struct pt_regs *regs, struct interrupt_state *state)
|
|
|
|
{
|
2021-03-12 20:50:17 +08:00
|
|
|
#ifdef CONFIG_PPC32
|
|
|
|
if (!arch_irq_disabled_regs(regs))
|
|
|
|
trace_hardirqs_off();
|
2021-03-12 20:50:18 +08:00
|
|
|
|
2021-03-12 20:50:47 +08:00
|
|
|
if (user_mode(regs)) {
|
|
|
|
kuep_lock();
|
2021-03-12 20:50:18 +08:00
|
|
|
account_cpu_user_entry();
|
2021-03-12 20:50:51 +08:00
|
|
|
} else {
|
|
|
|
kuap_save_and_lock(regs);
|
2021-03-12 20:50:47 +08:00
|
|
|
}
|
2021-03-12 20:50:17 +08:00
|
|
|
#endif
|
2021-01-30 21:08:46 +08:00
|
|
|
/*
|
|
|
|
* Book3E reconciles irq soft mask in asm
|
|
|
|
*/
|
2021-01-30 21:08:45 +08:00
|
|
|
#ifdef CONFIG_PPC_BOOK3S_64
|
2021-01-30 21:08:46 +08:00
|
|
|
if (irq_soft_mask_set_return(IRQS_ALL_DISABLED) == IRQS_ENABLED)
|
|
|
|
trace_hardirqs_off();
|
|
|
|
local_paca->irq_happened |= PACA_IRQ_HARD_DIS;
|
|
|
|
|
2021-01-30 21:08:45 +08:00
|
|
|
if (user_mode(regs)) {
|
|
|
|
CT_WARN_ON(ct_state() != CONTEXT_USER);
|
|
|
|
user_exit_irqoff();
|
2021-01-30 21:08:48 +08:00
|
|
|
|
|
|
|
account_cpu_user_entry();
|
|
|
|
account_stolen_time();
|
2021-01-30 21:08:45 +08:00
|
|
|
} else {
|
|
|
|
/*
|
|
|
|
* CT_WARN_ON comes here via program_check_exception,
|
|
|
|
* so avoid recursion.
|
|
|
|
*/
|
|
|
|
if (TRAP(regs) != 0x700)
|
|
|
|
CT_WARN_ON(ct_state() != CONTEXT_KERNEL);
|
|
|
|
}
|
|
|
|
#endif
|
2021-01-30 21:08:46 +08:00
|
|
|
|
|
|
|
#ifdef CONFIG_PPC_BOOK3E_64
|
|
|
|
state->ctx_state = exception_enter();
|
2021-01-30 21:08:48 +08:00
|
|
|
if (user_mode(regs))
|
|
|
|
account_cpu_user_entry();
|
2021-01-30 21:08:46 +08:00
|
|
|
#endif
|
2021-03-12 20:50:19 +08:00
|
|
|
|
|
|
|
booke_restore_dbcr0();
|
2021-01-30 21:08:37 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Care should be taken to note that interrupt_exit_prepare and
|
|
|
|
* interrupt_async_exit_prepare do not necessarily return immediately to
|
|
|
|
* regs context (e.g., if regs is usermode, we don't necessarily return to
|
|
|
|
* user mode). Other interrupts might be taken between here and return,
|
|
|
|
* context switch / preemption may occur in the exit path after this, or a
|
|
|
|
* signal may be delivered, etc.
|
|
|
|
*
|
|
|
|
* The real interrupt exit code is platform specific, e.g.,
|
|
|
|
* interrupt_exit_user_prepare / interrupt_exit_kernel_prepare for 64s.
|
|
|
|
*
|
|
|
|
* However interrupt_nmi_exit_prepare does return directly to regs, because
|
|
|
|
* NMIs do not do "exit work" or replay soft-masked interrupts.
|
|
|
|
*/
|
|
|
|
static inline void interrupt_exit_prepare(struct pt_regs *regs, struct interrupt_state *state)
|
|
|
|
{
|
2021-01-30 21:08:45 +08:00
|
|
|
#ifdef CONFIG_PPC_BOOK3E_64
|
2021-01-30 21:08:42 +08:00
|
|
|
exception_exit(state->ctx_state);
|
|
|
|
#endif
|
2021-01-30 21:08:45 +08:00
|
|
|
|
2021-03-12 20:50:47 +08:00
|
|
|
if (user_mode(regs))
|
|
|
|
kuep_unlock();
|
2021-01-30 21:08:45 +08:00
|
|
|
/*
|
|
|
|
* Book3S exits to user via interrupt_exit_user_prepare(), which does
|
|
|
|
* context tracking, which is a cleaner way to handle PREEMPT=y
|
|
|
|
* and avoid context entry/exit in e.g., preempt_schedule_irq()),
|
|
|
|
* which is likely to be where the core code wants to end up.
|
|
|
|
*
|
|
|
|
* The above comment explains why we can't do the
|
|
|
|
*
|
|
|
|
* if (user_mode(regs))
|
|
|
|
* user_exit_irqoff();
|
|
|
|
*
|
|
|
|
* sequence here.
|
|
|
|
*/
|
2021-01-30 21:08:37 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static inline void interrupt_async_enter_prepare(struct pt_regs *regs, struct interrupt_state *state)
|
|
|
|
{
|
2021-01-30 21:08:51 +08:00
|
|
|
#ifdef CONFIG_PPC_BOOK3S_64
|
|
|
|
if (cpu_has_feature(CPU_FTR_CTRL) &&
|
|
|
|
!test_thread_local_flags(_TLF_RUNLATCH))
|
|
|
|
__ppc64_runlatch_on();
|
|
|
|
#endif
|
|
|
|
|
2021-01-30 21:08:43 +08:00
|
|
|
interrupt_enter_prepare(regs, state);
|
2021-01-30 21:08:44 +08:00
|
|
|
irq_enter();
|
2021-01-30 21:08:37 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static inline void interrupt_async_exit_prepare(struct pt_regs *regs, struct interrupt_state *state)
|
|
|
|
{
|
2021-04-06 10:55:08 +08:00
|
|
|
/*
|
|
|
|
* Adjust at exit so the main handler sees the true NIA. This must
|
|
|
|
* come before irq_exit() because irq_exit can enable interrupts, and
|
|
|
|
* if another interrupt is taken before nap_adjust_return has run
|
|
|
|
* here, then that interrupt would return directly to idle nap return.
|
|
|
|
*/
|
|
|
|
nap_adjust_return(regs);
|
|
|
|
|
2021-01-30 21:08:44 +08:00
|
|
|
irq_exit();
|
2021-01-30 21:08:43 +08:00
|
|
|
interrupt_exit_prepare(regs, state);
|
2021-01-30 21:08:37 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
struct interrupt_nmi_state {
|
2021-01-30 21:08:49 +08:00
|
|
|
#ifdef CONFIG_PPC64
|
2021-01-30 21:08:50 +08:00
|
|
|
u8 irq_soft_mask;
|
|
|
|
u8 irq_happened;
|
2021-01-30 21:08:49 +08:00
|
|
|
u8 ftrace_enabled;
|
|
|
|
#endif
|
2021-01-30 21:08:37 +08:00
|
|
|
};
|
|
|
|
|
2021-03-16 18:41:59 +08:00
|
|
|
static inline bool nmi_disables_ftrace(struct pt_regs *regs)
|
|
|
|
{
|
|
|
|
/* Allow DEC and PMI to be traced when they are soft-NMI */
|
|
|
|
if (IS_ENABLED(CONFIG_PPC_BOOK3S_64)) {
|
|
|
|
if (TRAP(regs) == 0x900)
|
|
|
|
return false;
|
|
|
|
if (TRAP(regs) == 0xf00)
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (IS_ENABLED(CONFIG_PPC_BOOK3E)) {
|
|
|
|
if (TRAP(regs) == 0x260)
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2021-01-30 21:08:37 +08:00
|
|
|
static inline void interrupt_nmi_enter_prepare(struct pt_regs *regs, struct interrupt_nmi_state *state)
|
|
|
|
{
|
2021-01-30 21:08:49 +08:00
|
|
|
#ifdef CONFIG_PPC64
|
2021-01-30 21:08:50 +08:00
|
|
|
state->irq_soft_mask = local_paca->irq_soft_mask;
|
|
|
|
state->irq_happened = local_paca->irq_happened;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Set IRQS_ALL_DISABLED unconditionally so irqs_disabled() does
|
|
|
|
* the right thing, and set IRQ_HARD_DIS. We do not want to reconcile
|
|
|
|
* because that goes through irq tracing which we don't want in NMI.
|
|
|
|
*/
|
|
|
|
local_paca->irq_soft_mask = IRQS_ALL_DISABLED;
|
|
|
|
local_paca->irq_happened |= PACA_IRQ_HARD_DIS;
|
|
|
|
|
|
|
|
/* Don't do any per-CPU operations until interrupt state is fixed */
|
2021-03-16 18:41:59 +08:00
|
|
|
|
|
|
|
if (nmi_disables_ftrace(regs)) {
|
2021-01-30 21:08:49 +08:00
|
|
|
state->ftrace_enabled = this_cpu_get_ftrace_enabled();
|
|
|
|
this_cpu_set_ftrace_enabled(0);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Do not use nmi_enter() for pseries hash guest taking a real-mode
|
|
|
|
* NMI because not everything it touches is within the RMA limit.
|
|
|
|
*/
|
|
|
|
if (!IS_ENABLED(CONFIG_PPC_BOOK3S_64) ||
|
|
|
|
!firmware_has_feature(FW_FEATURE_LPAR) ||
|
|
|
|
radix_enabled() || (mfmsr() & MSR_DR))
|
|
|
|
nmi_enter();
|
2021-01-30 21:08:37 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static inline void interrupt_nmi_exit_prepare(struct pt_regs *regs, struct interrupt_nmi_state *state)
|
|
|
|
{
|
2021-01-30 21:08:49 +08:00
|
|
|
if (!IS_ENABLED(CONFIG_PPC_BOOK3S_64) ||
|
|
|
|
!firmware_has_feature(FW_FEATURE_LPAR) ||
|
|
|
|
radix_enabled() || (mfmsr() & MSR_DR))
|
|
|
|
nmi_exit();
|
|
|
|
|
2021-04-06 10:55:08 +08:00
|
|
|
/*
|
|
|
|
* nmi does not call nap_adjust_return because nmi should not create
|
|
|
|
* new work to do (must use irq_work for that).
|
|
|
|
*/
|
|
|
|
|
2021-01-30 21:08:49 +08:00
|
|
|
#ifdef CONFIG_PPC64
|
2021-03-16 18:41:59 +08:00
|
|
|
if (nmi_disables_ftrace(regs))
|
2021-01-30 21:08:49 +08:00
|
|
|
this_cpu_set_ftrace_enabled(state->ftrace_enabled);
|
2021-01-30 21:08:50 +08:00
|
|
|
|
|
|
|
/* Check we didn't change the pending interrupt mask. */
|
|
|
|
WARN_ON_ONCE((state->irq_happened | PACA_IRQ_HARD_DIS) != local_paca->irq_happened);
|
|
|
|
local_paca->irq_happened = state->irq_happened;
|
|
|
|
local_paca->irq_soft_mask = state->irq_soft_mask;
|
|
|
|
#endif
|
2021-01-30 21:08:37 +08:00
|
|
|
}
|
|
|
|
|
2021-02-11 14:36:36 +08:00
|
|
|
/*
|
|
|
|
* Don't use noinstr here like x86, but rather add NOKPROBE_SYMBOL to each
|
|
|
|
* function definition. The reason for this is the noinstr section is placed
|
|
|
|
* after the main text section, i.e., very far away from the interrupt entry
|
|
|
|
* asm. That creates problems with fitting linker stubs when building large
|
|
|
|
* kernels.
|
|
|
|
*/
|
|
|
|
#define interrupt_handler __visible noinline notrace __no_kcsan __no_sanitize_address
|
|
|
|
|
2021-01-30 21:08:36 +08:00
|
|
|
/**
|
|
|
|
* DECLARE_INTERRUPT_HANDLER_RAW - Declare raw interrupt handler function
|
|
|
|
* @func: Function name of the entry point
|
|
|
|
* @returns: Returns a value back to asm caller
|
|
|
|
*/
|
|
|
|
#define DECLARE_INTERRUPT_HANDLER_RAW(func) \
|
|
|
|
__visible long func(struct pt_regs *regs)
|
|
|
|
|
|
|
|
/**
|
|
|
|
* DEFINE_INTERRUPT_HANDLER_RAW - Define raw interrupt handler function
|
|
|
|
* @func: Function name of the entry point
|
|
|
|
* @returns: Returns a value back to asm caller
|
|
|
|
*
|
|
|
|
* @func is called from ASM entry code.
|
|
|
|
*
|
|
|
|
* This is a plain function which does no tracing, reconciling, etc.
|
|
|
|
* The macro is written so it acts as function definition. Append the
|
|
|
|
* body with a pair of curly brackets.
|
|
|
|
*
|
|
|
|
* raw interrupt handlers must not enable or disable interrupts, or
|
|
|
|
* schedule, tracing and instrumentation (ftrace, lockdep, etc) would
|
|
|
|
* not be advisable either, although may be possible in a pinch, the
|
|
|
|
* trace will look odd at least.
|
|
|
|
*
|
|
|
|
* A raw handler may call one of the other interrupt handler functions
|
|
|
|
* to be converted into that interrupt context without these restrictions.
|
|
|
|
*
|
|
|
|
* On PPC64, _RAW handlers may return with fast_interrupt_return.
|
|
|
|
*
|
|
|
|
* Specific handlers may have additional restrictions.
|
|
|
|
*/
|
|
|
|
#define DEFINE_INTERRUPT_HANDLER_RAW(func) \
|
|
|
|
static __always_inline long ____##func(struct pt_regs *regs); \
|
|
|
|
\
|
2021-02-11 14:36:36 +08:00
|
|
|
interrupt_handler long func(struct pt_regs *regs) \
|
2021-01-30 21:08:36 +08:00
|
|
|
{ \
|
|
|
|
long ret; \
|
|
|
|
\
|
|
|
|
ret = ____##func (regs); \
|
|
|
|
\
|
|
|
|
return ret; \
|
|
|
|
} \
|
2021-02-11 14:36:36 +08:00
|
|
|
NOKPROBE_SYMBOL(func); \
|
2021-01-30 21:08:36 +08:00
|
|
|
\
|
|
|
|
static __always_inline long ____##func(struct pt_regs *regs)
|
|
|
|
|
|
|
|
/**
|
|
|
|
* DECLARE_INTERRUPT_HANDLER - Declare synchronous interrupt handler function
|
|
|
|
* @func: Function name of the entry point
|
|
|
|
*/
|
|
|
|
#define DECLARE_INTERRUPT_HANDLER(func) \
|
|
|
|
__visible void func(struct pt_regs *regs)
|
|
|
|
|
|
|
|
/**
|
|
|
|
* DEFINE_INTERRUPT_HANDLER - Define synchronous interrupt handler function
|
|
|
|
* @func: Function name of the entry point
|
|
|
|
*
|
|
|
|
* @func is called from ASM entry code.
|
|
|
|
*
|
|
|
|
* The macro is written so it acts as function definition. Append the
|
|
|
|
* body with a pair of curly brackets.
|
|
|
|
*/
|
|
|
|
#define DEFINE_INTERRUPT_HANDLER(func) \
|
|
|
|
static __always_inline void ____##func(struct pt_regs *regs); \
|
|
|
|
\
|
2021-02-11 14:36:36 +08:00
|
|
|
interrupt_handler void func(struct pt_regs *regs) \
|
2021-01-30 21:08:36 +08:00
|
|
|
{ \
|
2021-01-30 21:08:37 +08:00
|
|
|
struct interrupt_state state; \
|
|
|
|
\
|
|
|
|
interrupt_enter_prepare(regs, &state); \
|
|
|
|
\
|
2021-01-30 21:08:36 +08:00
|
|
|
____##func (regs); \
|
2021-01-30 21:08:37 +08:00
|
|
|
\
|
|
|
|
interrupt_exit_prepare(regs, &state); \
|
2021-01-30 21:08:36 +08:00
|
|
|
} \
|
2021-02-11 14:36:36 +08:00
|
|
|
NOKPROBE_SYMBOL(func); \
|
2021-01-30 21:08:36 +08:00
|
|
|
\
|
|
|
|
static __always_inline void ____##func(struct pt_regs *regs)
|
|
|
|
|
|
|
|
/**
|
|
|
|
* DECLARE_INTERRUPT_HANDLER_RET - Declare synchronous interrupt handler function
|
|
|
|
* @func: Function name of the entry point
|
|
|
|
* @returns: Returns a value back to asm caller
|
|
|
|
*/
|
|
|
|
#define DECLARE_INTERRUPT_HANDLER_RET(func) \
|
|
|
|
__visible long func(struct pt_regs *regs)
|
|
|
|
|
|
|
|
/**
|
|
|
|
* DEFINE_INTERRUPT_HANDLER_RET - Define synchronous interrupt handler function
|
|
|
|
* @func: Function name of the entry point
|
|
|
|
* @returns: Returns a value back to asm caller
|
|
|
|
*
|
|
|
|
* @func is called from ASM entry code.
|
|
|
|
*
|
|
|
|
* The macro is written so it acts as function definition. Append the
|
|
|
|
* body with a pair of curly brackets.
|
|
|
|
*/
|
|
|
|
#define DEFINE_INTERRUPT_HANDLER_RET(func) \
|
|
|
|
static __always_inline long ____##func(struct pt_regs *regs); \
|
|
|
|
\
|
2021-02-11 14:36:36 +08:00
|
|
|
interrupt_handler long func(struct pt_regs *regs) \
|
2021-01-30 21:08:36 +08:00
|
|
|
{ \
|
2021-01-30 21:08:37 +08:00
|
|
|
struct interrupt_state state; \
|
2021-01-30 21:08:36 +08:00
|
|
|
long ret; \
|
|
|
|
\
|
2021-01-30 21:08:37 +08:00
|
|
|
interrupt_enter_prepare(regs, &state); \
|
|
|
|
\
|
2021-01-30 21:08:36 +08:00
|
|
|
ret = ____##func (regs); \
|
|
|
|
\
|
2021-01-30 21:08:37 +08:00
|
|
|
interrupt_exit_prepare(regs, &state); \
|
|
|
|
\
|
2021-01-30 21:08:36 +08:00
|
|
|
return ret; \
|
|
|
|
} \
|
2021-02-11 14:36:36 +08:00
|
|
|
NOKPROBE_SYMBOL(func); \
|
2021-01-30 21:08:36 +08:00
|
|
|
\
|
|
|
|
static __always_inline long ____##func(struct pt_regs *regs)
|
|
|
|
|
|
|
|
/**
|
|
|
|
* DECLARE_INTERRUPT_HANDLER_ASYNC - Declare asynchronous interrupt handler function
|
|
|
|
* @func: Function name of the entry point
|
|
|
|
*/
|
|
|
|
#define DECLARE_INTERRUPT_HANDLER_ASYNC(func) \
|
|
|
|
__visible void func(struct pt_regs *regs)
|
|
|
|
|
|
|
|
/**
|
|
|
|
* DEFINE_INTERRUPT_HANDLER_ASYNC - Define asynchronous interrupt handler function
|
|
|
|
* @func: Function name of the entry point
|
|
|
|
*
|
|
|
|
* @func is called from ASM entry code.
|
|
|
|
*
|
|
|
|
* The macro is written so it acts as function definition. Append the
|
|
|
|
* body with a pair of curly brackets.
|
|
|
|
*/
|
|
|
|
#define DEFINE_INTERRUPT_HANDLER_ASYNC(func) \
|
|
|
|
static __always_inline void ____##func(struct pt_regs *regs); \
|
|
|
|
\
|
2021-02-11 14:36:36 +08:00
|
|
|
interrupt_handler void func(struct pt_regs *regs) \
|
2021-01-30 21:08:36 +08:00
|
|
|
{ \
|
2021-01-30 21:08:37 +08:00
|
|
|
struct interrupt_state state; \
|
|
|
|
\
|
|
|
|
interrupt_async_enter_prepare(regs, &state); \
|
|
|
|
\
|
2021-01-30 21:08:36 +08:00
|
|
|
____##func (regs); \
|
2021-01-30 21:08:37 +08:00
|
|
|
\
|
|
|
|
interrupt_async_exit_prepare(regs, &state); \
|
2021-01-30 21:08:36 +08:00
|
|
|
} \
|
2021-02-11 14:36:36 +08:00
|
|
|
NOKPROBE_SYMBOL(func); \
|
2021-01-30 21:08:36 +08:00
|
|
|
\
|
|
|
|
static __always_inline void ____##func(struct pt_regs *regs)
|
|
|
|
|
|
|
|
/**
|
|
|
|
* DECLARE_INTERRUPT_HANDLER_NMI - Declare NMI interrupt handler function
|
|
|
|
* @func: Function name of the entry point
|
|
|
|
* @returns: Returns a value back to asm caller
|
|
|
|
*/
|
|
|
|
#define DECLARE_INTERRUPT_HANDLER_NMI(func) \
|
|
|
|
__visible long func(struct pt_regs *regs)
|
|
|
|
|
|
|
|
/**
|
|
|
|
* DEFINE_INTERRUPT_HANDLER_NMI - Define NMI interrupt handler function
|
|
|
|
* @func: Function name of the entry point
|
|
|
|
* @returns: Returns a value back to asm caller
|
|
|
|
*
|
|
|
|
* @func is called from ASM entry code.
|
|
|
|
*
|
|
|
|
* The macro is written so it acts as function definition. Append the
|
|
|
|
* body with a pair of curly brackets.
|
|
|
|
*/
|
|
|
|
#define DEFINE_INTERRUPT_HANDLER_NMI(func) \
|
|
|
|
static __always_inline long ____##func(struct pt_regs *regs); \
|
|
|
|
\
|
2021-02-11 14:36:36 +08:00
|
|
|
interrupt_handler long func(struct pt_regs *regs) \
|
2021-01-30 21:08:36 +08:00
|
|
|
{ \
|
2021-01-30 21:08:37 +08:00
|
|
|
struct interrupt_nmi_state state; \
|
2021-01-30 21:08:36 +08:00
|
|
|
long ret; \
|
|
|
|
\
|
2021-01-30 21:08:37 +08:00
|
|
|
interrupt_nmi_enter_prepare(regs, &state); \
|
|
|
|
\
|
2021-01-30 21:08:36 +08:00
|
|
|
ret = ____##func (regs); \
|
|
|
|
\
|
2021-01-30 21:08:37 +08:00
|
|
|
interrupt_nmi_exit_prepare(regs, &state); \
|
|
|
|
\
|
2021-01-30 21:08:36 +08:00
|
|
|
return ret; \
|
|
|
|
} \
|
2021-02-11 14:36:36 +08:00
|
|
|
NOKPROBE_SYMBOL(func); \
|
2021-01-30 21:08:36 +08:00
|
|
|
\
|
|
|
|
static __always_inline long ____##func(struct pt_regs *regs)
|
|
|
|
|
2021-01-30 21:08:38 +08:00
|
|
|
|
|
|
|
/* Interrupt handlers */
|
|
|
|
/* kernel/traps.c */
|
|
|
|
DECLARE_INTERRUPT_HANDLER_NMI(system_reset_exception);
|
|
|
|
#ifdef CONFIG_PPC_BOOK3S_64
|
|
|
|
DECLARE_INTERRUPT_HANDLER_ASYNC(machine_check_exception);
|
|
|
|
#else
|
|
|
|
DECLARE_INTERRUPT_HANDLER_NMI(machine_check_exception);
|
|
|
|
#endif
|
|
|
|
DECLARE_INTERRUPT_HANDLER(SMIException);
|
|
|
|
DECLARE_INTERRUPT_HANDLER(handle_hmi_exception);
|
|
|
|
DECLARE_INTERRUPT_HANDLER(unknown_exception);
|
|
|
|
DECLARE_INTERRUPT_HANDLER_ASYNC(unknown_async_exception);
|
2021-03-16 18:41:59 +08:00
|
|
|
DECLARE_INTERRUPT_HANDLER_NMI(unknown_nmi_exception);
|
2021-01-30 21:08:38 +08:00
|
|
|
DECLARE_INTERRUPT_HANDLER(instruction_breakpoint_exception);
|
|
|
|
DECLARE_INTERRUPT_HANDLER(RunModeException);
|
|
|
|
DECLARE_INTERRUPT_HANDLER(single_step_exception);
|
|
|
|
DECLARE_INTERRUPT_HANDLER(program_check_exception);
|
|
|
|
DECLARE_INTERRUPT_HANDLER(emulation_assist_interrupt);
|
|
|
|
DECLARE_INTERRUPT_HANDLER(alignment_exception);
|
|
|
|
DECLARE_INTERRUPT_HANDLER(StackOverflow);
|
|
|
|
DECLARE_INTERRUPT_HANDLER(stack_overflow_exception);
|
|
|
|
DECLARE_INTERRUPT_HANDLER(kernel_fp_unavailable_exception);
|
|
|
|
DECLARE_INTERRUPT_HANDLER(altivec_unavailable_exception);
|
|
|
|
DECLARE_INTERRUPT_HANDLER(vsx_unavailable_exception);
|
|
|
|
DECLARE_INTERRUPT_HANDLER(facility_unavailable_exception);
|
|
|
|
DECLARE_INTERRUPT_HANDLER(fp_unavailable_tm);
|
|
|
|
DECLARE_INTERRUPT_HANDLER(altivec_unavailable_tm);
|
|
|
|
DECLARE_INTERRUPT_HANDLER(vsx_unavailable_tm);
|
|
|
|
DECLARE_INTERRUPT_HANDLER_NMI(performance_monitor_exception_nmi);
|
|
|
|
DECLARE_INTERRUPT_HANDLER_ASYNC(performance_monitor_exception_async);
|
|
|
|
DECLARE_INTERRUPT_HANDLER_RAW(performance_monitor_exception);
|
|
|
|
DECLARE_INTERRUPT_HANDLER(DebugException);
|
|
|
|
DECLARE_INTERRUPT_HANDLER(altivec_assist_exception);
|
|
|
|
DECLARE_INTERRUPT_HANDLER(CacheLockingException);
|
|
|
|
DECLARE_INTERRUPT_HANDLER(SPEFloatingPointException);
|
|
|
|
DECLARE_INTERRUPT_HANDLER(SPEFloatingPointRoundException);
|
2021-03-16 18:41:59 +08:00
|
|
|
DECLARE_INTERRUPT_HANDLER_NMI(WatchdogException);
|
2021-01-30 21:08:38 +08:00
|
|
|
DECLARE_INTERRUPT_HANDLER(kernel_bad_stack);
|
|
|
|
|
|
|
|
/* slb.c */
|
|
|
|
DECLARE_INTERRUPT_HANDLER_RAW(do_slb_fault);
|
|
|
|
DECLARE_INTERRUPT_HANDLER(do_bad_slb_fault);
|
|
|
|
|
|
|
|
/* hash_utils.c */
|
|
|
|
DECLARE_INTERRUPT_HANDLER_RAW(do_hash_fault);
|
|
|
|
|
|
|
|
/* fault.c */
|
|
|
|
DECLARE_INTERRUPT_HANDLER_RET(do_page_fault);
|
|
|
|
DECLARE_INTERRUPT_HANDLER(do_bad_page_fault_segv);
|
|
|
|
|
|
|
|
/* process.c */
|
|
|
|
DECLARE_INTERRUPT_HANDLER(do_break);
|
|
|
|
|
|
|
|
/* time.c */
|
|
|
|
DECLARE_INTERRUPT_HANDLER_ASYNC(timer_interrupt);
|
|
|
|
|
|
|
|
/* mce.c */
|
|
|
|
DECLARE_INTERRUPT_HANDLER_NMI(machine_check_early);
|
|
|
|
DECLARE_INTERRUPT_HANDLER_NMI(hmi_exception_realmode);
|
|
|
|
|
|
|
|
DECLARE_INTERRUPT_HANDLER_ASYNC(TAUException);
|
|
|
|
|
powerpc/traps: Declare unrecoverable_exception() as __noreturn
unrecoverable_exception() is never expected to return, most callers
have an infiniteloop in case it returns.
Ensure it really never returns by terminating it with a BUG(), and
declare it __no_return.
It always GCC to really simplify functions calling it. In the exemple
below, it avoids the stack frame in the likely fast path and avoids
code duplication for the exit.
With this patch:
00000348 <interrupt_exit_kernel_prepare>:
348: 81 43 00 84 lwz r10,132(r3)
34c: 71 48 00 02 andi. r8,r10,2
350: 41 82 00 2c beq 37c <interrupt_exit_kernel_prepare+0x34>
354: 71 4a 40 00 andi. r10,r10,16384
358: 40 82 00 20 bne 378 <interrupt_exit_kernel_prepare+0x30>
35c: 80 62 00 70 lwz r3,112(r2)
360: 74 63 00 01 andis. r3,r3,1
364: 40 82 00 28 bne 38c <interrupt_exit_kernel_prepare+0x44>
368: 7d 40 00 a6 mfmsr r10
36c: 7c 11 13 a6 mtspr 81,r0
370: 7c 12 13 a6 mtspr 82,r0
374: 4e 80 00 20 blr
378: 48 00 00 00 b 378 <interrupt_exit_kernel_prepare+0x30>
37c: 94 21 ff f0 stwu r1,-16(r1)
380: 7c 08 02 a6 mflr r0
384: 90 01 00 14 stw r0,20(r1)
388: 48 00 00 01 bl 388 <interrupt_exit_kernel_prepare+0x40>
388: R_PPC_REL24 unrecoverable_exception
38c: 38 e2 00 70 addi r7,r2,112
390: 3d 00 00 01 lis r8,1
394: 7c c0 38 28 lwarx r6,0,r7
398: 7c c6 40 78 andc r6,r6,r8
39c: 7c c0 39 2d stwcx. r6,0,r7
3a0: 40 a2 ff f4 bne 394 <interrupt_exit_kernel_prepare+0x4c>
3a4: 38 60 00 01 li r3,1
3a8: 4b ff ff c0 b 368 <interrupt_exit_kernel_prepare+0x20>
Without this patch:
00000348 <interrupt_exit_kernel_prepare>:
348: 94 21 ff f0 stwu r1,-16(r1)
34c: 93 e1 00 0c stw r31,12(r1)
350: 7c 7f 1b 78 mr r31,r3
354: 81 23 00 84 lwz r9,132(r3)
358: 71 2a 00 02 andi. r10,r9,2
35c: 41 82 00 34 beq 390 <interrupt_exit_kernel_prepare+0x48>
360: 71 29 40 00 andi. r9,r9,16384
364: 40 82 00 28 bne 38c <interrupt_exit_kernel_prepare+0x44>
368: 80 62 00 70 lwz r3,112(r2)
36c: 74 63 00 01 andis. r3,r3,1
370: 40 82 00 3c bne 3ac <interrupt_exit_kernel_prepare+0x64>
374: 7d 20 00 a6 mfmsr r9
378: 7c 11 13 a6 mtspr 81,r0
37c: 7c 12 13 a6 mtspr 82,r0
380: 83 e1 00 0c lwz r31,12(r1)
384: 38 21 00 10 addi r1,r1,16
388: 4e 80 00 20 blr
38c: 48 00 00 00 b 38c <interrupt_exit_kernel_prepare+0x44>
390: 7c 08 02 a6 mflr r0
394: 90 01 00 14 stw r0,20(r1)
398: 48 00 00 01 bl 398 <interrupt_exit_kernel_prepare+0x50>
398: R_PPC_REL24 unrecoverable_exception
39c: 80 01 00 14 lwz r0,20(r1)
3a0: 81 3f 00 84 lwz r9,132(r31)
3a4: 7c 08 03 a6 mtlr r0
3a8: 4b ff ff b8 b 360 <interrupt_exit_kernel_prepare+0x18>
3ac: 39 02 00 70 addi r8,r2,112
3b0: 3d 40 00 01 lis r10,1
3b4: 7c e0 40 28 lwarx r7,0,r8
3b8: 7c e7 50 78 andc r7,r7,r10
3bc: 7c e0 41 2d stwcx. r7,0,r8
3c0: 40 a2 ff f4 bne 3b4 <interrupt_exit_kernel_prepare+0x6c>
3c4: 38 60 00 01 li r3,1
3c8: 4b ff ff ac b 374 <interrupt_exit_kernel_prepare+0x2c>
Signed-off-by: Christophe Leroy <christophe.leroy@csgroup.eu>
Reviewed-by: Nicholas Piggin <npiggin@gmail.com>
Signed-off-by: Michael Ellerman <mpe@ellerman.id.au>
Link: https://lore.kernel.org/r/1e883e9d93fdb256853d1434c8ad77c257349b2d.1615552866.git.christophe.leroy@csgroup.eu
2021-03-12 20:50:10 +08:00
|
|
|
void __noreturn unrecoverable_exception(struct pt_regs *regs);
|
2021-03-09 20:09:26 +08:00
|
|
|
|
2021-01-30 21:08:38 +08:00
|
|
|
void replay_system_reset(void);
|
|
|
|
void replay_soft_interrupts(void);
|
|
|
|
|
2021-01-30 21:08:39 +08:00
|
|
|
static inline void interrupt_cond_local_irq_enable(struct pt_regs *regs)
|
|
|
|
{
|
|
|
|
if (!arch_irq_disabled_regs(regs))
|
|
|
|
local_irq_enable();
|
|
|
|
}
|
|
|
|
|
2021-01-30 21:08:36 +08:00
|
|
|
#endif /* _ASM_POWERPC_INTERRUPT_H */
|