[PATCH 5/5] arm64: rework BTI exception handling

Anshuman Khandual anshuman.khandual at arm.com
Tue Jul 12 22:08:02 PDT 2022



On 7/12/22 21:44, Mark Rutland wrote:
> If a BTI exception is taken from EL1, the entry code will treat this as
> an unhandled exception and will panic() the kernel. This is inconsistent
> with the way we handle FPAC exceptions, which have a dedicated handler
> and only necessarily kill the thread from which the exception was taken
> from, and we don't log all the information that could be relevant to
> debug the issue.
> 
> The code in do_bti() has:
> 
> 	BUG_ON(!user_mode(regs));
> 
> ... and it seems like the intent was to call this for EL1 BTI
> exceptions, as with FPAC, but this was omitted due to an oversight.
> 
> This patch adds separate EL0 and EL1 BTI exception handlers, with the
> latter calling die() directly to report the original context the BTI
> exception was taken from. This matches our handling of FPAC exceptions.
> 
> Prior to this patch, a BTI failure is reported as:
> 
> | Unhandled 64-bit el1h sync exception on CPU0, ESR 0x0000000034000002 -- BTI
> | CPU: 0 PID: 1 Comm: swapper/0 Not tainted 5.19.0-rc3-00131-g7d937ff0221d-dirty #9
> | Hardware name: linux,dummy-virt (DT)
> | pstate: 20400809 (nzCv daif +PAN -UAO -TCO -DIT -SSBS BTYPE=-c)
> | pc : test_bti_callee+0x4/0x10
> | lr : test_bti_caller+0x1c/0x28
> | sp : ffff80000800bdf0
> | x29: ffff80000800bdf0 x28: 0000000000000000 x27: 0000000000000000
> | x26: 0000000000000000 x25: 0000000000000000 x24: 0000000000000000
> | x23: ffff80000a2b8000 x22: 0000000000000000 x21: 0000000000000000
> | x20: ffff8000099fa5b0 x19: ffff800009ff7000 x18: fffffbfffda37000
> | x17: 3120676e696d7573 x16: 7361202c6e6f6974 x15: 0000000041a90000
> | x14: 0040000000000041 x13: 0040000000000001 x12: ffff000001a90000
> | x11: fffffbfffda37480 x10: 0068000000000703 x9 : 0001000040000000
> | x8 : 0000000000090000 x7 : 0068000000000f03 x6 : 0060000000000f83
> | x5 : ffff80000a2b6000 x4 : ffff0000028d0000 x3 : ffff800009f78378
> | x2 : 0000000000000000 x1 : 0000000040210000 x0 : ffff8000080257e4
> | Kernel panic - not syncing: Unhandled exception
> | CPU: 0 PID: 1 Comm: swapper/0 Not tainted 5.19.0-rc3-00131-g7d937ff0221d-dirty #9
> | Hardware name: linux,dummy-virt (DT)
> | Call trace:
> |  dump_backtrace.part.0+0xcc/0xe0
> |  show_stack+0x18/0x5c
> |  dump_stack_lvl+0x64/0x80
> |  dump_stack+0x18/0x34
> |  panic+0x170/0x360
> |  arm64_exit_nmi.isra.0+0x0/0x80
> |  el1h_64_sync_handler+0x64/0xd0
> |  el1h_64_sync+0x64/0x68
> |  test_bti_callee+0x4/0x10
> |  smp_cpus_done+0xb0/0xbc
> |  smp_init+0x7c/0x8c
> |  kernel_init_freeable+0x128/0x28c
> |  kernel_init+0x28/0x13c
> |  ret_from_fork+0x10/0x20
> 
> With this patch applied, a BTI failure is reported as:
> 
> | Internal error: Oops - BTI: 0000000034000002 [#1] PREEMPT SMP
> | Modules linked in:
> | CPU: 0 PID: 1 Comm: swapper/0 Not tainted 5.19.0-rc3-00132-g0ad98265d582-dirty #8
> | Hardware name: linux,dummy-virt (DT)
> | pstate: 20400809 (nzCv daif +PAN -UAO -TCO -DIT -SSBS BTYPE=-c)
> | pc : test_bti_callee+0x4/0x10
> | lr : test_bti_caller+0x1c/0x28
> | sp : ffff80000800bdf0
> | x29: ffff80000800bdf0 x28: 0000000000000000 x27: 0000000000000000
> | x26: 0000000000000000 x25: 0000000000000000 x24: 0000000000000000
> | x23: ffff80000a2b8000 x22: 0000000000000000 x21: 0000000000000000
> | x20: ffff8000099fa5b0 x19: ffff800009ff7000 x18: fffffbfffda37000
> | x17: 3120676e696d7573 x16: 7361202c6e6f6974 x15: 0000000041a90000
> | x14: 0040000000000041 x13: 0040000000000001 x12: ffff000001a90000
> | x11: fffffbfffda37480 x10: 0068000000000703 x9 : 0001000040000000
> | x8 : 0000000000090000 x7 : 0068000000000f03 x6 : 0060000000000f83
> | x5 : ffff80000a2b6000 x4 : ffff0000028d0000 x3 : ffff800009f78378
> | x2 : 0000000000000000 x1 : 0000000040210000 x0 : ffff800008025804
> | Call trace:
> |  test_bti_callee+0x4/0x10
> |  smp_cpus_done+0xb0/0xbc
> |  smp_init+0x7c/0x8c
> |  kernel_init_freeable+0x128/0x28c
> |  kernel_init+0x28/0x13c
> |  ret_from_fork+0x10/0x20
> | Code: d50323bf d53cd040 d65f03c0 d503233f (d50323bf)
> 
> Signed-off-by: Mark Rutland <mark.rutland at arm.com>
> Cc: Alexandru Elisei <alexandru.elisei at arm.com>
> Cc: Amit Daniel Kachhap <amit.kachhap at arm.com>
> Cc: Catalin Marinas <catalin.marinas at arm.com>
> Cc: James Morse <james.morse at arm.com>
> Cc: Mark Brown <broonie at kernel.org>
> Cc: Will Deacon <will at kernel.org>

Reviewed-by: Anshuman Khandual <anshuman.khandual at arm.com>

> ---
>  arch/arm64/include/asm/exception.h |  3 ++-
>  arch/arm64/kernel/entry-common.c   | 14 +++++++++++++-
>  arch/arm64/kernel/traps.c          | 10 +++++++---
>  3 files changed, 22 insertions(+), 5 deletions(-)
> 
> diff --git a/arch/arm64/include/asm/exception.h b/arch/arm64/include/asm/exception.h
> index 3716ab3edd22..ff6eb08f270e 100644
> --- a/arch/arm64/include/asm/exception.h
> +++ b/arch/arm64/include/asm/exception.h
> @@ -59,7 +59,8 @@ asmlinkage void asm_exit_to_user_mode(struct pt_regs *regs);
>  
>  void do_mem_abort(unsigned long far, unsigned long esr, struct pt_regs *regs);
>  void do_undefinstr(struct pt_regs *regs);
> -void do_bti(struct pt_regs *regs);
> +void do_el0_bti(struct pt_regs *regs);
> +void do_el1_bti(struct pt_regs *regs, unsigned long esr);
>  void do_debug_exception(unsigned long addr_if_watchpoint, unsigned long esr,
>  			struct pt_regs *regs);
>  void do_fpsimd_acc(unsigned long esr, struct pt_regs *regs);
> diff --git a/arch/arm64/kernel/entry-common.c b/arch/arm64/kernel/entry-common.c
> index 1bbfdb8b76df..c643f42eea9e 100644
> --- a/arch/arm64/kernel/entry-common.c
> +++ b/arch/arm64/kernel/entry-common.c
> @@ -388,6 +388,15 @@ static void noinstr el1_undef(struct pt_regs *regs)
>  	exit_to_kernel_mode(regs);
>  }
>  
> +static void noinstr el1_bti(struct pt_regs *regs, unsigned long esr)
> +{
> +	enter_from_kernel_mode(regs);
> +	local_daif_inherit(regs);
> +	do_el1_bti(regs, esr);
> +	local_daif_mask();
> +	exit_to_kernel_mode(regs);
> +}
> +
>  static void noinstr el1_dbg(struct pt_regs *regs, unsigned long esr)
>  {
>  	unsigned long far = read_sysreg(far_el1);
> @@ -427,6 +436,9 @@ asmlinkage void noinstr el1h_64_sync_handler(struct pt_regs *regs)
>  	case ESR_ELx_EC_UNKNOWN:
>  		el1_undef(regs);
>  		break;
> +	case ESR_ELx_EC_BTI:
> +		el1_bti(regs, esr);
> +		break;
>  	case ESR_ELx_EC_BREAKPT_CUR:
>  	case ESR_ELx_EC_SOFTSTP_CUR:
>  	case ESR_ELx_EC_WATCHPT_CUR:
> @@ -594,7 +606,7 @@ static void noinstr el0_bti(struct pt_regs *regs)
>  {
>  	enter_from_user_mode(regs);
>  	local_daif_restore(DAIF_PROCCTX);
> -	do_bti(regs);
> +	do_el0_bti(regs);
>  	exit_to_user_mode(regs);
>  }
>  
> diff --git a/arch/arm64/kernel/traps.c b/arch/arm64/kernel/traps.c
> index d96aa17279d7..5425ad141e06 100644
> --- a/arch/arm64/kernel/traps.c
> +++ b/arch/arm64/kernel/traps.c
> @@ -501,12 +501,16 @@ void do_undefinstr(struct pt_regs *regs)
>  }
>  NOKPROBE_SYMBOL(do_undefinstr);
>  
> -void do_bti(struct pt_regs *regs)
> +void do_el0_bti(struct pt_regs *regs)
>  {
> -	BUG_ON(!user_mode(regs));
>  	force_signal_inject(SIGILL, ILL_ILLOPC, regs->pc, 0);
>  }
> -NOKPROBE_SYMBOL(do_bti);
> +
> +void do_el1_bti(struct pt_regs *regs, unsigned long esr)
> +{
> +	die("Oops - BTI", regs, esr);
> +}
> +NOKPROBE_SYMBOL(do_el1_bti);
>  
>  void do_el0_fpac(struct pt_regs *regs, unsigned long esr)
>  {



More information about the linux-arm-kernel mailing list