linux-arch.vger.kernel.org archive mirror
 help / color / mirror / Atom feed
From: Ley Foon Tan <lftan@altera.com>
To: linux-arch@vger.kernel.org, linux-kernel@vger.kernel.org,
	linux-doc@vger.kernel.org
Cc: Ley Foon Tan <lftan@altera.com>,
	lftan.linux@gmail.com, cltang@codesourcery.com
Subject: [PATCH v3 08/29] nios2: MMU Fault handling
Date: Mon, 8 Sep 2014 17:22:19 +0800	[thread overview]
Message-ID: <1410168160-3624-9-git-send-email-lftan@altera.com> (raw)
In-Reply-To: <1410168160-3624-1-git-send-email-lftan@altera.com>

This patch adds support for the handling of the MMU faults (exception
entry code introduced by a previous patch, kernel/entry.S).

Signed-off-by: Ley Foon Tan <lftan@altera.com>
---
 arch/nios2/mm/extable.c |  25 +++++
 arch/nios2/mm/fault.c   | 251 ++++++++++++++++++++++++++++++++++++++++++++++++
 2 files changed, 276 insertions(+)
 create mode 100644 arch/nios2/mm/extable.c
 create mode 100644 arch/nios2/mm/fault.c

diff --git a/arch/nios2/mm/extable.c b/arch/nios2/mm/extable.c
new file mode 100644
index 0000000..4d2fc5a
--- /dev/null
+++ b/arch/nios2/mm/extable.c
@@ -0,0 +1,25 @@
+/*
+ * Copyright (C) 2010, Tobias Klauser <tklauser@distanz.ch>
+ * Copyright (C) 2009, Wind River Systems Inc
+ *   Implemented by fredrik.markstrom@gmail.com and ivarholmqvist@gmail.com
+ *
+ * This file is subject to the terms and conditions of the GNU General Public
+ * License.  See the file "COPYING" in the main directory of this archive
+ * for more details.
+ */
+
+#include <linux/module.h>
+#include <linux/uaccess.h>
+
+int fixup_exception(struct pt_regs *regs)
+{
+	const struct exception_table_entry *fixup;
+
+	fixup = search_exception_tables(regs->ea);
+	if (fixup) {
+		regs->ea = fixup->fixup;
+		return 1;
+	}
+
+	return 0;
+}
diff --git a/arch/nios2/mm/fault.c b/arch/nios2/mm/fault.c
new file mode 100644
index 0000000..15a0bb5
--- /dev/null
+++ b/arch/nios2/mm/fault.c
@@ -0,0 +1,251 @@
+/*
+ * Copyright (C) 2009 Wind River Systems Inc
+ *   Implemented by fredrik.markstrom@gmail.com and ivarholmqvist@gmail.com
+ *
+ * based on arch/mips/mm/fault.c which is:
+ *
+ * Copyright (C) 1995-2000 Ralf Baechle
+ *
+ * This file is subject to the terms and conditions of the GNU General Public
+ * License.  See the file "COPYING" in the main directory of this archive
+ * for more details.
+ */
+
+#include <linux/signal.h>
+#include <linux/sched.h>
+#include <linux/interrupt.h>
+#include <linux/kernel.h>
+#include <linux/errno.h>
+#include <linux/string.h>
+#include <linux/types.h>
+#include <linux/ptrace.h>
+#include <linux/mman.h>
+#include <linux/mm.h>
+#include <linux/module.h>
+#include <linux/uaccess.h>
+#include <linux/ptrace.h>
+
+#include <asm/mmu_context.h>
+#include <asm/traps.h>
+
+#define EXC_SUPERV_INSN_ACCESS	9  /* Supervisor only instruction address */
+#define EXC_SUPERV_DATA_ACCESS	11 /* Supervisor only data address */
+#define EXC_X_PROTECTION_FAULT	13 /* TLB permission violation (x) */
+#define EXC_R_PROTECTION_FAULT	14 /* TLB permission violation (r) */
+#define EXC_W_PROTECTION_FAULT	15 /* TLB permission violation (w) */
+
+/*
+ * This routine handles page faults.  It determines the address,
+ * and the problem, and then passes it off to one of the appropriate
+ * routines.
+ */
+asmlinkage void do_page_fault(struct pt_regs *regs, unsigned long cause,
+				unsigned long address)
+{
+	struct vm_area_struct *vma = NULL;
+	struct task_struct *tsk = current;
+	struct mm_struct *mm = tsk->mm;
+	int code = SEGV_MAPERR;
+	int fault;
+	unsigned int flags = 0;
+
+	cause >>= 2;
+
+	/* Restart the instruction */
+	regs->ea -= 4;
+
+	/*
+	 * We fault-in kernel-space virtual memory on-demand. The
+	 * 'reference' page table is init_mm.pgd.
+	 *
+	 * NOTE! We MUST NOT take any locks for this case. We may
+	 * be in an interrupt or a critical region, and should
+	 * only copy the information from the master page table,
+	 * nothing more.
+	 */
+	if (unlikely(address >= VMALLOC_START && address <= VMALLOC_END)) {
+		if (user_mode(regs))
+			goto bad_area_nosemaphore;
+		else
+			goto vmalloc_fault;
+	}
+
+	if (unlikely(address >= TASK_SIZE))
+		goto bad_area_nosemaphore;
+
+	/*
+	 * If we're in an interrupt or have no user
+	 * context, we must not take the fault..
+	 */
+	if (in_atomic() || !mm)
+		goto bad_area_nosemaphore;
+
+	if (user_mode(regs))
+		flags |= FAULT_FLAG_USER;
+
+	if (!down_read_trylock(&mm->mmap_sem)) {
+		if (!user_mode(regs) && !search_exception_tables(regs->ea))
+			goto bad_area_nosemaphore;
+		down_read(&mm->mmap_sem);
+	}
+
+	vma = find_vma(mm, address);
+	if (!vma)
+		goto bad_area;
+	if (vma->vm_start <= address)
+		goto good_area;
+	if (!(vma->vm_flags & VM_GROWSDOWN))
+		goto bad_area;
+	if (expand_stack(vma, address))
+		goto bad_area;
+/*
+ * Ok, we have a good vm_area for this memory access, so
+ * we can handle it..
+ */
+good_area:
+	code = SEGV_ACCERR;
+
+	switch (cause) {
+	case EXC_SUPERV_INSN_ACCESS:
+		goto bad_area;
+	case EXC_SUPERV_DATA_ACCESS:
+		goto bad_area;
+	case EXC_X_PROTECTION_FAULT:
+		if (!(vma->vm_flags & VM_EXEC))
+			goto bad_area;
+		break;
+	case EXC_R_PROTECTION_FAULT:
+		if (!(vma->vm_flags & VM_READ))
+			goto bad_area;
+		break;
+	case EXC_W_PROTECTION_FAULT:
+		if (!(vma->vm_flags & VM_WRITE))
+			goto bad_area;
+		flags = FAULT_FLAG_WRITE;
+		break;
+	}
+
+survive:
+	/*
+	 * If for any reason at all we couldn't handle the fault,
+	 * make sure we exit gracefully rather than endlessly redo
+	 * the fault.
+	 */
+	fault = handle_mm_fault(mm, vma, address, flags);
+	if (unlikely(fault & VM_FAULT_ERROR)) {
+		if (fault & VM_FAULT_OOM)
+			goto out_of_memory;
+		else if (fault & VM_FAULT_SIGBUS)
+			goto do_sigbus;
+		BUG();
+	}
+	if (fault & VM_FAULT_MAJOR)
+		tsk->maj_flt++;
+	else
+		tsk->min_flt++;
+
+	up_read(&mm->mmap_sem);
+	return;
+
+/*
+ * Something tried to access memory that isn't in our memory map..
+ * Fix it, but check if it's kernel or user first..
+ */
+bad_area:
+	up_read(&mm->mmap_sem);
+
+bad_area_nosemaphore:
+	/* User mode accesses just cause a SIGSEGV */
+	if (user_mode(regs)) {
+		pr_alert("%s: unhandled page fault (%d) at 0x%08lx, "
+			"cause %ld\n", current->comm, SIGSEGV, address, cause);
+		show_regs(regs);
+		_exception(SIGSEGV, regs, code, address);
+		return;
+	}
+
+no_context:
+	/* Are we prepared to handle this kernel fault? */
+	if (fixup_exception(regs))
+		return;
+
+	/*
+	 * Oops. The kernel tried to access some bad page. We'll have to
+	 * terminate things with extreme prejudice.
+	 */
+	bust_spinlocks(1);
+
+	pr_alert("Unable to handle kernel %s at virtual address %08lx",
+		address < PAGE_SIZE ? "NULL pointer dereference" :
+		"paging request", address);
+	pr_alert("ea = %08lx, ra = %08lx, cause = %ld\n", regs->ea, regs->ra,
+		cause);
+	panic("Oops");
+	return;
+
+/*
+ * We ran out of memory, or some other thing happened to us that made
+ * us unable to handle the page fault gracefully.
+ */
+out_of_memory:
+	up_read(&mm->mmap_sem);
+	if (is_global_init(tsk)) {
+		yield();
+		down_read(&mm->mmap_sem);
+		goto survive;
+	}
+	if (!user_mode(regs))
+		goto no_context;
+	pagefault_out_of_memory();
+	return;
+
+do_sigbus:
+	up_read(&mm->mmap_sem);
+
+	/* Kernel mode? Handle exceptions or die */
+	if (!user_mode(regs))
+		goto no_context;
+
+	_exception(SIGBUS, regs, BUS_ADRERR, address);
+	return;
+
+vmalloc_fault:
+	{
+		/*
+		 * Synchronize this task's top level page-table
+		 * with the 'reference' page table.
+		 *
+		 * Do _not_ use "tsk" here. We might be inside
+		 * an interrupt in the middle of a task switch..
+		 */
+		int offset = pgd_index(address);
+		pgd_t *pgd, *pgd_k;
+		pud_t *pud, *pud_k;
+		pmd_t *pmd, *pmd_k;
+		pte_t *pte_k;
+
+		pgd = pgd_current + offset;
+		pgd_k = init_mm.pgd + offset;
+
+		if (!pgd_present(*pgd_k))
+			goto no_context;
+		set_pgd(pgd, *pgd_k);
+
+		pud = pud_offset(pgd, address);
+		pud_k = pud_offset(pgd_k, address);
+		if (!pud_present(*pud_k))
+			goto no_context;
+		pmd = pmd_offset(pud, address);
+		pmd_k = pmd_offset(pud_k, address);
+		if (!pmd_present(*pmd_k))
+			goto no_context;
+		set_pmd(pmd, *pmd_k);
+
+		pte_k = pte_offset_kernel(pmd_k, address);
+		if (!pte_present(*pte_k))
+			goto no_context;
+
+		flush_tlb_one(address);
+		return;
+	}
+}
-- 
1.8.2.1

  parent reply	other threads:[~2014-09-08  9:22 UTC|newest]

