aboutsummaryrefslogtreecommitdiff
path: root/sim
diff options
context:
space:
mode:
authorRon Unrau <runrau@cygnus>1998-07-31 22:02:12 +0000
committerRon Unrau <runrau@cygnus>1998-07-31 22:02:12 +0000
commitb8140a08bfadfd8cf95b83d19dd03c599f7f99f5 (patch)
tree047761e4e6a24c91f642077bbd2f30d0e44ef8e1 /sim
parent4f528afaf1f0dc867427fe92197e40deaa2a0696 (diff)
downloadfsf-binutils-gdb-b8140a08bfadfd8cf95b83d19dd03c599f7f99f5.zip
fsf-binutils-gdb-b8140a08bfadfd8cf95b83d19dd03c599f7f99f5.tar.gz
fsf-binutils-gdb-b8140a08bfadfd8cf95b83d19dd03c599f7f99f5.tar.bz2
* sim-main.h: shadow NUM_CORE_REGS from tm-txvu.h
* interp.c: use NUM_CORE_REGS * sky-gdb.c (set_fifo_breakpoints): use VIF interrupt bit for break * sky-pke.c (pke_issue): use interrupt bit for break points
Diffstat (limited to 'sim')
-rw-r--r--sim/mips/interp.c164
-rw-r--r--sim/mips/sim-main.h20
-rw-r--r--sim/mips/sky-pke.c17
3 files changed, 121 insertions, 80 deletions
diff --git a/sim/mips/interp.c b/sim/mips/interp.c
index 8b10b1f..4c9747d 100644
--- a/sim/mips/interp.c
+++ b/sim/mips/interp.c
@@ -46,9 +46,7 @@ code on the hardware.
#include "sky-libvpe.h"
#include "sky-pke.h"
#include "idecode.h"
-#include "support.h"
#include "sky-gdb.h"
-#undef SD
#endif
/* end-sanitize-sky */
@@ -179,6 +177,9 @@ static DECLARE_OPTION_HANDLER (mips_option_handler);
enum {
OPTION_DINERO_TRACE = OPTION_START,
OPTION_DINERO_FILE,
+ /* start-stanitize-branchbug4011 */
+ OPTION_BRANCH_BUG_4011,
+ /* end-stanitize-branchbug4011 */
OPTION_BOARD
};
@@ -194,6 +195,32 @@ mips_option_handler (sd, cpu, opt, arg, is_command)
int cpu_nr;
switch (opt)
{
+ /* start-sanitize-branchbug4011 */
+ case OPTION_BRANCH_BUG_4011:
+ {
+ for (cpu_nr = 0; cpu_nr < MAX_NR_PROCESSORS; cpu_nr++)
+ {
+ sim_cpu *cpu = STATE_CPU (sd, cpu_nr);
+ if (arg == NULL)
+ BRANCHBUG4011_OPTION = 1;
+ else if (strcmp (arg, "yes") == 0)
+ BRANCHBUG4011_OPTION = 1;
+ else if (strcmp (arg, "no") == 0)
+ BRANCHBUG4011_OPTION = 0;
+ else if (strcmp (arg, "on") == 0)
+ BRANCHBUG4011_OPTION = 1;
+ else if (strcmp (arg, "off") == 0)
+ BRANCHBUG4011_OPTION = 0;
+ else
+ {
+ fprintf (stderr, "Unrecognized check-4011-branch-bug option `%s'\n", arg);
+ return SIM_RC_FAIL;
+ }
+ }
+ return SIM_RC_OK;
+ }
+
+ /* end-sanitize-branchbug4011 */
case OPTION_DINERO_TRACE: /* ??? */
#if defined(TRACE)
/* Eventually the simTRACE flag could be treated as a toggle, to
@@ -266,6 +293,11 @@ static const OPTION mips_options[] =
{ {"dinero-trace", optional_argument, NULL, OPTION_DINERO_TRACE},
'\0', "on|off", "Enable dinero tracing",
mips_option_handler },
+ /* start-sanitize-branchbug4011 */
+ { {"check-4011-branch-bug", optional_argument, NULL, OPTION_BRANCH_BUG_4011},
+ '\0', "on|off", "Enable checking for 4011 branch bug",
+ mips_option_handler },
+ /* end-sanitize-branchbug4011 */
{ {"dinero-file", required_argument, NULL, OPTION_DINERO_FILE},
'\0', "FILE", "Write dinero trace to FILE",
mips_option_handler },
@@ -330,7 +362,6 @@ sim_open (kind, cb, abfd, argv)
SIM_ASSERT (STATE_MAGIC (sd) == SIM_MAGIC_NUMBER);
-
/* FIXME: watchpoints code shouldn't need this */
STATE_WATCHPOINTS (sd)->pc = &(PC);
STATE_WATCHPOINTS (sd)->sizeof_pc = sizeof (PC);
@@ -539,18 +570,18 @@ sim_open (kind, cb, abfd, argv)
#ifdef TARGET_SKY
/* Now the VU registers */
for( rn = 0; rn < NUM_VU_INTEGER_REGS; rn++ ) {
- cpu->register_widths[rn + NUM_R5900_REGS] = 16;
- cpu->register_widths[rn + NUM_R5900_REGS + NUM_VU_REGS] = 16;
+ cpu->register_widths[rn + NUM_CORE_REGS] = 16;
+ cpu->register_widths[rn + NUM_CORE_REGS + NUM_VU_REGS] = 16;
}
for( rn = NUM_VU_INTEGER_REGS; rn < NUM_VU_REGS; rn++ ) {
- cpu->register_widths[rn + NUM_R5900_REGS] = 32;
- cpu->register_widths[rn + NUM_R5900_REGS + NUM_VU_REGS] = 32;
+ cpu->register_widths[rn + NUM_CORE_REGS] = 32;
+ cpu->register_widths[rn + NUM_CORE_REGS + NUM_VU_REGS] = 32;
}
/* Finally the VIF registers */
for( rn = 2*NUM_VU_REGS; rn < 2*NUM_VU_REGS + 2*NUM_VIF_REGS; rn++ )
- cpu->register_widths[rn + NUM_R5900_REGS] = 32;
+ cpu->register_widths[rn + NUM_CORE_REGS] = 32;
cpu->cur_device = 0;
#endif
@@ -794,9 +825,9 @@ sim_store_register (sd,rn,memory,length)
/* start-sanitize-sky */
#ifdef TARGET_SKY
- if (rn >= NUM_R5900_REGS)
+ if (rn >= NUM_CORE_REGS)
{
- rn = rn - NUM_R5900_REGS;
+ rn = rn - NUM_CORE_REGS;
if( rn < NUM_VU_REGS )
{
@@ -969,9 +1000,9 @@ sim_fetch_register (sd,rn,memory,length)
/* start-sanitize-sky */
#ifdef TARGET_SKY
- if (rn >= NUM_R5900_REGS)
+ if (rn >= NUM_CORE_REGS)
{
- rn = rn - NUM_R5900_REGS;
+ rn = rn - NUM_CORE_REGS;
if (rn < NUM_VU_REGS)
{
@@ -3207,8 +3238,12 @@ decode_coproc (SIM_DESC sd,
case 0: /* standard CPU control and cache registers */
{
int code = ((instruction >> 21) & 0x1F);
+ int rt = ((instruction >> 16) & 0x1F);
+ int rd = ((instruction >> 11) & 0x1F);
+ int tail = instruction & 0x3ff;
/* R4000 Users Manual (second edition) lists the following CP0
instructions:
+ CODE><-RT><RD-><--TAIL--->
DMFC0 Doubleword Move From CP0 (VR4100 = 01000000001tttttddddd00000000000)
DMTC0 Doubleword Move To CP0 (VR4100 = 01000000101tttttddddd00000000000)
MFC0 word Move From CP0 (VR4100 = 01000000000tttttddddd00000000000)
@@ -3220,10 +3255,9 @@ decode_coproc (SIM_DESC sd,
CACHE Cache operation (VR4100 = 101111bbbbbpppppiiiiiiiiiiiiiiii)
ERET Exception return (VR4100 = 01000010000000000000000000011000)
*/
- if (((code == 0x00) || (code == 0x04)) && ((instruction & 0x7FF) == 0))
+ if (((code == 0x00) || (code == 0x04)) && tail == 0)
{
- int rt = ((instruction >> 16) & 0x1F);
- int rd = ((instruction >> 11) & 0x1F);
+ /* M[TF]C0 - 32 bit word */
switch (rd) /* NOTEs: Standard CP0 registers */
{
@@ -3312,12 +3346,36 @@ decode_coproc (SIM_DESC sd,
/* CPR[0,rd] = GPR[rt]; */
default:
if (code == 0x00)
+ GPR[rt] = (signed_word) (signed32) COP0_GPR[rd];
+ else
+ COP0_GPR[rd] = GPR[rt];
+#if 0
+ if (code == 0x00)
sim_io_printf(sd,"Warning: MFC0 %d,%d ignored (architecture specific)\n",rt,rd);
else
sim_io_printf(sd,"Warning: MTC0 %d,%d ignored (architecture specific)\n",rt,rd);
+#endif
}
}
- else if (code == 0x10 && (instruction & 0x3f) == 0x18)
+ /* start-sanitize-r5900 */
+ else if (((code == 0x00) || (code == 0x04)) && rd == 0x18 && tail > 0 && tail < NR_COP0_BP)
+ /* Break-point registers */
+ {
+ if (code == 0x00)
+ GPR[rt] = (signed_word) (signed32) COP0_BP[tail];
+ else
+ COP0_BP[tail] = GPR[rt];
+ }
+ else if (((code == 0x00) || (code == 0x04)) && rd == 0x19 && tail > 0 && tail < NR_COP0_P)
+ /* Performance registers */
+ {
+ if (code == 0x00)
+ GPR[rt] = (signed_word) (signed32) COP0_P[tail];
+ else
+ COP0_P[tail] = GPR[rt];
+ }
+ /* end-sanitize-r5900 */
+ else if (code == 0x10 && (tail & 0x3f) == 0x18)
{
/* ERET */
if (SR & status_ERL)
@@ -3333,7 +3391,7 @@ decode_coproc (SIM_DESC sd,
SR &= ~status_EXL;
}
}
- else if (code == 0x10 && (instruction & 0x3f) == 0x10)
+ else if (code == 0x10 && (tail & 0x3f) == 0x10)
{
/* RFE */
#ifdef SUBTARGET_R3900
@@ -3345,7 +3403,7 @@ decode_coproc (SIM_DESC sd,
/* TODO: CACHE register */
#endif /* SUBTARGET_R3900 */
}
- else if (code == 0x10 && (instruction & 0x3f) == 0x1F)
+ else if (code == 0x10 && (tail & 0x3f) == 0x1F)
{
/* DERET */
Debug &= ~Debug_DM;
@@ -3378,10 +3436,6 @@ decode_coproc (SIM_DESC sd,
int i_10_6 = (instruction >> 6) & 0x1f;
int i_5_0 = instruction & 0x03f;
int interlock = instruction & 0x01;
- /* setup for semantic.c-like actions below */
- typedef unsigned_4 instruction_word;
- int CIA = cia;
- int NIA = cia + 4;
handle = 1;
@@ -3392,33 +3446,8 @@ decode_coproc (SIM_DESC sd,
/* NOTREACHED */
}
-#define MY_INDEX itable_COPz_NORMAL
-#define MY_PREFIX COPz_NORMAL
-#define MY_NAME "COPz_NORMAL"
+ /* BC2T/BC2F/BC2TL/BC2FL handled in r5900.igen */
- /* classify & execute basic COP2 instructions */
- if(i_25_21 == 0x08 && i_20_16 == 0x00) /* BC2F */
- {
- address_word offset = EXTEND16(i_15_0) << 2;
- if(! vu0_busy()) DELAY_SLOT(cia + 4 + offset);
- }
- else if(i_25_21 == 0x08 && i_20_16==0x02) /* BC2FL */
- {
- address_word offset = EXTEND16(i_15_0) << 2;
- if(! vu0_busy()) DELAY_SLOT(cia + 4 + offset);
- else NULLIFY_NEXT_INSTRUCTION();
- }
- else if(i_25_21 == 0x08 && i_20_16 == 0x01) /* BC2T */
- {
- address_word offset = EXTEND16(i_15_0) << 2;
- if(vu0_busy()) DELAY_SLOT(cia + 4 + offset);
- }
- else if(i_25_21 == 0x08 && i_20_16 == 0x03) /* BC2TL */
- {
- address_word offset = EXTEND16(i_15_0) << 2;
- if(vu0_busy()) DELAY_SLOT(cia + 4 + offset);
- else NULLIFY_NEXT_INSTRUCTION();
- }
else if((i_25_21 == 0x02 && i_10_1 == 0x000) || /* CFC2 */
(i_25_21 == 0x01)) /* QMFC2 */
{
@@ -3430,17 +3459,19 @@ decode_coproc (SIM_DESC sd,
while(vu0_busy() && interlock)
vu0_issue(sd);
- /* perform VU register address */
+ /* perform VU register access */
if(i_25_21 == 0x01) /* QMFC2 */
{
- unsigned_16 xyzw;
+ unsigned_4 x,y,z,w;
+
/* one word at a time, argh! */
- read_vu_vec_reg(&(vu0_device.regs), id, 0, A4_16(& xyzw, 3));
- read_vu_vec_reg(&(vu0_device.regs), id, 1, A4_16(& xyzw, 2));
- read_vu_vec_reg(&(vu0_device.regs), id, 2, A4_16(& xyzw, 1));
- read_vu_vec_reg(&(vu0_device.regs), id, 3, A4_16(& xyzw, 0));
- GPR[rt] = T2H_8(* A8_16(& xyzw, 1));
- GPR1[rt] = T2H_8(* A8_16(& xyzw, 0));
+ read_vu_vec_reg(&(vu0_device.regs), id, 3, &w);
+ read_vu_vec_reg(&(vu0_device.regs), id, 2, &z);
+ read_vu_vec_reg(&(vu0_device.regs), id, 1, &y);
+ read_vu_vec_reg(&(vu0_device.regs), id, 0, &x);
+
+ GPR[rt] = U8_4(T2H_4(y), T2H_4(x));
+ GPR1[rt] = U8_4(T2H_4(w), T2H_4(z));
}
else /* CFC2 */
{
@@ -3466,17 +3497,21 @@ decode_coproc (SIM_DESC sd,
vu0_issue(sd);
}
- /* perform VU register address */
+ /* perform VU register access */
if(i_25_21 == 0x05) /* QMTC2 */
{
- unsigned_16 xyzw = U16_8(GPR1[rt], GPR[rt]);
+ unsigned_4 x,y,z,w;
+
+ x = H2T_4(V4_8(GPR[rt], 1));
+ y = H2T_4(V4_8(GPR[rt], 0));
+ z = H2T_4(V4_8(GPR1[rt], 1));
+ w = H2T_4(V4_8(GPR1[rt], 0));
- xyzw = H2T_16(xyzw);
/* one word at a time, argh! */
- write_vu_vec_reg(&(vu0_device.regs), id, 0, A4_16(& xyzw, 3));
- write_vu_vec_reg(&(vu0_device.regs), id, 1, A4_16(& xyzw, 2));
- write_vu_vec_reg(&(vu0_device.regs), id, 2, A4_16(& xyzw, 1));
- write_vu_vec_reg(&(vu0_device.regs), id, 3, A4_16(& xyzw, 0));
+ write_vu_vec_reg(&(vu0_device.regs), id, 3, & w);
+ write_vu_vec_reg(&(vu0_device.regs), id, 2, & z);
+ write_vu_vec_reg(&(vu0_device.regs), id, 1, & y);
+ write_vu_vec_reg(&(vu0_device.regs), id, 0, & x);
}
else /* CTC2 */
{
@@ -3560,9 +3595,6 @@ decode_coproc (SIM_DESC sd,
/* NOTREACHED */
}
- /* cleanup for semantic.c-like actions above */
- PC = NIA;
-
#undef MY_INDEX
#undef MY_PREFIX
#undef MY_NAME
diff --git a/sim/mips/sim-main.h b/sim/mips/sim-main.h
index a664122..1f91594 100644
--- a/sim/mips/sim-main.h
+++ b/sim/mips/sim-main.h
@@ -568,10 +568,10 @@ struct _sim_cpu {
#define NUM_VIF_REGS 26
#define FIRST_VEC_REG 25
-#define NUM_R5900_REGS 128
+#define NUM_CORE_REGS 128
#undef NUM_REGS
-#define NUM_REGS (NUM_R5900_REGS + 2*(NUM_VU_REGS) + 2*(NUM_VIF_REGS))
+#define NUM_REGS (NUM_CORE_REGS + 2*(NUM_VU_REGS) + 2*(NUM_VIF_REGS))
#endif /* no tm-txvu.h */
#endif /* TARGET_SKY */
/* end-sanitize-sky */
@@ -1027,15 +1027,20 @@ char* pr_uword64 PARAMS ((uword64 addr));
#endif
void sky_sim_engine_halt PARAMS ((SIM_DESC sd, sim_cpu *last, sim_cia cia));
-#define SIM_ENGINE_HALT_HOOK(sd, last, cia) sky_sim_engine_halt(sd, last, cia);
+#define SIM_ENGINE_HALT_HOOK(sd, last, cia) sky_sim_engine_halt(sd, last, cia)
#ifdef SIM_ENGINE_RESTART_HOOK
#undef SIM_ENGINE_RESTART_HOOK
#endif
void sky_sim_engine_restart PARAMS ((SIM_DESC sd, sim_cpu *last, sim_cia cia));
-#define SIM_ENGINE_RESTART_HOOK(sd, L, pc) sky_sim_engine_restart(sd, L, pc);
+#define SIM_ENGINE_RESTART_HOOK(sd, L, pc) sky_sim_engine_restart(sd, L, pc)
+/* for resume/suspend modules */
+SIM_RC sky_sim_module_install PARAMS ((SIM_DESC sd));
+
+#define MODULE_LIST sky_sim_module_install,
+
#ifndef TM_TXVU_H /* In case GDB hasn't been configured yet */
enum txvu_cpu_context
{
@@ -1049,7 +1054,7 @@ enum txvu_cpu_context
};
/* memory segment for communication with GDB */
-#define GDB_COMM_AREA 0x21010000
+#define GDB_COMM_AREA 0x21010000 /* Random choice */
#define GDB_COMM_SIZE 0x4000
/* Memory address containing last device to execute */
@@ -1059,8 +1064,11 @@ enum txvu_cpu_context
#define FIFO_BPT_CNT (GDB_COMM_AREA + 4)
#define FIFO_BPT_TBL (GDB_COMM_AREA + 8)
+/* Each element of the breakpoint table is three four-byte integers. */
+#define BPT_ELEM_SZ 4*3
+
#define TXVU_VU_BRK_MASK 0x02 /* Breakpoint bit is #57 for VU insns */
-#define TXVU_VIF_BRK_MASK 0x0f /* Breakpoint opcode for VIF insns */
+#define TXVU_VIF_BRK_MASK 0x80 /* Use interrupt bit for VIF insns */
#endif /* !TM_TXVU_H */
#endif /* TARGET_SKY */
diff --git a/sim/mips/sky-pke.c b/sim/mips/sky-pke.c
index e89385f..9a2c151 100644
--- a/sim/mips/sky-pke.c
+++ b/sim/mips/sky-pke.c
@@ -594,7 +594,6 @@ pke_issue(SIM_DESC sd, struct pke_device* me)
/* store word in PKECODE register */
me->regs[PKE_REG_CODE][0] = fw;
-
/* 2 -- test go / no-go for PKE execution */
/* switch on STAT:PSS if PSS-pending and in idle state */
@@ -633,6 +632,15 @@ pke_issue(SIM_DESC sd, struct pke_device* me)
/* handle interrupts */
if(intr)
{
+ /* check to see if the interrupt bit is being used for a breakpoint */
+ if (is_vif_breakpoint ((fqw->source_address & ~15) | (me->qw_pc << 2)))
+ {
+ sim_cpu *cpu = STATE_CPU (sd, 0);
+ unsigned_4 pc_addr = (fqw->source_address & ~15) | (me->qw_pc << 2);
+
+ sim_engine_halt (sd, cpu, NULL, pc_addr, sim_stopped, SIM_SIGTRAP);
+ }
+
/* are we resuming an interrupt-stalled instruction? */
if(me->flags & PKE_FLAG_INT_NOLOOP)
{
@@ -735,13 +743,6 @@ pke_issue(SIM_DESC sd, struct pke_device* me)
pke_code_mpg(me, fw);
else if(me->pke_number == 1 && IS_PKE_CMD(cmd, MSKPATH3))
pke_code_mskpath3(me, fw);
- else if(cmd == TXVU_VIF_BRK_MASK)
- {
- sim_cpu *cpu = STATE_CPU (sd, 0);
- unsigned_4 pc_addr = (fqw->source_address & ~15) | (me->qw_pc << 2);
-
- sim_engine_halt (sd, cpu, NULL, pc_addr, sim_stopped, SIM_SIGTRAP);
- }
/* ... no other commands ... */
else
pke_code_error(me, fw);