summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--sys/arch/m88k/m88k/m88100_machdep.c329
1 files changed, 171 insertions, 158 deletions
diff --git a/sys/arch/m88k/m88k/m88100_machdep.c b/sys/arch/m88k/m88k/m88100_machdep.c
index e00fed95207..359019d215d 100644
--- a/sys/arch/m88k/m88k/m88100_machdep.c
+++ b/sys/arch/m88k/m88k/m88100_machdep.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: m88100_machdep.c,v 1.5 2007/11/20 21:46:18 miod Exp $ */
+/* $OpenBSD: m88100_machdep.c,v 1.6 2007/12/02 21:17:17 miod Exp $ */
/*
* Mach Operating System
* Copyright (c) 1993-1991 Carnegie Mellon University
@@ -26,8 +26,6 @@
* rights to redistribute these changes.
*/
-#include "assym.h" /* EF_xxx */
-
#include <sys/param.h>
#include <sys/systm.h>
@@ -46,10 +44,22 @@ const struct {
unsigned char offset;
unsigned char size;
} dmt_en_info[16] = {
- {0, 0}, {3, DMT_BYTE}, {2, DMT_BYTE}, {2, DMT_HALF},
- {1, DMT_BYTE}, {0, 0}, {0, 0}, {0, 0},
- {0, DMT_BYTE}, {0, 0}, {0, 0}, {0, 0},
- {0, DMT_HALF}, {0, 0}, {0, 0}, {0, DMT_WORD}
+ {0, 0},
+ {3, DMT_BYTE},
+ {2, DMT_BYTE},
+ {2, DMT_HALF},
+ {1, DMT_BYTE},
+ {0, 0},
+ {0, 0},
+ {0, 0},
+ {0, DMT_BYTE},
+ {0, 0},
+ {0, 0},
+ {0, 0},
+ {0, DMT_HALF},
+ {0, 0},
+ {0, 0},
+ {0, DMT_WORD}
};
#ifdef DATA_DEBUG
@@ -64,56 +74,69 @@ int data_access_emulation_debug = 0;
#define DAE_DEBUG(stuff)
#endif
+void dae_print_one(u_int, u_int, u_int, u_int);
+void dae_process(struct trapframe *, u_int, u_int, u_int, u_int);
+
void
-dae_print(u_int *eframe)
+dae_print(u_int *f)
{
- int x;
- u_int dmax, dmdx, dmtx;
+ struct trapframe *eframe = (void *)f;
- if (!ISSET(eframe[EF_DMT0], DMT_VALID))
+ if (!ISSET(eframe->tf_dmt0, DMT_VALID))
return;
- for (x = 0; x < 3; x++) {
- dmtx = eframe[EF_DMT0 + x * 3];
- if (!ISSET(dmtx, DMT_VALID))
- continue;
+ dae_print_one(0, eframe->tf_dma0, eframe->tf_dmd0, eframe->tf_dmt0);
+ dae_print_one(1, eframe->tf_dma1, eframe->tf_dmd1, eframe->tf_dmt1);
+ dae_print_one(2, eframe->tf_dma2, eframe->tf_dmd2, eframe->tf_dmt2);
+}
- dmdx = eframe[EF_DMD0 + x * 3];
- dmax = eframe[EF_DMA0 + x * 3];
+void
+dae_print_one(u_int x, u_int dmax, u_int dmdx, u_int dmtx)
+{
+ if (!ISSET(dmtx, DMT_VALID))
+ return;
- if (ISSET(dmtx, DMT_WRITE))
- printf("[DMT%d=%x: st.%c %x to %x as %d %s %s]\n",
- x, dmtx, dmtx & DMT_DAS ? 's' : 'u', dmdx, dmax,
- DMT_ENBITS(dmtx),
- dmtx & DMT_DOUB1 ? "double": "not double",
- dmtx & DMT_LOCKBAR ? "xmem": "not xmem");
- else
- printf("[DMT%d=%x: ld.%c r%d <- %x as %d %s %s]\n",
- x, dmtx, dmtx & DMT_DAS ? 's' : 'u',
- DMT_DREGBITS(dmtx), dmax, DMT_ENBITS(dmtx),
- dmtx & DMT_DOUB1 ? "double": "not double",
- dmtx & DMT_LOCKBAR ? "xmem": "not xmem");
- }
+ if (ISSET(dmtx, DMT_WRITE))
+ printf("[DMT%d=%x: st.%c %x to %x as %d %s %s]\n",
+ x, dmtx, dmtx & DMT_DAS ? 's' : 'u', dmdx, dmax,
+ DMT_ENBITS(dmtx),
+ dmtx & DMT_DOUB1 ? "double": "not double",
+ dmtx & DMT_LOCKBAR ? "xmem": "not xmem");
+ else
+ printf("[DMT%d=%x: ld.%c r%d <- %x as %d %s %s]\n",
+ x, dmtx, dmtx & DMT_DAS ? 's' : 'u',
+ DMT_DREGBITS(dmtx), dmax, DMT_ENBITS(dmtx),
+ dmtx & DMT_DOUB1 ? "double": "not double",
+ dmtx & DMT_LOCKBAR ? "xmem": "not xmem");
}
void
-data_access_emulation(u_int *eframe)
+data_access_emulation(u_int *f)
{
- int x;
- u_int dmax, dmdx, dmtx;
- u_int v, reg;
+ struct trapframe *eframe = (void *)f;
- dmtx = eframe[EF_DMT0];
- if (!ISSET(dmtx, DMT_VALID) || ISSET(dmtx, DMT_SKIP))
+ if (!ISSET(eframe->tf_dmt0, DMT_VALID) ||
+ ISSET(eframe->tf_dmt0, DMT_SKIP))
return;
- for (x = 0; x < 3; x++) {
- dmtx = eframe[EF_DMT0 + x * 3];
- if (!ISSET(dmtx, DMT_VALID))
- continue;
+ dae_process(eframe, 0,
+ eframe->tf_dma0, eframe->tf_dmd0, eframe->tf_dmt0);
+ dae_process(eframe, 1,
+ eframe->tf_dma1, eframe->tf_dmd1, eframe->tf_dmt1);
+ dae_process(eframe, 2,
+ eframe->tf_dma2, eframe->tf_dmd2, eframe->tf_dmt2);
+
+ eframe->tf_dmt0 |= DMT_SKIP;
+}
- dmdx = eframe[EF_DMD0 + x * 3];
- dmax = eframe[EF_DMA0 + x * 3];
+void
+dae_process(struct trapframe *eframe, u_int x,
+ u_int dmax, u_int dmdx, u_int dmtx)
+{
+ u_int v, reg;
+
+ if (!ISSET(dmtx, DMT_VALID))
+ return;
DAE_DEBUG(
if (ISSET(dmtx, DMT_WRITE))
@@ -130,136 +153,126 @@ data_access_emulation(u_int *eframe)
dmtx & DMT_LOCKBAR ? "xmem": "not xmem")
);
- dmax += dmt_en_info[DMT_ENBITS(dmtx)].offset;
- reg = DMT_DREGBITS(dmtx);
+ dmax += dmt_en_info[DMT_ENBITS(dmtx)].offset;
+ reg = DMT_DREGBITS(dmtx);
- if (!ISSET(dmtx, DMT_LOCKBAR)) {
- /* the fault is not during an XMEM */
+ if (!ISSET(dmtx, DMT_LOCKBAR)) {
+ /* the fault is not during an XMEM */
- if (x == 2 && ISSET(dmtx, DMT_DOUB1)) {
- /* pipeline 2 (earliest stage) for a double */
+ if (x == 2 && ISSET(dmtx, DMT_DOUB1)) {
+ /* pipeline 2 (earliest stage) for a double */
- if (ISSET(dmtx, DMT_WRITE)) {
- /*
- * STORE DOUBLE WILL BE REINITIATED
- * BY rte
- */
- } else {
- /* EMULATE ld.d INSTRUCTION */
- v = do_load_word(dmax, dmtx & DMT_DAS);
- if (reg != 0)
- eframe[EF_R0 + reg] = v;
- v = do_load_word(dmax ^ 4,
- dmtx & DMT_DAS);
- if (reg != 31)
- eframe[EF_R0 + reg + 1] = v;
- }
+ if (ISSET(dmtx, DMT_WRITE)) {
+ /*
+ * STORE DOUBLE WILL BE REINITIATED BY rte
+ */
} else {
- /* not pipeline #2 with a double */
- if (dmtx & DMT_WRITE) {
- switch (dmt_en_info[DMT_ENBITS(dmtx)].size) {
- case DMT_BYTE:
- DAE_DEBUG(
- printf("[byte %x -> [%x(%c)]\n",
- dmdx & 0xff, dmax,
- ISSET(dmtx, DMT_DAS) ? 's' : 'u')
- );
- do_store_byte(dmax, dmdx,
- dmtx & DMT_DAS);
- break;
- case DMT_HALF:
- DAE_DEBUG(
- printf("[half %x -> [%x(%c)]\n",
- dmdx & 0xffff, dmax,
- ISSET(dmtx, DMT_DAS) ? 's' : 'u')
- );
- do_store_half(dmax, dmdx,
- dmtx & DMT_DAS);
- break;
- case DMT_WORD:
- DAE_DEBUG(
- printf("[word %x -> [%x(%c)]\n",
- dmdx, dmax,
- ISSET(dmtx, DMT_DAS) ? 's' : 'u')
- );
- do_store_word(dmax, dmdx,
- dmtx & DMT_DAS);
- break;
- }
- } else {
- /* else it's a read */
- switch (dmt_en_info[DMT_ENBITS(dmtx)].size) {
- case DMT_BYTE:
- v = do_load_byte(dmax,
- dmtx & DMT_DAS);
- if (!ISSET(dmtx, DMT_SIGNED))
- v &= 0x000000ff;
- break;
- case DMT_HALF:
- v = do_load_half(dmax,
- dmtx & DMT_DAS);
- if (!ISSET(dmtx, DMT_SIGNED))
- v &= 0x0000ffff;
- break;
- case DMT_WORD:
- v = do_load_word(dmax,
- dmtx & DMT_DAS);
- break;
- }
- DAE_DEBUG(
- if (reg == 0)
- printf("[no write to r0 done]\n");
- else
- printf("[r%d <- %x]\n", reg, v);
- );
- if (reg != 0)
- eframe[EF_R0 + reg] = v;
- }
+ /* EMULATE ld.d INSTRUCTION */
+ v = do_load_word(dmax, dmtx & DMT_DAS);
+ if (reg != 0)
+ eframe->tf_r[reg] = v;
+ v = do_load_word(dmax ^ 4, dmtx & DMT_DAS);
+ if (reg != 31)
+ eframe->tf_r[reg + 1] = v;
}
} else {
- /* if lockbar is set... it's part of an XMEM */
- /*
- * According to Motorola's "General Information",
- * the DMT_DOUB1 bit is never set in this case, as it
- * should be.
- * If lockbar is set (as it is if we're here) and if
- * the write is not set, then it's the same as if DOUB1
- * was set...
- */
- if (!ISSET(dmtx, DMT_WRITE)) {
- if (x != 2) {
- /* RERUN xmem WITH DMD(x+1) */
- x++;
- dmdx = eframe[EF_DMD0 + x * 3];
- } else {
- /* RERUN xmem WITH DMD2 */
- }
-
- if (dmt_en_info[DMT_ENBITS(dmtx)].size ==
- DMT_WORD) {
- v = do_xmem_word(dmax, dmdx,
+ /* not pipeline #2 with a double */
+ if (dmtx & DMT_WRITE) {
+ switch (dmt_en_info[DMT_ENBITS(dmtx)].size) {
+ case DMT_BYTE:
+ DAE_DEBUG(
+ printf("[byte %x -> [%x(%c)]\n",
+ dmdx & 0xff, dmax,
+ ISSET(dmtx, DMT_DAS) ? 's' : 'u')
+ );
+ do_store_byte(dmax, dmdx,
dmtx & DMT_DAS);
- } else {
- v = do_xmem_byte(dmax, dmdx,
+ break;
+ case DMT_HALF:
+ DAE_DEBUG(
+ printf("[half %x -> [%x(%c)]\n",
+ dmdx & 0xffff, dmax,
+ ISSET(dmtx, DMT_DAS) ? 's' : 'u')
+ );
+ do_store_half(dmax, dmdx,
dmtx & DMT_DAS);
+ break;
+ case DMT_WORD:
+ DAE_DEBUG(
+ printf("[word %x -> [%x(%c)]\n",
+ dmdx, dmax,
+ ISSET(dmtx, DMT_DAS) ? 's' : 'u')
+ );
+ do_store_word(dmax, dmdx,
+ dmtx & DMT_DAS);
+ break;
}
- if (reg != 0)
- eframe[EF_R0 + reg] = v;
} else {
- if (x == 0) {
- if (reg != 0)
- eframe[EF_R0 + reg] = dmdx;
- eframe[EF_SFIP] = eframe[EF_SNIP];
- eframe[EF_SNIP] = eframe[EF_SXIP];
- eframe[EF_SXIP] = 0;
- /* xmem RERUN ON rte */
- eframe[EF_DMT0] = 0;
- return;
+ /* else it's a read */
+ switch (dmt_en_info[DMT_ENBITS(dmtx)].size) {
+ case DMT_BYTE:
+ v = do_load_byte(dmax, dmtx & DMT_DAS);
+ if (!ISSET(dmtx, DMT_SIGNED))
+ v &= 0x000000ff;
+ break;
+ case DMT_HALF:
+ v = do_load_half(dmax, dmtx & DMT_DAS);
+ if (!ISSET(dmtx, DMT_SIGNED))
+ v &= 0x0000ffff;
+ break;
+ case DMT_WORD:
+ v = do_load_word(dmax, dmtx & DMT_DAS);
+ break;
}
+ DAE_DEBUG(
+ if (reg == 0)
+ printf("[no write to r0 done]\n");
+ else
+ printf("[r%d <- %x]\n", reg, v);
+ );
+ if (reg != 0)
+ eframe->tf_r[reg] = v;
+ }
+ }
+ } else {
+ /* if lockbar is set... it's part of an XMEM */
+ /*
+ * According to Motorola's "General Information",
+ * the DMT_DOUB1 bit is never set in this case, as it
+ * should be.
+ * If lockbar is set (as it is if we're here) and if
+ * the write is not set, then it's the same as if DOUB1
+ * was set...
+ */
+ if (!ISSET(dmtx, DMT_WRITE)) {
+ if (x != 2) {
+ /* RERUN xmem WITH DMD(x+1) */
+ dmdx =
+ x == 0 ? eframe->tf_dmd1 : eframe->tf_dmd2;
+ } else {
+ /* RERUN xmem WITH DMD2 */
+ }
+
+ if (dmt_en_info[DMT_ENBITS(dmtx)].size == DMT_WORD) {
+ v = do_xmem_word(dmax, dmdx, dmtx & DMT_DAS);
+ } else {
+ v = do_xmem_byte(dmax, dmdx, dmtx & DMT_DAS);
+ }
+ if (reg != 0)
+ eframe->tf_r[reg] = v;
+ } else {
+ if (x == 0) {
+ if (reg != 0)
+ eframe->tf_r[reg] = dmdx;
+ eframe->tf_sfip = eframe->tf_snip;
+ eframe->tf_snip = eframe->tf_sxip;
+ eframe->tf_sxip = 0;
+ /* xmem RERUN ON rte */
+ eframe->tf_dmt0 = 0;
+ return;
}
}
}
- eframe[EF_DMT0] |= DMT_SKIP;
}
/*