2019-06-03 13:44:50 +08:00
|
|
|
/* SPDX-License-Identifier: GPL-2.0-only */
|
2012-03-05 19:49:34 +08:00
|
|
|
/*
|
|
|
|
* Based on arch/arm/include/asm/exception.h
|
|
|
|
*
|
|
|
|
* Copyright (C) 2012 ARM Ltd.
|
|
|
|
*/
|
|
|
|
#ifndef __ASM_EXCEPTION_H
|
|
|
|
#define __ASM_EXCEPTION_H
|
|
|
|
|
2018-01-16 03:38:59 +08:00
|
|
|
#include <asm/esr.h>
|
2019-10-26 00:42:10 +08:00
|
|
|
#include <asm/kprobes.h>
|
2019-10-26 00:42:11 +08:00
|
|
|
#include <asm/ptrace.h>
|
2018-01-16 03:38:59 +08:00
|
|
|
|
2016-03-26 05:22:05 +08:00
|
|
|
#include <linux/interrupt.h>
|
2015-08-12 22:16:19 +08:00
|
|
|
|
|
|
|
#ifdef CONFIG_FUNCTION_GRAPH_TRACER
|
|
|
|
#define __exception_irq_entry __irq_entry
|
|
|
|
#else
|
2019-10-26 00:42:10 +08:00
|
|
|
#define __exception_irq_entry __kprobes
|
2015-08-12 22:16:19 +08:00
|
|
|
#endif
|
2012-03-05 19:49:34 +08:00
|
|
|
|
2018-01-16 03:38:59 +08:00
|
|
|
static inline u32 disr_to_esr(u64 disr)
|
|
|
|
{
|
|
|
|
unsigned int esr = ESR_ELx_EC_SERROR << ESR_ELx_EC_SHIFT;
|
|
|
|
|
|
|
|
if ((disr & DISR_EL1_IDS) == 0)
|
|
|
|
esr |= (disr & DISR_EL1_ESR_MASK);
|
|
|
|
else
|
|
|
|
esr |= (disr & ESR_ELx_ISS_MASK);
|
|
|
|
|
|
|
|
return esr;
|
|
|
|
}
|
|
|
|
|
2021-06-07 17:46:20 +08:00
|
|
|
asmlinkage void handle_bad_stack(struct pt_regs *regs);
|
|
|
|
|
arm64: entry: handle all vectors with C
We have 16 architectural exception vectors, and depending on kernel
configuration we handle 8 or 12 of these with C code, with the remaining
8 or 4 of these handled as special cases in the entry assembly.
It would be nicer if the entry assembly were uniform for all exceptions,
and we deferred any specific handling of the exceptions to C code. This
way the entry assembly can be more easily templated without ifdeffery or
special cases, and it's easier to modify the handling of these cases in
future (e.g. to dump additional registers other context).
This patch reworks the entry code so that we always have a C handler for
every architectural exception vector, with the entry assembly being
completely uniform. We now have to handle exceptions from EL1t and EL1h,
and also have to handle exceptions from AArch32 even when the kernel is
built without CONFIG_COMPAT. To make this clear and to simplify
templating, we rename the top-level exception handlers with a consistent
naming scheme:
asm: <el+sp>_<regsize>_<type>
c: <el+sp>_<regsize>_<type>_handler
.. where:
<el+sp> is `el1t`, `el1h`, or `el0t`
<regsize> is `64` or `32`
<type> is `sync`, `irq`, `fiq`, or `error`
... e.g.
asm: el1h_64_sync
c: el1h_64_sync_handler
... with lower-level handlers simply using "el1" and "compat" as today.
For unexpected exceptions, this information is passed to
__panic_unhandled(), so it can report the specific vector an unexpected
exception was taken from, e.g.
| Unhandled 64-bit el1t sync exception
For vectors we never expect to enter legitimately, the C code is
generated using a macro to avoid code duplication. The exceptions are
handled via __panic_unhandled(), replacing bad_mode() (which is
removed).
The `kernel_ventry` and `entry_handler` assembly macros are updated to
handle the new naming scheme. In theory it should be possible to
generate the entry functions at the same time as the vectors using a
single table, but this will require reworking the linker script to split
the two into separate sections, so for now we have separate tables.
Signed-off-by: Mark Rutland <mark.rutland@arm.com>
Acked-by: Catalin Marinas <catalin.marinas@arm.com>
Acked-by: Marc Zyngier <maz@kernel.org>
Reviewed-by: Joey Gouly <joey.gouly@arm.com>
Cc: James Morse <james.morse@arm.com>
Cc: Will Deacon <will@kernel.org>
Link: https://lore.kernel.org/r/20210607094624.34689-15-mark.rutland@arm.com
Signed-off-by: Will Deacon <will@kernel.org>
2021-06-07 17:46:18 +08:00
|
|
|
asmlinkage void el1t_64_sync_handler(struct pt_regs *regs);
|
|
|
|
asmlinkage void el1t_64_irq_handler(struct pt_regs *regs);
|
|
|
|
asmlinkage void el1t_64_fiq_handler(struct pt_regs *regs);
|
|
|
|
asmlinkage void el1t_64_error_handler(struct pt_regs *regs);
|
|
|
|
|
|
|
|
asmlinkage void el1h_64_sync_handler(struct pt_regs *regs);
|
|
|
|
asmlinkage void el1h_64_irq_handler(struct pt_regs *regs);
|
|
|
|
asmlinkage void el1h_64_fiq_handler(struct pt_regs *regs);
|
|
|
|
asmlinkage void el1h_64_error_handler(struct pt_regs *regs);
|
|
|
|
|
|
|
|
asmlinkage void el0t_64_sync_handler(struct pt_regs *regs);
|
|
|
|
asmlinkage void el0t_64_irq_handler(struct pt_regs *regs);
|
|
|
|
asmlinkage void el0t_64_fiq_handler(struct pt_regs *regs);
|
|
|
|
asmlinkage void el0t_64_error_handler(struct pt_regs *regs);
|
|
|
|
|
|
|
|
asmlinkage void el0t_32_sync_handler(struct pt_regs *regs);
|
|
|
|
asmlinkage void el0t_32_irq_handler(struct pt_regs *regs);
|
|
|
|
asmlinkage void el0t_32_fiq_handler(struct pt_regs *regs);
|
|
|
|
asmlinkage void el0t_32_error_handler(struct pt_regs *regs);
|
2020-12-14 19:33:53 +08:00
|
|
|
|
2021-06-07 17:46:10 +08:00
|
|
|
asmlinkage void call_on_irq_stack(struct pt_regs *regs,
|
|
|
|
void (*func)(struct pt_regs *));
|
arm64: entry: move bulk of ret_to_user to C
In `ret_to_user` we perform some conditional work depending on the
thread flags, then perform some IRQ/context tracking which is intended
to balance with the IRQ/context tracking performed in the entry C code.
For simplicity and consistency, it would be preferable to move this all
to C. As a step towards that, this patch moves the conditional work and
IRQ/context tracking into a C helper function. To aid bisectability,
this is called from the `ret_to_user` assembly, and a subsequent patch
will move the call to C code.
As local_daif_mask() handles all necessary tracing and PMR manipulation,
we no longer need to handle this explicitly. As we call
exit_to_user_mode() directly, the `user_enter_irqoff` macro is no longer
used, and can be removed. As enter_from_user_mode() and
exit_to_user_mode() are no longer called from assembly, these can be
made static, and as these are typically very small, they are marked
__always_inline to avoid the overhead of a function call.
For now, enablement of single-step is left in entry.S, and for this we
still need to read the flags in ret_to_user(). It is safe to read this
separately as TIF_SINGLESTEP is not part of _TIF_WORK_MASK.
There should be no functional change as a result of this patch.
Signed-off-by: Mark Rutland <mark.rutland@arm.com>
Cc: James Morse <james.morse@arm.com>
Cc: Joey Gouly <joey.gouly@arm.com>
Cc: Marc Zyngier <maz@kernel.org>
Cc: Will Deacon <will@kernel.org>
Reviewed-by: Joey Gouly <joey.gouly@arm.com>
Link: https://lore.kernel.org/r/20210802140733.52716-4-mark.rutland@arm.com
[catalin.marinas@arm.com: removed unused gic_prio_kentry_setup macro]
Signed-off-by: Catalin Marinas <catalin.marinas@arm.com>
2021-08-02 22:07:32 +08:00
|
|
|
asmlinkage void asm_exit_to_user_mode(struct pt_regs *regs);
|
|
|
|
|
2020-11-21 04:33:46 +08:00
|
|
|
void do_mem_abort(unsigned long far, unsigned int esr, struct pt_regs *regs);
|
2019-10-26 00:42:15 +08:00
|
|
|
void do_undefinstr(struct pt_regs *regs);
|
2020-03-17 00:50:45 +08:00
|
|
|
void do_bti(struct pt_regs *regs);
|
2019-10-26 00:42:15 +08:00
|
|
|
void do_debug_exception(unsigned long addr_if_watchpoint, unsigned int esr,
|
|
|
|
struct pt_regs *regs);
|
|
|
|
void do_fpsimd_acc(unsigned int esr, struct pt_regs *regs);
|
|
|
|
void do_sve_acc(unsigned int esr, struct pt_regs *regs);
|
2022-04-19 19:22:24 +08:00
|
|
|
void do_sme_acc(unsigned int esr, struct pt_regs *regs);
|
2019-10-26 00:42:15 +08:00
|
|
|
void do_fpsimd_exc(unsigned int esr, struct pt_regs *regs);
|
|
|
|
void do_sysinstr(unsigned int esr, struct pt_regs *regs);
|
|
|
|
void do_sp_pc_abort(unsigned long addr, unsigned int esr, struct pt_regs *regs);
|
|
|
|
void bad_el0_sync(struct pt_regs *regs, int reason, unsigned int esr);
|
|
|
|
void do_cp15instr(unsigned int esr, struct pt_regs *regs);
|
2020-01-17 02:35:47 +08:00
|
|
|
void do_el0_svc(struct pt_regs *regs);
|
|
|
|
void do_el0_svc_compat(struct pt_regs *regs);
|
2020-09-14 16:36:53 +08:00
|
|
|
void do_ptrauth_fault(struct pt_regs *regs, unsigned int esr);
|
2021-06-07 17:46:07 +08:00
|
|
|
void do_serror(struct pt_regs *regs, unsigned int esr);
|
arm64: entry: move bulk of ret_to_user to C
In `ret_to_user` we perform some conditional work depending on the
thread flags, then perform some IRQ/context tracking which is intended
to balance with the IRQ/context tracking performed in the entry C code.
For simplicity and consistency, it would be preferable to move this all
to C. As a step towards that, this patch moves the conditional work and
IRQ/context tracking into a C helper function. To aid bisectability,
this is called from the `ret_to_user` assembly, and a subsequent patch
will move the call to C code.
As local_daif_mask() handles all necessary tracing and PMR manipulation,
we no longer need to handle this explicitly. As we call
exit_to_user_mode() directly, the `user_enter_irqoff` macro is no longer
used, and can be removed. As enter_from_user_mode() and
exit_to_user_mode() are no longer called from assembly, these can be
made static, and as these are typically very small, they are marked
__always_inline to avoid the overhead of a function call.
For now, enablement of single-step is left in entry.S, and for this we
still need to read the flags in ret_to_user(). It is safe to read this
separately as TIF_SINGLESTEP is not part of _TIF_WORK_MASK.
There should be no functional change as a result of this patch.
Signed-off-by: Mark Rutland <mark.rutland@arm.com>
Cc: James Morse <james.morse@arm.com>
Cc: Joey Gouly <joey.gouly@arm.com>
Cc: Marc Zyngier <maz@kernel.org>
Cc: Will Deacon <will@kernel.org>
Reviewed-by: Joey Gouly <joey.gouly@arm.com>
Link: https://lore.kernel.org/r/20210802140733.52716-4-mark.rutland@arm.com
[catalin.marinas@arm.com: removed unused gic_prio_kentry_setup macro]
Signed-off-by: Catalin Marinas <catalin.marinas@arm.com>
2021-08-02 22:07:32 +08:00
|
|
|
void do_notify_resume(struct pt_regs *regs, unsigned long thread_flags);
|
2021-06-07 17:46:20 +08:00
|
|
|
|
|
|
|
void panic_bad_stack(struct pt_regs *regs, unsigned int esr, unsigned long far);
|
2012-03-05 19:49:34 +08:00
|
|
|
#endif /* __ASM_EXCEPTION_H */
|