patch-2.2.4 linux/arch/sparc64/mm/fault.c
Next file: linux/arch/sparc64/mm/generic.c
Previous file: linux/arch/sparc64/math-emu/soft-fp.h
Back to the patch index
Back to the overall index
- Lines: 231
- Date:
Tue Mar 16 21:52:06 1999
- Orig file:
v2.2.3/linux/arch/sparc64/mm/fault.c
- Orig date:
Thu Nov 19 09:56:27 1998
diff -u --recursive --new-file v2.2.3/linux/arch/sparc64/mm/fault.c linux/arch/sparc64/mm/fault.c
@@ -1,8 +1,8 @@
-/* $Id: fault.c,v 1.26 1998/11/08 11:14:03 davem Exp $
+/* $Id: fault.c,v 1.34 1999/03/16 12:12:28 jj Exp $
* arch/sparc64/mm/fault.c: Page fault handlers for the 64-bit Sparc.
*
* Copyright (C) 1996 David S. Miller (davem@caip.rutgers.edu)
- * Copyright (C) 1997 Jakub Jelinek (jj@sunsite.mff.cuni.cz)
+ * Copyright (C) 1997, 1999 Jakub Jelinek (jj@ultra.linux.cz)
*/
#include <asm/head.h>
@@ -14,6 +14,8 @@
#include <linux/signal.h>
#include <linux/mm.h>
#include <linux/smp_lock.h>
+#include <linux/init.h>
+#include <linux/interrupt.h>
#include <asm/page.h>
#include <asm/pgtable.h>
@@ -26,7 +28,7 @@
extern struct sparc_phys_banks sp_banks[SPARC_PHYS_BANKS];
/* Nice, simple, prom library does all the sweating for us. ;) */
-unsigned long prom_probe_memory (void)
+unsigned long __init prom_probe_memory (void)
{
register struct linux_mlist_p1275 *mlist;
register unsigned long bytes, base_paddr, tally;
@@ -35,7 +37,7 @@
i = 0;
mlist = *prom_meminfo()->p1275_available;
bytes = tally = mlist->num_bytes;
- base_paddr = (unsigned int) mlist->start_adr;
+ base_paddr = mlist->start_adr;
sp_banks[0].base_addr = base_paddr;
sp_banks[0].num_bytes = bytes;
@@ -55,12 +57,12 @@
break;
}
- sp_banks[i].base_addr = (unsigned long) mlist->start_adr;
+ sp_banks[i].base_addr = mlist->start_adr;
sp_banks[i].num_bytes = mlist->num_bytes;
}
i++;
- sp_banks[i].base_addr = 0xdeadbeef;
+ sp_banks[i].base_addr = 0xdeadbeefbeefdeadUL;
sp_banks[i].num_bytes = 0;
/* Now mask all bank sizes on a page boundary, it is all we can
@@ -72,26 +74,12 @@
return tally;
}
-/* Traverse the memory lists in the prom to see how much physical we
- * have.
- */
-unsigned long
-probe_memory(void)
-{
- unsigned long total;
-
- total = prom_probe_memory();
-
- /* Oh man, much nicer, keep the dirt in promlib. */
- return total;
-}
-
void unhandled_fault(unsigned long address, struct task_struct *tsk,
struct pt_regs *regs)
{
if((unsigned long) address < PAGE_SIZE) {
printk(KERN_ALERT "Unable to handle kernel NULL "
- "pointer dereference");
+ "pointer dereference\n");
} else {
printk(KERN_ALERT "Unable to handle kernel paging request "
"at virtual address %016lx\n", (unsigned long)address);
@@ -100,22 +88,74 @@
(unsigned long) tsk->mm->context);
printk(KERN_ALERT "tsk->mm->pgd = %016lx\n",
(unsigned long) tsk->mm->pgd);
- lock_kernel();
die_if_kernel("Oops", regs);
- unlock_kernel();
}
/* #define DEBUG_EXCEPTIONS */
/* #define DEBUG_LOCKUPS */
+/* #define INSN_VPTE_LOOKUP */
+
+static inline u32 get_user_insn(unsigned long tpc)
+{
+ u32 insn;
+#ifndef INSN_VPTE_LOOKUP
+ pgd_t *pgdp = pgd_offset(current->mm, tpc);
+ pmd_t *pmdp;
+ pte_t *ptep;
+
+ if(pgd_none(*pgdp))
+ return 0;
+ pmdp = pmd_offset(pgdp, tpc);
+ if(pmd_none(*pmdp))
+ return 0;
+ ptep = pte_offset(pmdp, tpc);
+ if(!pte_present(*ptep))
+ return 0;
+ insn = *(unsigned int *)(pte_page(*ptep) + (tpc & ~PAGE_MASK));
+#else
+ register unsigned long pte asm("l1");
+
+ /* So that we don't pollute TLB, we read the instruction
+ * using PHYS bypass. For that, we of course need
+ * to know its page table entry. Do this by simulating
+ * dtlb_miss handler. -jj */
+ pte = ((((long)tpc) >> (PAGE_SHIFT-3)) & ~7);
+ asm volatile ("
+ rdpr %%pstate, %%l0
+ wrpr %%l0, %2, %%pstate
+ wrpr %%g0, 1, %%tl
+ mov %%l1, %%g6
+ ldxa [%%g3 + %%l1] %3, %%g5
+ mov %%g5, %%l1
+ wrpr %%g0, 0, %%tl
+ wrpr %%l0, 0, %%pstate
+ " : "=r" (pte) : "0" (pte), "i" (PSTATE_MG|PSTATE_IE), "i" (ASI_S) : "l0");
+
+ if ((long)pte >= 0) return 0;
+
+ pte = (pte & _PAGE_PADDR) + (tpc & ~PAGE_MASK);
+ asm ("lduwa [%1] %2, %0" : "=r" (insn) : "r" (pte), "i" (ASI_PHYS_USE_EC));
+#endif
+
+ return insn;
+}
+
asmlinkage void do_sparc64_fault(struct pt_regs *regs, unsigned long address, int write)
{
struct mm_struct *mm = current->mm;
struct vm_area_struct *vma;
+ unsigned int insn = 0;
#ifdef DEBUG_LOCKUPS
static unsigned long lastaddr, lastpc;
static int lastwrite, lockcnt;
#endif
+ /*
+ * If we're in an interrupt or have no user
+ * context, we must not take the fault..
+ */
+ if (in_interrupt() || mm == &init_mm)
+ goto do_kernel_fault;
down(&mm->mmap_sem);
#ifdef DEBUG_LOCKUPS
@@ -135,6 +175,21 @@
vma = find_vma(mm, address);
if(!vma)
goto bad_area;
+#ifndef INSN_VPTE_LOOKUP
+ write &= 0xf;
+#else
+ if (write & 0x10) {
+ write = 0;
+ if((vma->vm_flags & VM_WRITE)) {
+ if (regs->tstate & TSTATE_PRIV)
+ insn = *(unsigned int *)regs->tpc;
+ else
+ insn = get_user_insn(regs->tpc);
+ if ((insn & 0xc0200000) == 0xc0200000 && (insn & 0x1780000) != 0x1680000)
+ write = 1;
+ }
+ }
+#endif
if(vma->vm_start <= address)
goto good_area;
if(!(vma->vm_flags & VM_GROWSDOWN))
@@ -168,16 +223,44 @@
do_kernel_fault:
{
- unsigned long g2 = regs->u_regs[UREG_G2];
+ unsigned long g2;
+ unsigned char asi = ASI_P;
+
+ if (!insn) {
+ if (regs->tstate & TSTATE_PRIV)
+ insn = *(unsigned int *)regs->tpc;
+ else
+ insn = get_user_insn(regs->tpc);
+ }
+ if (write != 1 && (insn & 0xc0800000) == 0xc0800000) {
+ if (insn & 0x2000)
+ asi = (regs->tstate >> 24);
+ else
+ asi = (insn >> 5);
+ if ((asi & 0xf2) == 0x82) {
+ /* This was a non-faulting load. Just clear the
+ destination register(s) and continue with the next
+ instruction. -jj */
+ if (insn & 0x1000000) {
+ extern int handle_ldf_stq(u32, struct pt_regs *);
+
+ handle_ldf_stq(insn, regs);
+ } else {
+ extern int handle_ld_nf(u32, struct pt_regs *);
+
+ handle_ld_nf(insn, regs);
+ }
+ return;
+ }
+ }
+
+ g2 = regs->u_regs[UREG_G2];
/* Is this in ex_table? */
if (regs->tstate & TSTATE_PRIV) {
- unsigned char asi = ASI_P;
- unsigned int insn;
unsigned long fixup;
-
- insn = *(unsigned int *)regs->tpc;
- if ((insn & 0xc0800000) == 0xc0800000) {
+
+ if (asi == ASI_P && (insn & 0xc0800000) == 0xc0800000) {
if (insn & 0x2000)
asi = (regs->tstate >> 24);
else
FUNET's LINUX-ADM group, linux-adm@nic.funet.fi
TCL-scripts by Sam Shen (who was at: slshen@lbl.gov)