* 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
This commit is contained in:
Ron Unrau 1998-07-31 22:02:12 +00:00
parent 4f528afaf1
commit b8140a08bf
3 changed files with 121 additions and 80 deletions

View File

@ -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 */
{
@ -3311,13 +3345,37 @@ decode_coproc (SIM_DESC sd,
GPR[rt] = 0xDEADC0DE; /* CPR[0,rd] */
/* 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

View File

@ -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 */

View File

@ -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);