Thread overview: 97+ messages / expand[flat|nested]  mbox.gz  Atom feed  top
2014-09-08  9:22 [PATCH v3 00/29] nios2 Linux kernel port Ley Foon Tan
2014-09-08  9:22 ` Ley Foon Tan
2014-09-08  9:22 ` [PATCH v3 01/29] asm-generic: add generic futex for !CONFIG_SMP Ley Foon Tan
2014-09-23 10:20   ` LF.Tan
2014-09-23 10:20     ` LF.Tan
2014-09-23 10:40     ` Arnd Bergmann
2014-09-23 10:40       ` Arnd Bergmann
2014-09-24 10:18       ` Ley Foon Tan
2014-09-24 10:18         ` Ley Foon Tan
2014-09-24 10:40         ` Arnd Bergmann
2014-09-24 10:40           ` Arnd Bergmann
2014-09-24 10:57           ` Ley Foon Tan
2014-09-24 11:10             ` Arnd Bergmann
2014-09-24 11:10               ` Arnd Bergmann
2014-09-25  8:33               ` Ley Foon Tan
2014-09-25  8:33                 ` Ley Foon Tan
2014-09-25 10:54                 ` Arnd Bergmann
2014-09-25 10:54                   ` Arnd Bergmann
2014-09-26  2:22                   ` Ley Foon Tan
2014-09-26  2:22                     ` Ley Foon Tan
2014-09-26  6:41                     ` Geert Uytterhoeven
2014-09-26  6:41                       ` Geert Uytterhoeven
2014-09-26  8:30                       ` Ley Foon Tan
2014-09-26 13:17               ` Tobias Klauser
2014-09-26 13:17                 ` Tobias Klauser
2014-09-26 13:33                 ` Arnd Bergmann
2014-09-26 13:33                   ` Arnd Bergmann
2014-09-23 21:47     ` Thomas Gleixner
2014-09-23 21:47       ` Thomas Gleixner
2014-09-24 10:29       ` Ley Foon Tan
2014-09-24 10:29         ` Ley Foon Tan
2014-09-08  9:22 ` [PATCH v3 02/29] nios2: Assembly macros and definitions Ley Foon Tan
2014-09-08  9:22   ` Ley Foon Tan
2014-09-08  9:22 ` [PATCH v3 03/29] nios2: Kernel booting and initialization Ley Foon Tan
2014-09-08  9:22   ` Ley Foon Tan
2014-09-08  9:22 ` [PATCH v3 04/29] nios2: Exception handling Ley Foon Tan
2014-09-08  9:22   ` Ley Foon Tan
2014-09-08  9:22 ` [PATCH v3 05/29] nios2: Traps exception handling Ley Foon Tan
2014-09-08  9:22 ` [PATCH v3 06/29] nios2: Memory management Ley Foon Tan
2014-09-08  9:22   ` Ley Foon Tan
2014-09-08  9:22 ` [PATCH v3 07/29] nios2: I/O Mapping Ley Foon Tan
2014-09-08  9:22   ` Ley Foon Tan
2014-09-08  9:22 ` Ley Foon Tan [this message]
2014-09-08  9:22   ` [PATCH v3 08/29] nios2: MMU Fault handling Ley Foon Tan
2014-09-08  9:22 ` [PATCH v3 09/29] nios2: Page table management Ley Foon Tan
2014-09-08  9:22 ` [PATCH v3 10/29] nios2: Process management Ley Foon Tan
2014-09-08  9:22   ` Ley Foon Tan
2014-09-08  9:22 ` [PATCH v3 11/29] nios2: Cache handling Ley Foon Tan
2014-09-08  9:22   ` Ley Foon Tan
2014-09-08  9:22 ` [PATCH v3 12/29] nios2: TLB handling Ley Foon Tan
2014-09-08  9:22   ` Ley Foon Tan
2014-09-08  9:22 ` [PATCH v3 13/29] nios2: Interrupt handling Ley Foon Tan
2014-09-08  9:22   ` Ley Foon Tan
2014-09-08  9:22 ` [PATCH v3 14/29] nios2: DMA mapping API Ley Foon Tan
2014-09-08  9:22   ` Ley Foon Tan
2014-09-08  9:22 ` [PATCH v3 15/29] Add ELF machine define for Nios2 Ley Foon Tan
2014-09-08  9:22   ` Ley Foon Tan
2014-09-08  9:22 ` [PATCH v3 16/29] nios2: ELF definitions Ley Foon Tan
2014-09-08  9:22   ` Ley Foon Tan
2014-09-08  9:22 ` [PATCH v3 17/29] nios2: System calls handling Ley Foon Tan
2014-09-08  9:22   ` Ley Foon Tan
2014-09-08  9:22 ` [PATCH v3 18/29] nios2: Signal handling support Ley Foon Tan
2014-09-08  9:22   ` Ley Foon Tan
2014-09-08  9:22 ` [PATCH v3 19/29] nios2: Library functions Ley Foon Tan
2014-09-08  9:22   ` Ley Foon Tan
2014-09-08  9:22 ` [PATCH v3 20/29] nios2: Device tree support Ley Foon Tan
2014-09-08  9:22   ` Ley Foon Tan
2014-09-08  9:22 ` [PATCH v3 21/29] nios2: Time keeping Ley Foon Tan
2014-09-08  9:22   ` Ley Foon Tan
2014-09-08  9:22 ` [PATCH v3 22/29] nios2: Cpuinfo handling Ley Foon Tan
2014-09-08  9:22   ` Ley Foon Tan
2014-09-08  9:22 ` [PATCH v3 23/29] nios2: Miscellaneous header files Ley Foon Tan
2014-09-08  9:22 ` [PATCH v3 24/29] nios2: Nios2 registers Ley Foon Tan
2014-09-08  9:22   ` Ley Foon Tan
2014-09-08  9:22 ` [PATCH v3 25/29] nios2: Module support Ley Foon Tan
2014-09-08  9:22   ` Ley Foon Tan
2014-09-09 21:25   ` Valdis.Kletnieks
2014-09-10  7:29     ` Chung-Lin Tang
2014-09-10  7:29       ` Chung-Lin Tang
2014-09-10  7:49       ` Tobias Klauser
2014-09-08  9:22 ` [PATCH v3 26/29] nios2: ptrace support Ley Foon Tan
2014-09-08  9:22   ` Ley Foon Tan
2014-09-16  9:43   ` Tobias Klauser
2014-09-16  9:43     ` Tobias Klauser
2014-09-18  5:10     ` Ley Foon Tan
2014-09-08  9:22 ` [PATCH v3 27/29] MAINTAINERS: Add nios2 maintainer Ley Foon Tan
2014-09-08  9:22   ` Ley Foon Tan
2014-09-08  9:22 ` [PATCH v3 28/29] Documentation: Add documentation for Nios2 architecture Ley Foon Tan
2014-09-08  9:22   ` Ley Foon Tan
2014-09-26  2:27   ` Ley Foon Tan
2014-09-26  2:27     ` Ley Foon Tan
2014-09-26  8:31     ` Ley Foon Tan
2014-09-08  9:22 ` [PATCH v3 29/29] nios2: Build infrastructure Ley Foon Tan
2014-09-08  9:22   ` Ley Foon Tan
2014-09-08 12:13 ` [PATCH v3 00/29] nios2 Linux kernel port David Howells
2014-09-09  2:02 ` Al Viro
2014-09-09  2:02   ` Al Viro

Reply instructions:

You may reply publicly to this message via plain-text email
using any one of the following methods:

* Save the following mbox file, import it into your mail client,
  and reply-to-all from there: mbox

  Avoid top-posting and favor interleaved quoting:
  https://en.wikipedia.org/wiki/Posting_style#Interleaved_style

* Reply using the --to, --cc, and --in-reply-to
  switches of git-send-email(1):

  git send-email \
    --in-reply-to=1410168160-3624-9-git-send-email-lftan@altera.com \
    --to=lftan@altera.com \
    --cc=cltang@codesourcery.com \
    --cc=lftan.linux@gmail.com \
    --cc=linux-arch@vger.kernel.org \
    --cc=linux-doc@vger.kernel.org \
    --cc=linux-kernel@vger.kernel.org \
    /path/to/YOUR_REPLY

  https://kernel.org/pub/software/scm/git/docs/git-send-email.html

* If your mail client supports setting the In-Reply-To header
  via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for NNTP newsgroup(s).