|
@@ -27,19 +27,6 @@
|
|
|
|
|
|
#include "fault.h"
|
|
|
|
|
|
-/*
|
|
|
- * Fault status register encodings. We steal bit 31 for our own purposes.
|
|
|
- */
|
|
|
-#define FSR_LNX_PF (1 << 31)
|
|
|
-#define FSR_WRITE (1 << 11)
|
|
|
-#define FSR_FS4 (1 << 10)
|
|
|
-#define FSR_FS3_0 (15)
|
|
|
-
|
|
|
-static inline int fsr_fs(unsigned int fsr)
|
|
|
-{
|
|
|
- return (fsr & FSR_FS3_0) | (fsr & FSR_FS4) >> 6;
|
|
|
-}
|
|
|
-
|
|
|
#ifdef CONFIG_MMU
|
|
|
|
|
|
#ifdef CONFIG_KPROBES
|
|
@@ -123,8 +110,10 @@ void show_pte(struct mm_struct *mm, unsigned long addr)
|
|
|
|
|
|
pte = pte_offset_map(pmd, addr);
|
|
|
printk(", *pte=%08llx", (long long)pte_val(*pte));
|
|
|
+#ifndef CONFIG_ARM_LPAE
|
|
|
printk(", *ppte=%08llx",
|
|
|
(long long)pte_val(pte[PTE_HWTABLE_PTRS]));
|
|
|
+#endif
|
|
|
pte_unmap(pte);
|
|
|
} while(0);
|
|
|
|
|
@@ -441,6 +430,12 @@ do_translation_fault(unsigned long addr, unsigned int fsr,
|
|
|
pmd = pmd_offset(pud, addr);
|
|
|
pmd_k = pmd_offset(pud_k, addr);
|
|
|
|
|
|
+#ifdef CONFIG_ARM_LPAE
|
|
|
+ /*
|
|
|
+ * Only one hardware entry per PMD with LPAE.
|
|
|
+ */
|
|
|
+ index = 0;
|
|
|
+#else
|
|
|
/*
|
|
|
* On ARM one Linux PGD entry contains two hardware entries (see page
|
|
|
* tables layout in pgtable.h). We normally guarantee that we always
|
|
@@ -450,6 +445,7 @@ do_translation_fault(unsigned long addr, unsigned int fsr,
|
|
|
* for the first of pair.
|
|
|
*/
|
|
|
index = (addr >> SECTION_SHIFT) & 1;
|
|
|
+#endif
|
|
|
if (pmd_none(pmd_k[index]))
|
|
|
goto bad_area;
|
|
|
|
|
@@ -489,55 +485,20 @@ do_bad(unsigned long addr, unsigned int fsr, struct pt_regs *regs)
|
|
|
return 1;
|
|
|
}
|
|
|
|
|
|
-static struct fsr_info {
|
|
|
+struct fsr_info {
|
|
|
int (*fn)(unsigned long addr, unsigned int fsr, struct pt_regs *regs);
|
|
|
int sig;
|
|
|
int code;
|
|
|
const char *name;
|
|
|
-} fsr_info[] = {
|
|
|
- /*
|
|
|
- * The following are the standard ARMv3 and ARMv4 aborts. ARMv5
|
|
|
- * defines these to be "precise" aborts.
|
|
|
- */
|
|
|
- { do_bad, SIGSEGV, 0, "vector exception" },
|
|
|
- { do_bad, SIGBUS, BUS_ADRALN, "alignment exception" },
|
|
|
- { do_bad, SIGKILL, 0, "terminal exception" },
|
|
|
- { do_bad, SIGBUS, BUS_ADRALN, "alignment exception" },
|
|
|
- { do_bad, SIGBUS, 0, "external abort on linefetch" },
|
|
|
- { do_translation_fault, SIGSEGV, SEGV_MAPERR, "section translation fault" },
|
|
|
- { do_bad, SIGBUS, 0, "external abort on linefetch" },
|
|
|
- { do_page_fault, SIGSEGV, SEGV_MAPERR, "page translation fault" },
|
|
|
- { do_bad, SIGBUS, 0, "external abort on non-linefetch" },
|
|
|
- { do_bad, SIGSEGV, SEGV_ACCERR, "section domain fault" },
|
|
|
- { do_bad, SIGBUS, 0, "external abort on non-linefetch" },
|
|
|
- { do_bad, SIGSEGV, SEGV_ACCERR, "page domain fault" },
|
|
|
- { do_bad, SIGBUS, 0, "external abort on translation" },
|
|
|
- { do_sect_fault, SIGSEGV, SEGV_ACCERR, "section permission fault" },
|
|
|
- { do_bad, SIGBUS, 0, "external abort on translation" },
|
|
|
- { do_page_fault, SIGSEGV, SEGV_ACCERR, "page permission fault" },
|
|
|
- /*
|
|
|
- * The following are "imprecise" aborts, which are signalled by bit
|
|
|
- * 10 of the FSR, and may not be recoverable. These are only
|
|
|
- * supported if the CPU abort handler supports bit 10.
|
|
|
- */
|
|
|
- { do_bad, SIGBUS, 0, "unknown 16" },
|
|
|
- { do_bad, SIGBUS, 0, "unknown 17" },
|
|
|
- { do_bad, SIGBUS, 0, "unknown 18" },
|
|
|
- { do_bad, SIGBUS, 0, "unknown 19" },
|
|
|
- { do_bad, SIGBUS, 0, "lock abort" }, /* xscale */
|
|
|
- { do_bad, SIGBUS, 0, "unknown 21" },
|
|
|
- { do_bad, SIGBUS, BUS_OBJERR, "imprecise external abort" }, /* xscale */
|
|
|
- { do_bad, SIGBUS, 0, "unknown 23" },
|
|
|
- { do_bad, SIGBUS, 0, "dcache parity error" }, /* xscale */
|
|
|
- { do_bad, SIGBUS, 0, "unknown 25" },
|
|
|
- { do_bad, SIGBUS, 0, "unknown 26" },
|
|
|
- { do_bad, SIGBUS, 0, "unknown 27" },
|
|
|
- { do_bad, SIGBUS, 0, "unknown 28" },
|
|
|
- { do_bad, SIGBUS, 0, "unknown 29" },
|
|
|
- { do_bad, SIGBUS, 0, "unknown 30" },
|
|
|
- { do_bad, SIGBUS, 0, "unknown 31" }
|
|
|
};
|
|
|
|
|
|
+/* FSR definition */
|
|
|
+#ifdef CONFIG_ARM_LPAE
|
|
|
+#include "fsr-3level.c"
|
|
|
+#else
|
|
|
+#include "fsr-2level.c"
|
|
|
+#endif
|
|
|
+
|
|
|
void __init
|
|
|
hook_fault_code(int nr, int (*fn)(unsigned long, unsigned int, struct pt_regs *),
|
|
|
int sig, int code, const char *name)
|
|
@@ -573,42 +534,6 @@ do_DataAbort(unsigned long addr, unsigned int fsr, struct pt_regs *regs)
|
|
|
arm_notify_die("", regs, &info, fsr, 0);
|
|
|
}
|
|
|
|
|
|
-
|
|
|
-static struct fsr_info ifsr_info[] = {
|
|
|
- { do_bad, SIGBUS, 0, "unknown 0" },
|
|
|
- { do_bad, SIGBUS, 0, "unknown 1" },
|
|
|
- { do_bad, SIGBUS, 0, "debug event" },
|
|
|
- { do_bad, SIGSEGV, SEGV_ACCERR, "section access flag fault" },
|
|
|
- { do_bad, SIGBUS, 0, "unknown 4" },
|
|
|
- { do_translation_fault, SIGSEGV, SEGV_MAPERR, "section translation fault" },
|
|
|
- { do_bad, SIGSEGV, SEGV_ACCERR, "page access flag fault" },
|
|
|
- { do_page_fault, SIGSEGV, SEGV_MAPERR, "page translation fault" },
|
|
|
- { do_bad, SIGBUS, 0, "external abort on non-linefetch" },
|
|
|
- { do_bad, SIGSEGV, SEGV_ACCERR, "section domain fault" },
|
|
|
- { do_bad, SIGBUS, 0, "unknown 10" },
|
|
|
- { do_bad, SIGSEGV, SEGV_ACCERR, "page domain fault" },
|
|
|
- { do_bad, SIGBUS, 0, "external abort on translation" },
|
|
|
- { do_sect_fault, SIGSEGV, SEGV_ACCERR, "section permission fault" },
|
|
|
- { do_bad, SIGBUS, 0, "external abort on translation" },
|
|
|
- { do_page_fault, SIGSEGV, SEGV_ACCERR, "page permission fault" },
|
|
|
- { do_bad, SIGBUS, 0, "unknown 16" },
|
|
|
- { do_bad, SIGBUS, 0, "unknown 17" },
|
|
|
- { do_bad, SIGBUS, 0, "unknown 18" },
|
|
|
- { do_bad, SIGBUS, 0, "unknown 19" },
|
|
|
- { do_bad, SIGBUS, 0, "unknown 20" },
|
|
|
- { do_bad, SIGBUS, 0, "unknown 21" },
|
|
|
- { do_bad, SIGBUS, 0, "unknown 22" },
|
|
|
- { do_bad, SIGBUS, 0, "unknown 23" },
|
|
|
- { do_bad, SIGBUS, 0, "unknown 24" },
|
|
|
- { do_bad, SIGBUS, 0, "unknown 25" },
|
|
|
- { do_bad, SIGBUS, 0, "unknown 26" },
|
|
|
- { do_bad, SIGBUS, 0, "unknown 27" },
|
|
|
- { do_bad, SIGBUS, 0, "unknown 28" },
|
|
|
- { do_bad, SIGBUS, 0, "unknown 29" },
|
|
|
- { do_bad, SIGBUS, 0, "unknown 30" },
|
|
|
- { do_bad, SIGBUS, 0, "unknown 31" },
|
|
|
-};
|
|
|
-
|
|
|
void __init
|
|
|
hook_ifault_code(int nr, int (*fn)(unsigned long, unsigned int, struct pt_regs *),
|
|
|
int sig, int code, const char *name)
|
|
@@ -641,6 +566,7 @@ do_PrefetchAbort(unsigned long addr, unsigned int ifsr, struct pt_regs *regs)
|
|
|
arm_notify_die("", regs, &info, ifsr, 0);
|
|
|
}
|
|
|
|
|
|
+#ifndef CONFIG_ARM_LPAE
|
|
|
static int __init exceptions_init(void)
|
|
|
{
|
|
|
if (cpu_architecture() >= CPU_ARCH_ARMv6) {
|
|
@@ -663,3 +589,4 @@ static int __init exceptions_init(void)
|
|
|
}
|
|
|
|
|
|
arch_initcall(exceptions_init);
|
|
|
+#endif
|