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

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)