Cog/Frameworks/lazyusf/lazyusf/r4300/pure_interp.c
2015-03-01 21:28:09 -08:00

746 lines
26 KiB
C

/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
* Mupen64plus - pure_interp.c *
* Mupen64Plus homepage: http://code.google.com/p/mupen64plus/ *
* Copyright (C) 2015 Nebuleon <nebuleon.fumika@gmail.com> *
* *
* This program is free software; you can redistribute it and/or modify *
* it under the terms of the GNU General Public License as published by *
* the Free Software Foundation; either version 2 of the License, or *
* (at your option) any later version. *
* *
* This program is distributed in the hope that it will be useful, *
* but WITHOUT ANY WARRANTY; without even the implied warranty of *
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
* GNU General Public License for more details. *
* *
* You should have received a copy of the GNU General Public License *
* along with this program; if not, write to the *
* Free Software Foundation, Inc., *
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. *
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
#include <stdint.h>
#include "usf/usf.h"
#include "api/m64p_types.h"
#include "api/callbacks.h"
#include "memory/memory.h"
#include "main/main.h"
#include "osal/preproc.h"
/* TLBWrite requires invalid_code and blocks from cached_interp.h, but only if
* (at run time) the active core is not the Pure Interpreter. */
#include "cached_interp.h"
#include "r4300.h"
#include "cp0.h"
#include "cp1.h"
#include "exception.h"
#include "interupt.h"
#include "tlb.h"
#include "usf/usf_internal.h"
static void InterpretOpcode(usf_state_t * state);
#define PCADDR state->interp_PC.addr
#define ADD_TO_PC(x) state->interp_PC.addr += x*4;
#define DECLARE_INSTRUCTION(name) static void name(usf_state_t * state, uint32_t op)
#define DECLARE_JUMP(name, destination, condition, link, likely, cop1) \
static void name(usf_state_t * state, uint32_t op) \
{ \
const int take_jump = (condition); \
const unsigned int jump_target = (destination); \
long long int *link_register = (link); \
if (cop1 && check_cop1_unusable(state)) return; \
if (link_register != &state->reg[0]) \
{ \
*link_register=state->interp_PC.addr + 8; \
sign_extended(*link_register); \
} \
if (!likely || take_jump) \
{ \
state->interp_PC.addr += 4; \
state->delay_slot=1; \
InterpretOpcode(state); \
update_count(state); \
state->delay_slot=0; \
if (take_jump && !state->skip_jump) \
{ \
state->interp_PC.addr = jump_target; \
} \
} \
else \
{ \
state->interp_PC.addr += 8; \
update_count(state); \
} \
state->last_addr = state->interp_PC.addr; \
if (state->next_interupt <= state->g_cp0_regs[CP0_COUNT_REG]) gen_interupt(state); \
} \
static void name##_IDLE(usf_state_t * state, uint32_t op) \
{ \
const int take_jump = (condition); \
int skip; \
if (cop1 && check_cop1_unusable(state)) return; \
if (take_jump) \
{ \
update_count(state); \
skip = state->next_interupt - state->g_cp0_regs[CP0_COUNT_REG]; \
if (skip > 3) state->g_cp0_regs[CP0_COUNT_REG] += (skip & 0xFFFFFFFC); \
else name(state, op); \
} \
else name(state, op); \
}
#define CHECK_MEMORY()
#define RD_OF(op) (((op) >> 11) & 0x1F)
#define RS_OF(op) (((op) >> 21) & 0x1F)
#define RT_OF(op) (((op) >> 16) & 0x1F)
#define SA_OF(op) (((op) >> 6) & 0x1F)
#define IMM16S_OF(op) ((int16_t) (op))
#define IMM16U_OF(op) ((uint16_t) (op))
#define FD_OF(op) (((op) >> 6) & 0x1F)
#define FS_OF(op) (((op) >> 11) & 0x1F)
#define FT_OF(op) (((op) >> 16) & 0x1F)
#define JUMP_OF(op) ((op) & UINT32_C(0x3FFFFFF))
/* Determines whether a relative jump in a 16-bit immediate goes back to the
* same instruction without doing any work in its delay slot. The jump is
* relative to the instruction in the delay slot, so 1 instruction backwards
* (-1) goes back to the jump. */
#define IS_RELATIVE_IDLE_LOOP(op, addr) \
(IMM16S_OF(op) == -1 && *fast_mem_access(state, (addr) + 4) == 0)
/* Determines whether an absolute jump in a 26-bit immediate goes back to the
* same instruction without doing any work in its delay slot. The jump is
* in the same 256 MiB segment as the delay slot, so if the jump instruction
* is at the last address in its segment, it does not jump back to itself. */
#define IS_ABSOLUTE_IDLE_LOOP(op, addr) \
(JUMP_OF(op) == ((addr) & UINT32_C(0x0FFFFFFF)) >> 2 \
&& ((addr) & UINT32_C(0x0FFFFFFF)) != UINT32_C(0x0FFFFFFC) \
&& *fast_mem_access(state, (addr) + 4) == 0)
#define sign_extended(a) a = (int64_t) ((int32_t) (a))
#define sign_extendedb(a) a = (int64_t) ((int8_t) (a))
#define sign_extendedh(a) a = (int64_t) ((int16_t) (a))
/* These macros are like those in macros.h, but they parse opcode fields. */
#define rrt state->reg[RT_OF(op)]
#define rrd state->reg[RD_OF(op)]
#define rfs FS_OF(op)
#define rrs state->reg[RS_OF(op)]
#define rsa SA_OF(op)
#define irt state->reg[RT_OF(op)]
#define ioffset IMM16S_OF(op)
#define iimmediate IMM16S_OF(op)
#define irs state->reg[RS_OF(op)]
#define ibase state->reg[RS_OF(op)]
#define jinst_index JUMP_OF(op)
#define lfbase RS_OF(op)
#define lfft FT_OF(op)
#define lfoffset IMM16S_OF(op)
#define cfft FT_OF(op)
#define cffs FS_OF(op)
#define cffd FD_OF(op)
// 32 bits macros
#ifndef M64P_BIG_ENDIAN
#define rrt32 *((int32_t*) &state->reg[RT_OF(op)])
#define rrd32 *((int32_t*) &state->reg[RD_OF(op)])
#define rrs32 *((int32_t*) &state->reg[RS_OF(op)])
#define irs32 *((int32_t*) &state->reg[RS_OF(op)])
#define irt32 *((int32_t*) &state->reg[RT_OF(op)])
#else
#define rrt32 *((int32_t*) &state->reg[RT_OF(op)] + 1)
#define rrd32 *((int32_t*) &state->reg[RD_OF(op)] + 1)
#define rrs32 *((int32_t*) &state->reg[RS_OF(op)] + 1)
#define irs32 *((int32_t*) &state->reg[RS_OF(op)] + 1)
#define irt32 *((int32_t*) &state->reg[RT_OF(op)] + 1)
#endif
// two functions are defined from the macros above but never used
// these prototype declarations will prevent a warning
#if defined(__GNUC__)
static void JR_IDLE(usf_state_t *, uint32_t) __attribute__((used));
static void JALR_IDLE(usf_state_t *, uint32_t) __attribute__((used));
#endif
#include "interpreter.def"
#include <stdio.h>
#include <inttypes.h>
#ifdef DEBUG_INFO
#include "debugger/dbg_decoder.h"
#include <string.h>
#endif
void InterpretOpcode(usf_state_t * state)
{
uint32_t op = *fast_mem_access(state, state->PC->addr);
#ifdef DEBUG_INFO
if (0)
{
char instr[256];
char arguments[256];
r4300_decode_op(op, instr, arguments, state->PC->addr);
fprintf(state->debug_log, "%08x: %-16s %s\n", state->PC->addr, instr, arguments);
}
#endif
switch ((op >> 26) & 0x3F) {
case 0: /* SPECIAL prefix */
switch (op & 0x3F) {
case 0: /* SPECIAL opcode 0: SLL */
if (RD_OF(op) != 0) SLL(state, op);
else NOP(state, 0);
break;
case 2: /* SPECIAL opcode 2: SRL */
if (RD_OF(op) != 0) SRL(state, op);
else NOP(state, 0);
break;
case 3: /* SPECIAL opcode 3: SRA */
if (RD_OF(op) != 0) SRA(state, op);
else NOP(state, 0);
break;
case 4: /* SPECIAL opcode 4: SLLV */
if (RD_OF(op) != 0) SLLV(state, op);
else NOP(state, 0);
break;
case 6: /* SPECIAL opcode 6: SRLV */
if (RD_OF(op) != 0) SRLV(state, op);
else NOP(state, 0);
break;
case 7: /* SPECIAL opcode 7: SRAV */
if (RD_OF(op) != 0) SRAV(state, op);
else NOP(state, 0);
break;
case 8: JR(state, op); break;
case 9: /* SPECIAL opcode 9: JALR */
/* Note: This can omit the check for Rd == 0 because the JALR
* function checks for link_register != &reg[0]. If you're
* using this as a reference for a JIT, do check Rd == 0 in it. */
JALR(state, op);
break;
case 12: SYSCALL(state, op); break;
case 13: /* SPECIAL opcode 13: BREAK */
BREAK(state, op);
break;
case 15: SYNC(state, op); break;
case 16: /* SPECIAL opcode 16: MFHI */
if (RD_OF(op) != 0) MFHI(state, op);
else NOP(state, 0);
break;
case 17: MTHI(state, op); break;
case 18: /* SPECIAL opcode 18: MFLO */
if (RD_OF(op) != 0) MFLO(state, op);
else NOP(state, 0);
break;
case 19: MTLO(state, op); break;
case 20: /* SPECIAL opcode 20: DSLLV */
if (RD_OF(op) != 0) DSLLV(state, op);
else NOP(state, 0);
break;
case 22: /* SPECIAL opcode 22: DSRLV */
if (RD_OF(op) != 0) DSRLV(state, op);
else NOP(state, 0);
break;
case 23: /* SPECIAL opcode 23: DSRAV */
if (RD_OF(op) != 0) DSRAV(state, op);
else NOP(state, 0);
break;
case 24: MULT(state, op); break;
case 25: MULTU(state, op); break;
case 26: DIV(state, op); break;
case 27: DIVU(state, op); break;
case 28: DMULT(state, op); break;
case 29: DMULTU(state, op); break;
case 30: DDIV(state, op); break;
case 31: DDIVU(state, op); break;
case 32: /* SPECIAL opcode 32: ADD */
if (RD_OF(op) != 0) ADD(state, op);
else NOP(state, 0);
break;
case 33: /* SPECIAL opcode 33: ADDU */
if (RD_OF(op) != 0) ADDU(state, op);
else NOP(state, 0);
break;
case 34: /* SPECIAL opcode 34: SUB */
if (RD_OF(op) != 0) SUB(state, op);
else NOP(state, 0);
break;
case 35: /* SPECIAL opcode 35: SUBU */
if (RD_OF(op) != 0) SUBU(state, op);
else NOP(state, 0);
break;
case 36: /* SPECIAL opcode 36: AND */
if (RD_OF(op) != 0) AND(state, op);
else NOP(state, 0);
break;
case 37: /* SPECIAL opcode 37: OR */
if (RD_OF(op) != 0) OR(state, op);
else NOP(state, 0);
break;
case 38: /* SPECIAL opcode 38: XOR */
if (RD_OF(op) != 0) XOR(state, op);
else NOP(state, 0);
break;
case 39: /* SPECIAL opcode 39: NOR */
if (RD_OF(op) != 0) NOR(state, op);
else NOP(state, 0);
break;
case 42: /* SPECIAL opcode 42: SLT */
if (RD_OF(op) != 0) SLT(state, op);
else NOP(state, 0);
break;
case 43: /* SPECIAL opcode 43: SLTU */
if (RD_OF(op) != 0) SLTU(state, op);
else NOP(state, 0);
break;
case 44: /* SPECIAL opcode 44: DADD */
if (RD_OF(op) != 0) DADD(state, op);
else NOP(state, 0);
break;
case 45: /* SPECIAL opcode 45: DADDU */
if (RD_OF(op) != 0) DADDU(state, op);
else NOP(state, 0);
break;
case 46: /* SPECIAL opcode 46: DSUB */
if (RD_OF(op) != 0) DSUB(state, op);
else NOP(state, 0);
break;
case 47: /* SPECIAL opcode 47: DSUBU */
if (RD_OF(op) != 0) DSUBU(state, op);
else NOP(state, 0);
break;
case 48: /* SPECIAL opcode 48: TGE (Not implemented) */
case 49: /* SPECIAL opcode 49: TGEU (Not implemented) */
case 50: /* SPECIAL opcode 50: TLT (Not implemented) */
case 51: /* SPECIAL opcode 51: TLTU (Not implemented) */
NI(state, op);
break;
case 52: TEQ(state, op); break;
case 54: /* SPECIAL opcode 54: TNE (Not implemented) */
NI(state, op);
break;
case 56: /* SPECIAL opcode 56: DSLL */
if (RD_OF(op) != 0) DSLL(state, op);
else NOP(state, 0);
break;
case 58: /* SPECIAL opcode 58: DSRL */
if (RD_OF(op) != 0) DSRL(state, op);
else NOP(state, 0);
break;
case 59: /* SPECIAL opcode 59: DSRA */
if (RD_OF(op) != 0) DSRA(state, op);
else NOP(state, 0);
break;
case 60: /* SPECIAL opcode 60: DSLL32 */
if (RD_OF(op) != 0) DSLL32(state, op);
else NOP(state, 0);
break;
case 62: /* SPECIAL opcode 62: DSRL32 */
if (RD_OF(op) != 0) DSRL32(state, op);
else NOP(state, 0);
break;
case 63: /* SPECIAL opcode 63: DSRA32 */
if (RD_OF(op) != 0) DSRA32(state, op);
else NOP(state, 0);
break;
default: /* SPECIAL opcodes 1, 5, 10, 11, 14, 21, 40, 41, 53, 55, 57,
61: Reserved Instructions */
RESERVED(state, op);
break;
} /* switch (op & 0x3F) for the SPECIAL prefix */
break;
case 1: /* REGIMM prefix */
switch ((op >> 16) & 0x1F) {
case 0: /* REGIMM opcode 0: BLTZ */
if (IS_RELATIVE_IDLE_LOOP(op, state->PC->addr)) BLTZ_IDLE(state, op);
else BLTZ(state, op);
break;
case 1: /* REGIMM opcode 1: BGEZ */
if (IS_RELATIVE_IDLE_LOOP(op, state->PC->addr)) BGEZ_IDLE(state, op);
else BGEZ(state, op);
break;
case 2: /* REGIMM opcode 2: BLTZL */
if (IS_RELATIVE_IDLE_LOOP(op, state->PC->addr)) BLTZL_IDLE(state, op);
else BLTZL(state, op);
break;
case 3: /* REGIMM opcode 3: BGEZL */
if (IS_RELATIVE_IDLE_LOOP(op, state->PC->addr)) BGEZL_IDLE(state, op);
else BGEZL(state, op);
break;
case 8: /* REGIMM opcode 8: TGEI (Not implemented) */
case 9: /* REGIMM opcode 9: TGEIU (Not implemented) */
case 10: /* REGIMM opcode 10: TLTI (Not implemented) */
case 11: /* REGIMM opcode 11: TLTIU (Not implemented) */
case 12: /* REGIMM opcode 12: TEQI (Not implemented) */
case 14: /* REGIMM opcode 14: TNEI (Not implemented) */
NI(state, op);
break;
case 16: /* REGIMM opcode 16: BLTZAL */
if (IS_RELATIVE_IDLE_LOOP(op, state->PC->addr)) BLTZAL_IDLE(state, op);
else BLTZAL(state, op);
break;
case 17: /* REGIMM opcode 17: BGEZAL */
if (IS_RELATIVE_IDLE_LOOP(op, state->PC->addr)) BGEZAL_IDLE(state, op);
else BGEZAL(state, op);
break;
case 18: /* REGIMM opcode 18: BLTZALL */
if (IS_RELATIVE_IDLE_LOOP(op, state->PC->addr)) BLTZALL_IDLE(state, op);
else BLTZALL(state, op);
break;
case 19: /* REGIMM opcode 19: BGEZALL */
if (IS_RELATIVE_IDLE_LOOP(op, state->PC->addr)) BGEZALL_IDLE(state, op);
else BGEZALL(state, op);
break;
default: /* REGIMM opcodes 4..7, 13, 15, 20..31:
Reserved Instructions */
RESERVED(state, op);
break;
} /* switch ((op >> 16) & 0x1F) for the REGIMM prefix */
break;
case 2: /* Major opcode 2: J */
if (IS_ABSOLUTE_IDLE_LOOP(op, state->PC->addr)) J_IDLE(state, op);
else J(state, op);
break;
case 3: /* Major opcode 3: JAL */
if (IS_ABSOLUTE_IDLE_LOOP(op, state->PC->addr)) JAL_IDLE(state, op);
else JAL(state, op);
break;
case 4: /* Major opcode 4: BEQ */
if (IS_RELATIVE_IDLE_LOOP(op, state->PC->addr)) BEQ_IDLE(state, op);
else BEQ(state, op);
break;
case 5: /* Major opcode 5: BNE */
if (IS_RELATIVE_IDLE_LOOP(op, state->PC->addr)) BNE_IDLE(state, op);
else BNE(state, op);
break;
case 6: /* Major opcode 6: BLEZ */
if (IS_RELATIVE_IDLE_LOOP(op, state->PC->addr)) BLEZ_IDLE(state, op);
else BLEZ(state, op);
break;
case 7: /* Major opcode 7: BGTZ */
if (IS_RELATIVE_IDLE_LOOP(op, state->PC->addr)) BGTZ_IDLE(state, op);
else BGTZ(state, op);
break;
case 8: /* Major opcode 8: ADDI */
if (RT_OF(op) != 0) ADDI(state, op);
else NOP(state, 0);
break;
case 9: /* Major opcode 9: ADDIU */
if (RT_OF(op) != 0) ADDIU(state, op);
else NOP(state, 0);
break;
case 10: /* Major opcode 10: SLTI */
if (RT_OF(op) != 0) SLTI(state, op);
else NOP(state, 0);
break;
case 11: /* Major opcode 11: SLTIU */
if (RT_OF(op) != 0) SLTIU(state, op);
else NOP(state, 0);
break;
case 12: /* Major opcode 12: ANDI */
if (RT_OF(op) != 0) ANDI(state, op);
else NOP(state, 0);
break;
case 13: /* Major opcode 13: ORI */
if (RT_OF(op) != 0) ORI(state, op);
else NOP(state, 0);
break;
case 14: /* Major opcode 14: XORI */
if (RT_OF(op) != 0) XORI(state, op);
else NOP(state, 0);
break;
case 15: /* Major opcode 15: LUI */
if (RT_OF(op) != 0) LUI(state, op);
else NOP(state, 0);
break;
case 16: /* Coprocessor 0 prefix */
switch ((op >> 21) & 0x1F) {
case 0: /* Coprocessor 0 opcode 0: MFC0 */
if (RT_OF(op) != 0) MFC0(state, op);
else NOP(state, 0);
break;
case 4: MTC0(state, op); break;
case 16: /* Coprocessor 0 opcode 16: TLB */
switch (op & 0x3F) {
case 1: TLBR(state, op); break;
case 2: TLBWI(state, op); break;
case 6: TLBWR(state, op); break;
case 8: TLBP(state, op); break;
case 24: ERET(state, op); break;
default: /* TLB sub-opcodes 0, 3..5, 7, 9..23, 25..63:
Reserved Instructions */
RESERVED(state, op);
break;
} /* switch (op & 0x3F) for Coprocessor 0 TLB opcodes */
break;
default: /* Coprocessor 0 opcodes 1..3, 4..15, 17..31:
Reserved Instructions */
RESERVED(state, op);
break;
} /* switch ((op >> 21) & 0x1F) for the Coprocessor 0 prefix */
break;
case 17: /* Coprocessor 1 prefix */
switch ((op >> 21) & 0x1F) {
case 0: /* Coprocessor 1 opcode 0: MFC1 */
if (RT_OF(op) != 0) MFC1(state, op);
else NOP(state, 0);
break;
case 1: /* Coprocessor 1 opcode 1: DMFC1 */
if (RT_OF(op) != 0) DMFC1(state, op);
else NOP(state, 0);
break;
case 2: /* Coprocessor 1 opcode 2: CFC1 */
if (RT_OF(op) != 0) CFC1(state, op);
else NOP(state, 0);
break;
case 4: MTC1(state, op); break;
case 5: DMTC1(state, op); break;
case 6: CTC1(state, op); break;
case 8: /* Coprocessor 1 opcode 8: Branch on C1 condition... */
switch ((op >> 16) & 0x3) {
case 0: /* opcode 0: BC1F */
if (IS_RELATIVE_IDLE_LOOP(op, state->PC->addr)) BC1F_IDLE(state, op);
else BC1F(state, op);
break;
case 1: /* opcode 1: BC1T */
if (IS_RELATIVE_IDLE_LOOP(op, state->PC->addr)) BC1T_IDLE(state, op);
else BC1T(state, op);
break;
case 2: /* opcode 2: BC1FL */
if (IS_RELATIVE_IDLE_LOOP(op, state->PC->addr)) BC1FL_IDLE(state, op);
else BC1FL(state, op);
break;
case 3: /* opcode 3: BC1TL */
if (IS_RELATIVE_IDLE_LOOP(op, state->PC->addr)) BC1TL_IDLE(state, op);
else BC1TL(state, op);
break;
} /* switch ((op >> 16) & 0x3) for branches on C1 condition */
break;
case 16: /* Coprocessor 1 S-format opcodes */
switch (op & 0x3F) {
case 0: ADD_S(state, op); break;
case 1: SUB_S(state, op); break;
case 2: MUL_S(state, op); break;
case 3: DIV_S(state, op); break;
case 4: SQRT_S(state, op); break;
case 5: ABS_S(state, op); break;
case 6: MOV_S(state, op); break;
case 7: NEG_S(state, op); break;
case 8: ROUND_L_S(state, op); break;
case 9: TRUNC_L_S(state, op); break;
case 10: CEIL_L_S(state, op); break;
case 11: FLOOR_L_S(state, op); break;
case 12: ROUND_W_S(state, op); break;
case 13: TRUNC_W_S(state, op); break;
case 14: CEIL_W_S(state, op); break;
case 15: FLOOR_W_S(state, op); break;
case 33: CVT_D_S(state, op); break;
case 36: CVT_W_S(state, op); break;
case 37: CVT_L_S(state, op); break;
case 48: C_F_S(state, op); break;
case 49: C_UN_S(state, op); break;
case 50: C_EQ_S(state, op); break;
case 51: C_UEQ_S(state, op); break;
case 52: C_OLT_S(state, op); break;
case 53: C_ULT_S(state, op); break;
case 54: C_OLE_S(state, op); break;
case 55: C_ULE_S(state, op); break;
case 56: C_SF_S(state, op); break;
case 57: C_NGLE_S(state, op); break;
case 58: C_SEQ_S(state, op); break;
case 59: C_NGL_S(state, op); break;
case 60: C_LT_S(state, op); break;
case 61: C_NGE_S(state, op); break;
case 62: C_LE_S(state, op); break;
case 63: C_NGT_S(state, op); break;
default: /* Coprocessor 1 S-format opcodes 16..32, 34..35, 38..47:
Reserved Instructions */
RESERVED(state, op);
break;
} /* switch (op & 0x3F) for Coprocessor 1 S-format opcodes */
break;
case 17: /* Coprocessor 1 D-format opcodes */
switch (op & 0x3F) {
case 0: ADD_D(state, op); break;
case 1: SUB_D(state, op); break;
case 2: MUL_D(state, op); break;
case 3: DIV_D(state, op); break;
case 4: SQRT_D(state, op); break;
case 5: ABS_D(state, op); break;
case 6: MOV_D(state, op); break;
case 7: NEG_D(state, op); break;
case 8: ROUND_L_D(state, op); break;
case 9: TRUNC_L_D(state, op); break;
case 10: CEIL_L_D(state, op); break;
case 11: FLOOR_L_D(state, op); break;
case 12: ROUND_W_D(state, op); break;
case 13: TRUNC_W_D(state, op); break;
case 14: CEIL_W_D(state, op); break;
case 15: FLOOR_W_D(state, op); break;
case 32: CVT_S_D(state, op); break;
case 36: CVT_W_D(state, op); break;
case 37: CVT_L_D(state, op); break;
case 48: C_F_D(state, op); break;
case 49: C_UN_D(state, op); break;
case 50: C_EQ_D(state, op); break;
case 51: C_UEQ_D(state, op); break;
case 52: C_OLT_D(state, op); break;
case 53: C_ULT_D(state, op); break;
case 54: C_OLE_D(state, op); break;
case 55: C_ULE_D(state, op); break;
case 56: C_SF_D(state, op); break;
case 57: C_NGLE_D(state, op); break;
case 58: C_SEQ_D(state, op); break;
case 59: C_NGL_D(state, op); break;
case 60: C_LT_D(state, op); break;
case 61: C_NGE_D(state, op); break;
case 62: C_LE_D(state, op); break;
case 63: C_NGT_D(state, op); break;
default: /* Coprocessor 1 D-format opcodes 16..31, 33..35, 38..47:
Reserved Instructions */
RESERVED(state, op);
break;
} /* switch (op & 0x3F) for Coprocessor 1 D-format opcodes */
break;
case 20: /* Coprocessor 1 W-format opcodes */
switch (op & 0x3F) {
case 32: CVT_S_W(state, op); break;
case 33: CVT_D_W(state, op); break;
default: /* Coprocessor 1 W-format opcodes 0..31, 34..63:
Reserved Instructions */
RESERVED(state, op);
break;
}
break;
case 21: /* Coprocessor 1 L-format opcodes */
switch (op & 0x3F) {
case 32: CVT_S_L(state, op); break;
case 33: CVT_D_L(state, op); break;
default: /* Coprocessor 1 L-format opcodes 0..31, 34..63:
Reserved Instructions */
RESERVED(state, op);
break;
}
break;
default: /* Coprocessor 1 opcodes 3, 7, 9..15, 18..19, 22..31:
Reserved Instructions */
RESERVED(state, op);
break;
} /* switch ((op >> 21) & 0x1F) for the Coprocessor 1 prefix */
break;
case 20: /* Major opcode 20: BEQL */
if (IS_RELATIVE_IDLE_LOOP(op, state->PC->addr)) BEQL_IDLE(state, op);
else BEQL(state, op);
break;
case 21: /* Major opcode 21: BNEL */
if (IS_RELATIVE_IDLE_LOOP(op, state->PC->addr)) BNEL_IDLE(state, op);
else BNEL(state, op);
break;
case 22: /* Major opcode 22: BLEZL */
if (IS_RELATIVE_IDLE_LOOP(op, state->PC->addr)) BLEZL_IDLE(state, op);
else BLEZL(state, op);
break;
case 23: /* Major opcode 23: BGTZL */
if (IS_RELATIVE_IDLE_LOOP(op, state->PC->addr)) BGTZL_IDLE(state, op);
else BGTZL(state, op);
break;
case 24: /* Major opcode 24: DADDI */
if (RT_OF(op) != 0) DADDI(state, op);
else NOP(state, 0);
break;
case 25: /* Major opcode 25: DADDIU */
if (RT_OF(op) != 0) DADDIU(state, op);
else NOP(state, 0);
break;
case 26: /* Major opcode 26: LDL */
if (RT_OF(op) != 0) LDL(state, op);
else NOP(state, 0);
break;
case 27: /* Major opcode 27: LDR */
if (RT_OF(op) != 0) LDR(state, op);
else NOP(state, 0);
break;
case 32: /* Major opcode 32: LB */
if (RT_OF(op) != 0) LB(state, op);
else NOP(state, 0);
break;
case 33: /* Major opcode 33: LH */
if (RT_OF(op) != 0) LH(state, op);
else NOP(state, 0);
break;
case 34: /* Major opcode 34: LWL */
if (RT_OF(op) != 0) LWL(state, op);
else NOP(state, 0);
break;
case 35: /* Major opcode 35: LW */
if (RT_OF(op) != 0) LW(state, op);
else NOP(state, 0);
break;
case 36: /* Major opcode 36: LBU */
if (RT_OF(op) != 0) LBU(state, op);
else NOP(state, 0);
break;
case 37: /* Major opcode 37: LHU */
if (RT_OF(op) != 0) LHU(state, op);
else NOP(state, 0);
break;
case 38: /* Major opcode 38: LWR */
if (RT_OF(op) != 0) LWR(state, op);
else NOP(state, 0);
break;
case 39: /* Major opcode 39: LWU */
if (RT_OF(op) != 0) LWU(state, op);
else NOP(state, 0);
break;
case 40: SB(state, op); break;
case 41: SH(state, op); break;
case 42: SWL(state, op); break;
case 43: SW(state, op); break;
case 44: SDL(state, op); break;
case 45: SDR(state, op); break;
case 46: SWR(state, op); break;
case 47: CACHE(state, op); break;
case 48: /* Major opcode 48: LL */
if (RT_OF(op) != 0) LL(state, op);
else NOP(state, 0);
break;
case 49: LWC1(state, op); break;
case 52: /* Major opcode 52: LLD (Not implemented) */
NI(state, op);
break;
case 53: LDC1(state, op); break;
case 55: /* Major opcode 55: LD */
if (RT_OF(op) != 0) LD(state, op);
else NOP(state, 0);
break;
case 56: /* Major opcode 56: SC */
if (RT_OF(op) != 0) SC(state, op);
else NOP(state, 0);
break;
case 57: SWC1(state, op); break;
case 60: /* Major opcode 60: SCD (Not implemented) */
NI(state, op);
break;
case 61: SDC1(state, op); break;
case 63: SD(state, op); break;
default: /* Major opcodes 18..19, 28..31, 50..51, 54, 58..59, 62:
Reserved Instructions */
RESERVED(state, op);
break;
} /* switch ((op >> 26) & 0x3F) */
}
void pure_interpreter(usf_state_t * state)
{
state->stop = 0;
state->PC = &state->interp_PC;
while (!state->stop)
{
r4300_checkpoint(state);
InterpretOpcode(state);
}
}