Skip to content

Commit 9adeb8e

Browse files
labbottctmarinas
authored andcommitted
arm64: Handle el1 synchronous instruction aborts cleanly
Executing from a non-executable area gives an ugly message: lkdtm: Performing direct entry EXEC_RODATA lkdtm: attempting ok execution at ffff0000084c0e08 lkdtm: attempting bad execution at ffff000008880700 Bad mode in Synchronous Abort handler detected on CPU2, code 0x8400000e -- IABT (current EL) CPU: 2 PID: 998 Comm: sh Not tainted 4.7.0-rc2+ torvalds#13 Hardware name: linux,dummy-virt (DT) task: ffff800077e35780 ti: ffff800077970000 task.ti: ffff800077970000 PC is at lkdtm_rodata_do_nothing+0x0/0x8 LR is at execute_location+0x74/0x88 The 'IABT (current EL)' indicates the error but it's a bit cryptic without knowledge of the ARM ARM. There is also no indication of the specific address which triggered the fault. The increase in kernel page permissions makes hitting this case more likely as well. Handling the case in the vectors gives a much more familiar looking error message: lkdtm: Performing direct entry EXEC_RODATA lkdtm: attempting ok execution at ffff0000084c0840 lkdtm: attempting bad execution at ffff000008880680 Unable to handle kernel paging request at virtual address ffff000008880680 pgd = ffff8000089b2000 [ffff000008880680] *pgd=00000000489b4003, *pud=0000000048904003, *pmd=0000000000000000 Internal error: Oops: 8400000e [#1] PREEMPT SMP Modules linked in: CPU: 1 PID: 997 Comm: sh Not tainted 4.7.0-rc1+ torvalds#24 Hardware name: linux,dummy-virt (DT) task: ffff800077f9f080 ti: ffff800008a1c000 task.ti: ffff800008a1c000 PC is at lkdtm_rodata_do_nothing+0x0/0x8 LR is at execute_location+0x74/0x88 Acked-by: Mark Rutland <mark.rutland@arm.com> Signed-off-by: Laura Abbott <labbott@redhat.com> Signed-off-by: Catalin Marinas <catalin.marinas@arm.com>
1 parent ad05711 commit 9adeb8e

File tree

2 files changed

+19
-2
lines changed

2 files changed

+19
-2
lines changed

arch/arm64/kernel/entry.S

Lines changed: 7 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -353,6 +353,8 @@ el1_sync:
353353
lsr x24, x1, #ESR_ELx_EC_SHIFT // exception class
354354
cmp x24, #ESR_ELx_EC_DABT_CUR // data abort in EL1
355355
b.eq el1_da
356+
cmp x24, #ESR_ELx_EC_IABT_CUR // instruction abort in EL1
357+
b.eq el1_ia
356358
cmp x24, #ESR_ELx_EC_SYS64 // configurable trap
357359
b.eq el1_undef
358360
cmp x24, #ESR_ELx_EC_SP_ALIGN // stack alignment exception
@@ -364,6 +366,11 @@ el1_sync:
364366
cmp x24, #ESR_ELx_EC_BREAKPT_CUR // debug exception in EL1
365367
b.ge el1_dbg
366368
b el1_inv
369+
370+
el1_ia:
371+
/*
372+
* Fall through to the Data abort case
373+
*/
367374
el1_da:
368375
/*
369376
* Data abort handling

arch/arm64/mm/fault.c

Lines changed: 12 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -153,6 +153,11 @@ int ptep_set_access_flags(struct vm_area_struct *vma,
153153
}
154154
#endif
155155

156+
static bool is_el1_instruction_abort(unsigned int esr)
157+
{
158+
return ESR_ELx_EC(esr) == ESR_ELx_EC_IABT_CUR;
159+
}
160+
156161
/*
157162
* The kernel tried to access some page that wasn't present.
158163
*/
@@ -161,8 +166,9 @@ static void __do_kernel_fault(struct mm_struct *mm, unsigned long addr,
161166
{
162167
/*
163168
* Are we prepared to handle this kernel fault?
169+
* We are almost certainly not prepared to handle instruction faults.
164170
*/
165-
if (fixup_exception(regs))
171+
if (!is_el1_instruction_abort(esr) && fixup_exception(regs))
166172
return;
167173

168174
/*
@@ -267,7 +273,8 @@ static inline bool is_permission_fault(unsigned int esr)
267273
unsigned int ec = ESR_ELx_EC(esr);
268274
unsigned int fsc_type = esr & ESR_ELx_FSC_TYPE;
269275

270-
return (ec == ESR_ELx_EC_DABT_CUR && fsc_type == ESR_ELx_FSC_PERM);
276+
return (ec == ESR_ELx_EC_DABT_CUR && fsc_type == ESR_ELx_FSC_PERM) ||
277+
(ec == ESR_ELx_EC_IABT_CUR && fsc_type == ESR_ELx_FSC_PERM);
271278
}
272279

273280
static bool is_el0_instruction_abort(unsigned int esr)
@@ -312,6 +319,9 @@ static int __kprobes do_page_fault(unsigned long addr, unsigned int esr,
312319
if (regs->orig_addr_limit == KERNEL_DS)
313320
die("Accessing user space memory with fs=KERNEL_DS", regs, esr);
314321

322+
if (is_el1_instruction_abort(esr))
323+
die("Attempting to execute userspace memory", regs, esr);
324+
315325
if (!search_exception_tables(regs->pc))
316326
die("Accessing user space memory outside uaccess.h routines", regs, esr);
317327
}

0 commit comments

Comments
 (0)