summaryrefslogtreecommitdiff
path: root/sys/arch
diff options
context:
space:
mode:
Diffstat (limited to 'sys/arch')
-rw-r--r--sys/arch/sparc/sparc/locore.s105
1 files changed, 84 insertions, 21 deletions
diff --git a/sys/arch/sparc/sparc/locore.s b/sys/arch/sparc/sparc/locore.s
index 51a1cb6d9f0..2c2ed132bbe 100644
--- a/sys/arch/sparc/sparc/locore.s
+++ b/sys/arch/sparc/sparc/locore.s
@@ -1,6 +1,11 @@
/*
* Copyright (c) 1992, 1993
* The Regents of the University of California. All rights reserved.
+ * Copyright (c) 1994 Theo de Raadt
+ *
+ * All advertising materials mentioning features or use of this software
+ * must display the following acknowledgement:
+ * This product includes software developed by Theo de Raadt.
*
* This software was developed by the Computer Systems Engineering group
* at Lawrence Berkeley Laboratory under DARPA contract BG 91-66 and
@@ -328,7 +333,7 @@ start:
_trapbase:
/* trap 0 is special since we cannot receive it */
b dostart; nop; nop; nop ! 00 = reset (fake)
- VTRAP(T_TEXTFAULT, memfault) ! 01 = instr. fetch fault
+ VTRAP(T_TEXTFAULT, 0) ! 01 = instr. fetch fault (backpatch to memfault)
TRAP(T_ILLINST) ! 02 = illegal instruction
TRAP(T_PRIVINST) ! 03 = privileged instruction
TRAP(T_FPDISABLED) ! 04 = fp instr, but EF bit off in psr
@@ -336,7 +341,7 @@ _trapbase:
WINDOW_UF ! 06 = window underflow
TRAP(T_ALIGN) ! 07 = address alignment error
VTRAP(T_FPE, fp_exception) ! 08 = fp exception
- VTRAP(T_DATAFAULT, memfault) ! 09 = data fetch fault
+ VTRAP(T_DATAFAULT, 0) ! 09 = data fetch fault (backpatch to memfault)
TRAP(T_TAGOF) ! 0a = tag overflow
UTRAP(0x0b)
UTRAP(0x0c)
@@ -1073,21 +1078,15 @@ ctw_invalid:
* a+8 AC_ASYNC_ERR
* a+12 AC_ASYNC_VA
*/
-memfault:
+
+#if defined(SUN4)
+memfault_sun4:
TRAP_SETUP(-CCFSZ-80)
INCR(_cnt+V_FAULTS) ! cnt.v_faults++ (clobbers %o0,%o1)
st %g1, [%sp + CCFSZ + 20] ! save g1
rd %y, %l4 ! save y
-#if defined(SUN4) && (defined(SUN4C) || defined(SUN4M))
- sethi %hi(_cputyp), %o0 ! what cpu are we running on?
- ld [%o0 + %lo(_cputyp)], %o0
- cmp %o0, CPU_SUN4
- bne 9f
- nop
-#endif
-#if defined(SUN4)
/*
* registers:
* memerr.ctrl = memory error control reg., error if 0x80 set
@@ -1121,9 +1120,9 @@ memfault:
xnormal_mem_fault:
/*
- * have to make SUN4 emulate SUN4C. 4C code expects
- * SER in %o1 and the offending VA in %o2, everything else is ok.
- * (must figure out if SER_WRITE should be set)
+ * have to make SUN4 emulate SUN4C. SUN4C code expects SER in
+ * %o1 and the offending VA in %o2, everything else remains the
+ * same, but we must figure out if SER_WRITE should be set.
*/
set AC_BUS_ERR, %o0 ! bus error register
cmp %l3, T_TEXTFAULT ! text fault always on PC
@@ -1139,13 +1138,17 @@ xnormal_mem_fault:
sethi %hi(SER_WRITE), %o5 ! damn SER_WRITE wont fit simm13
! or %lo(SER_WRITE), %o5, %o5! not necessary since %lo is zero
or %o5, %o1, %o1 ! set SER_WRITE
-#if defined(SUN4C) || defined(SUN4M)
ba normal_mem_fault
nop ! XXX make efficient later
-#endif /* SUN4C || SUN4M */
#endif /* SUN4 */
-9:
-#if defined(SUN4C) || defined(SUN4M)
+
+#if defined(SUN4C)
+memfault_sun4c:
+ TRAP_SETUP(-CCFSZ-80)
+ INCR(_cnt+V_FAULTS) ! cnt.v_faults++ (clobbers %o0,%o1)
+
+ st %g1, [%sp + CCFSZ + 20] ! save g1
+ rd %y, %l4 ! save y
#if AC_SYNC_ERR + 4 != AC_SYNC_VA || \
AC_SYNC_ERR + 8 != AC_ASYNC_ERR || AC_SYNC_ERR + 12 != AC_ASYNC_VA
@@ -1225,7 +1228,28 @@ xnormal_mem_fault:
b return_from_trap
wr %l0, 0, %psr
/* NOTREACHED */
-#endif /* SUN4C || SUN4M */
+#endif /* SUN4C */
+
+#if defined(SUN4M)
+memfault_sun4m:
+ TRAP_SETUP(-CCFSZ-80)
+ INCR(_cnt+V_FAULTS) ! cnt.v_faults++ (clobbers %o0,%o1)
+
+ st %g1, [%sp + CCFSZ + 20] ! save g1
+ rd %y, %l4 ! save y
+
+ mov SRMMU_FAULTSTAT, %o0
+ std %g2, [%sp + CCFSZ + 24] ! save g2, g3
+ lda [%o0] ASI_SRMMU, %o1 ! srmmu fault status
+ mov SRMMU_FAULTADDR, %o0
+ std %g4, [%sp + CCFSZ + 32] ! (sneak g4,g5 in here)
+ lda [%o0] ASI_SRMMU, %o2 ! srmmu fault address
+ ! XXX check for memory errors?
+ std %g6, [%sp + CCFSZ + 40]
+ b normal_mem_fault ! no, just a regular fault
+ wr %l0, PSR_ET, %psr ! (and reenable traps)
+ /* NOTREACHED */
+#endif /* SUN4M */
normal_mem_fault:
/*
@@ -2518,6 +2542,8 @@ dostart:
is_sun4m:
#if defined(SUN4M)
mov SUN4CM_PGSHIFT, %g5
+
+ set memfault_sun4m, %o1 ! ptr to our memfault routine
b start_havetype
mov CPU_SUN4M, %g4
#else
@@ -2537,6 +2563,7 @@ is_sun4c:
set AC_CONTEXT, %g1 ! paranoia: set context to kernel
stba %g0, [%g1] ASI_CONTROL
+ set memfault_sun4c, %o1 ! ptr to our memfault routine
b start_havetype
mov CPU_SUN4C, %g4 ! XXX CPU_SUN4
#else
@@ -2570,6 +2597,7 @@ is_sun4:
set AC_CONTEXT, %g1 ! paranoia: set context to kernel
stba %g0, [%g1] ASI_CONTROL
+ set memfault_sun4, %o1 ! ptr to our memfault routine
b start_havetype
mov CPU_SUN4, %g4
#else
@@ -2587,6 +2615,41 @@ is_sun4:
start_havetype:
/*
+ * Back-patch the T_{TEXT,DATA}FAULT vectors to branch to the
+ * correct memfault routine. There is a different memfault routine
+ * for each processor. Here we carefully synthesize the offsets
+ * and merge them into the "ba" instruction which already exists.
+ */
+ set KERNBASE, %o2 ! unmapped as yet
+ sub %o1, %o2, %o1
+ set _trapbase-KERNBASE, %o2 ! and store in trap table
+ set 0xffc00000, %o4 ! 22bit offset in "bcc" instruction
+
+ ! patch instr fetch fault vector to point to our memfault routine
+ ld [%o2 + (16*T_TEXTFAULT+4)], %o3 ! fetch "ba" instruction
+ and %o3, %o4, %o3 ! mask out garbage
+ sub %o1, %o2, %o5 ! generate relative offset
+ sub %o5, (16*T_TEXTFAULT+4), %o5
+ srl %o5, 2, %o5 ! long-word offset
+ or %o3, %o5, %o3 ! merge into "ba" instruction
+ st %o3, [%o2 + (16*T_TEXTFAULT+4)]
+
+ ! patch data fetch fault vector to point to our memfault routine
+ ld [%o2 + (16*T_DATAFAULT+4)], %o3 ! fetch "ba" instruction
+ and %o3, %o4, %o3 ! mask out garbage
+ sub %o1, %o2, %o5 ! generate relative offset
+ sub %o5, (16*T_DATAFAULT+4), %o5
+ srl %o5, 2, %o5 ! long-word offset
+ or %o3, %o5, %o3 ! merge into "ba" instruction
+ st %o3, [%o2 + (16*T_DATAFAULT+4)]
+
+ /*
+ * XXX
+ * We just modified the text segment. We should flush that cache
+ * line!
+ */
+
+ /*
* Step 1: double map low RAM (addresses [0.._end-start-1])
* to KERNBASE (addresses [KERNBASE.._end-1]). None of these
* are `bad' aliases (since they are all on segment boundaries)
@@ -2653,7 +2716,7 @@ start_havetype:
#if defined(MMU_3L)
set AC_IDPROM+1, %l3
lduba [%l3] ASI_CONTROL, %l3
- cmp %l3, 0x24 ! XXX - SUN4_400
+ cmp %l3, SUN4_400
bne no_3mmu
add %l0, 2, %l0 ! get to proper half-word in RG space
add %l1, 2, %l1
@@ -3772,7 +3835,7 @@ Lsw_havectx:
sethi %hi(_cputyp), %o1 ! what cpu are we running on?
ld [%o1 + %lo(_cputyp)], %o1
cmp %o1, CPU_SUN4M
- bne 1f
+ beq 1f
nop
#endif
#if defined(SUN4) || defined(SUN4C)