[PATCH v2 0/5] arm64: report EL1 exceptions better
Mark Rutland
mark.rutland at arm.com
Tue Sep 13 03:17:27 PDT 2022
We treat BTI, FPAC, and UNDEFINED exceptions as fatal when taken from
EL1, but due to the way we make these fatal, we end up throwing away
information that could be used to diagnose the cause of the exception.
We have an anti-pattern where in the handler for the exception we do:
BUG_ON(!user_mode(regs));
... and consequently, the BUG handler logs the context of the exception
handler which contained the BUG_ON(), rather than the context from which
the fatal exception was taken.
For example, today we report UNDEFINED exceptions as:
| kernel BUG at arch/arm64/kernel/traps.c:497!
| Internal error: Oops - BUG: 0 [#1] PREEMPT SMP
| Modules linked in:
| CPU: 0 PID: 0 Comm: swapper Not tainted 5.19.0-rc3-00127-geff044f1b04e-dirty #3
| Hardware name: linux,dummy-virt (DT)
| pstate: 000000c5 (nzcv daIF -PAN -UAO -TCO -DIT -SSBS BTYPE=--)
| pc : do_undefinstr+0x28c/0x2ac
| lr : do_undefinstr+0x298/0x2ac
| sp : ffff800009f63bc0
| x29: ffff800009f63bc0 x28: ffff800009f73c00 x27: ffff800009644a70
| x26: ffff8000096778a8 x25: 0000000000000040 x24: 0000000000000000
| x23: 00000000800000c5 x22: ffff800009894060 x21: ffff800009f63d90
| x20: 0000000000000000 x19: ffff800009f63c40 x18: 0000000000000006
| x17: 0000000000403000 x16: 00000000bfbfd000 x15: ffff800009f63830
| x14: ffffffffffffffff x13: 0000000000000000 x12: 0000000000000019
| x11: 0101010101010101 x10: 0000000000161b98 x9 : 0000000000000000
| x8 : 0000000000000000 x7 : 0000000000000000 x6 : 0000000000000000
| x5 : ffff800009f761d0 x4 : 0000000000000000 x3 : ffff80000a2b80f8
| x2 : 0000000000000000 x1 : ffff800009f73c00 x0 : 00000000800000c5
| Call trace:
| do_undefinstr+0x28c/0x2ac
| el1_undef+0x2c/0x4c
| el1h_64_sync_handler+0x84/0xd0
| el1h_64_sync+0x64/0x68
| setup_arch+0x550/0x598
| start_kernel+0x88/0x6ac
| __primary_switched+0xb8/0xc0
| Code: 17ffff95 a9425bf5 17ffffb8 a9025bf5 (d4210000)
... where the code dump is for the BRK instruction in the BUG_ON(), and
the register contents are for the innards of do_undefinstr(). The
instruction which triggered the UNDEFINED exception is nowhere in the
dump.
This series reworks the handling of these exceptions to be more useful,
reporting the original context of the exception, and calling die()
directly rather than using BUG_ON().
For example, as of this series being applied, UNDEFINED exceptions are
reported as:
| Internal error: Oops - Undefined instruction: 0 [#1] PREEMPT SMP
| Modules linked in:
| CPU: 0 PID: 0 Comm: swapper Not tainted 5.19.0-rc3-00128-gf27cfcc80e52-dirty #5
| Hardware name: linux,dummy-virt (DT)
| pstate: 800000c5 (Nzcv daIF -PAN -UAO -TCO -DIT -SSBS BTYPE=--)
| pc : setup_arch+0x550/0x598
| lr : setup_arch+0x50c/0x598
| sp : ffff800009f63d90
| x29: ffff800009f63d90 x28: 0000000081000200 x27: ffff800009644a70
| x26: ffff8000096778c8 x25: 0000000000000040 x24: 0000000000000000
| x23: 0000000000000100 x22: ffff800009f69a58 x21: ffff80000a2b80b8
| x20: 0000000000000000 x19: 0000000000000000 x18: 0000000000000006
| x17: 0000000000403000 x16: 00000000bfbfd000 x15: ffff800009f63830
| x14: ffffffffffffffff x13: 0000000000000000 x12: 0000000000000019
| x11: 0101010101010101 x10: 0000000000161b98 x9 : 0000000000000000
| x8 : 0000000000000000 x7 : 0000000000000000 x6 : 0000000000000000
| x5 : 0000000000000008 x4 : 0000000000000010 x3 : 0000000000000000
| x2 : 0000000000000000 x1 : 0000000000000000 x0 : 0000000000000000
| Call trace:
| setup_arch+0x550/0x598
| start_kernel+0x88/0x6ac
| __primary_switched+0xb8/0xc0
| Code: b4000080 90ffed80 912ac000 97db745f (00000000)
... where the code dump includes the instruction which triggered the
exception, and the register contents are for the context the instruction
was executed in.
Since v1 [1]:
* Rebase (trivial) to v6.0-rc3
* Fix typos
* Consistently pass ESR to die()
* Add Reviewed-by tags
[1] https://lore.kernel.org/linux-arm-kernel/20220712161452.4142444-1-mark.rutland@arm.com/
Mark.
Mark Rutland (5):
arm64: report EL1 UNDEFs better
arm64: die(): pass 'err' as long
arm64: consistently pass ESR_ELx to die()
arm64: rework FPAC exception handling
arm64: rework BTI exception handling
arch/arm64/include/asm/exception.h | 8 +++--
arch/arm64/include/asm/system_misc.h | 2 +-
arch/arm64/kernel/entry-common.c | 32 +++++++++++++------
arch/arm64/kernel/traps.c | 48 +++++++++++++++++-----------
4 files changed, 57 insertions(+), 33 deletions(-)
--
2.30.2
More information about the linux-arm-kernel
mailing list