flycast/core/hw/sh4/interpr/sh4_opcodes.cpp

2151 lines
32 KiB
C++
Raw Normal View History

2013-12-19 17:10:14 +00:00
/*
sh4 base core
most of it is (very) old
could use many cleanups, lets hope someone does them
*/
//All non fpu opcodes :)
#include "types.h"
#include "hw/pvr/pvr_mem.h"
#include "../sh4_interpreter.h"
#include "hw/sh4/sh4_mem.h"
#include "hw/sh4/sh4_mmr.h"
#include "../sh4_core.h"
#include "../modules/ccn.h"
#include "../sh4_interrupts.h"
#include "../modules/tmu.h"
#include "hw/gdrom/gdrom_if.h"
#include "hw/sh4/sh4_opcode.h"
void dofoo(sh4_opcode op)
{
r[op.n()]=gbr;
}
#define GetN(str) ((str>>8) & 0xf)
#define GetM(str) ((str>>4) & 0xf)
#define GetImm4(str) ((str>>0) & 0xf)
#define GetImm8(str) ((str>>0) & 0xff)
#define GetSImm8(str) ((s8)((str>>0) & 0xff))
#define GetImm12(str) ((str>>0) & 0xfff)
#define GetSImm12(str) (((s16)((GetImm12(str))<<4))>>4)
#define iNimp cpu_iNimp
#define iWarn cpu_iWarn
//Read Mem macros
#define ReadMemU32(to,addr) to=ReadMem32(addr)
#define ReadMemS32(to,addr) to=(s32)ReadMem32(addr)
#define ReadMemS16(to,addr) to=(u32)(s32)(s16)ReadMem16(addr)
#define ReadMemS8(to,addr) to=(u32)(s32)(s8)ReadMem8(addr)
2013-12-19 17:10:14 +00:00
//Base,offset format
#define ReadMemBOU32(to,addr,offset) ReadMemU32(to,addr+offset)
#define ReadMemBOS16(to,addr,offset) ReadMemS16(to,addr+offset)
#define ReadMemBOS8(to,addr,offset) ReadMemS8(to,addr+offset)
2013-12-19 17:10:14 +00:00
//Write Mem Macros
#define WriteMemU32(addr,data) WriteMem32(addr,(u32)data)
#define WriteMemU16(addr,data) WriteMem16(addr,(u16)data)
#define WriteMemU8(addr,data) WriteMem8(addr,(u8)data)
2013-12-19 17:10:14 +00:00
//Base,offset format
#define WriteMemBOU32(addr,offset,data) WriteMemU32(addr+offset,data)
#define WriteMemBOU16(addr,offset,data) WriteMemU16(addr+offset,data)
#define WriteMemBOU8(addr,offset,data) WriteMemU8(addr+offset,data)
2013-12-19 17:10:14 +00:00
// 0xxx
void cpu_iNimp(u32 op, const char* info)
{
2019-07-01 13:22:04 +00:00
ERROR_LOG(INTERPRETER, "Unimplemented opcode: %08X next_pc: %08X pr: %08X msg: %s", op, next_pc, pr, info);
2015-02-25 18:31:33 +00:00
//next_pc = pr; //debug hackfix: try to recover by returning from call
2013-12-19 17:10:14 +00:00
die("iNimp reached\n");
//sh4_cpu.Stop();
}
void cpu_iWarn(u32 op, const char* info)
{
2019-07-01 13:22:04 +00:00
INFO_LOG(INTERPRETER, "Check opcode : %X : %s @ %X", op, info, curr_pc);
2013-12-19 17:10:14 +00:00
}
//this file contains ALL register to register full moves
//
//stc GBR,<REG_N>
2013-12-19 17:10:14 +00:00
sh4op(i0000_nnnn_0001_0010)
{
u32 n = GetN(op);
r[n] = gbr;
}
2013-12-19 17:10:14 +00:00
//stc VBR,<REG_N>
2013-12-19 17:10:14 +00:00
sh4op(i0000_nnnn_0010_0010)
{
u32 n = GetN(op);
r[n] = vbr;
}
2013-12-19 17:10:14 +00:00
//stc SSR,<REG_N>
2013-12-19 17:10:14 +00:00
sh4op(i0000_nnnn_0011_0010)
{
u32 n = GetN(op);
r[n] = ssr;
}
2013-12-19 17:10:14 +00:00
//stc SGR,<REG_N>
2013-12-19 17:10:14 +00:00
sh4op(i0000_nnnn_0011_1010)
{
u32 n = GetN(op);
r[n] = sgr;
}
2013-12-19 17:10:14 +00:00
//stc SPC,<REG_N>
2013-12-19 17:10:14 +00:00
sh4op(i0000_nnnn_0100_0010)
{
u32 n = GetN(op);
r[n] = spc;
}
2013-12-19 17:10:14 +00:00
//stc RM_BANK,<REG_N>
2013-12-19 17:10:14 +00:00
sh4op(i0000_nnnn_1mmm_0010)
{
u32 n = GetN(op);
u32 m = GetM(op) & 0x7;
r[n] = r_bank[m];
}
2013-12-19 17:10:14 +00:00
//sts FPUL,<REG_N>
2013-12-19 17:10:14 +00:00
sh4op(i0000_nnnn_0101_1010)
{
u32 n = GetN(op);
r[n] = fpul;
}
2013-12-19 17:10:14 +00:00
//stc DBR,<REG_N>
2013-12-19 17:10:14 +00:00
sh4op(i0000_nnnn_1111_1010)
{
u32 n = GetN(op);
r[n] = dbr;
}
//sts MACH,<REG_N>
2013-12-19 17:10:14 +00:00
sh4op(i0000_nnnn_0000_1010)
{
u32 n = GetN(op);
r[n] = mac.h;
}
2013-12-19 17:10:14 +00:00
//sts MACL,<REG_N>
2013-12-19 17:10:14 +00:00
sh4op(i0000_nnnn_0001_1010)
{
u32 n = GetN(op);
r[n]=mac.l;
}
2013-12-19 17:10:14 +00:00
//sts PR,<REG_N>
2013-12-19 17:10:14 +00:00
sh4op(i0000_nnnn_0010_1010)
{
u32 n = GetN(op);
r[n] = pr;
}
2013-12-19 17:10:14 +00:00
//mov.b @(R0,<REG_M>),<REG_N>
2013-12-19 17:10:14 +00:00
sh4op(i0000_nnnn_mmmm_1100)
{
u32 n = GetN(op);
u32 m = GetM(op);
ReadMemBOS8(r[n],r[0],r[m]);
//r[n]= (u32)(s8)ReadMem8(r[0]+r[m]);
}
2013-12-19 17:10:14 +00:00
//mov.w @(R0,<REG_M>),<REG_N>
2013-12-19 17:10:14 +00:00
sh4op(i0000_nnnn_mmmm_1101)
{//ToDo : Check This [26/4/05]
u32 n = GetN(op);
u32 m = GetM(op);
ReadMemBOS16(r[n],r[0],r[m]);
//r[n] = (u32)(s16)ReadMem16(r[0] + r[m]);
}
2013-12-19 17:10:14 +00:00
//mov.l @(R0,<REG_M>),<REG_N>
2013-12-19 17:10:14 +00:00
sh4op(i0000_nnnn_mmmm_1110)
{
u32 n = GetN(op);
u32 m = GetM(op);
2013-12-19 17:10:14 +00:00
ReadMemBOU32(r[n],r[0],r[m]);
}
2013-12-19 17:10:14 +00:00
//mov.b <REG_M>,@(R0,<REG_N>)
2013-12-19 17:10:14 +00:00
sh4op(i0000_nnnn_mmmm_0100)
{
u32 n = GetN(op);
u32 m = GetM(op);
WriteMemBOU8(r[0],r[n], r[m]);
//WriteMem8(r[0] + r[n], (u8)r[m]);
}
//mov.w <REG_M>,@(R0,<REG_N>)
2013-12-19 17:10:14 +00:00
sh4op(i0000_nnnn_mmmm_0101)
{
u32 n = GetN(op);
u32 m = GetM(op);
WriteMemBOU16(r[0] , r[n], r[m]);
}
//mov.l <REG_M>,@(R0,<REG_N>)
2013-12-19 17:10:14 +00:00
sh4op(i0000_nnnn_mmmm_0110)
{
u32 n = GetN(op);
u32 m = GetM(op);
WriteMemBOU32(r[0], r[n], r[m]);
}
//
// 1xxx
//mov.l <REG_M>,@(<disp>,<REG_N>)
sh4op(i0001_nnnn_mmmm_iiii)
{
u32 n = GetN(op);
u32 m = GetM(op);
u32 disp = GetImm4(op);
WriteMemBOU32(r[n] , (disp <<2), r[m]);
}
//
// 2xxx
//mov.b <REG_M>,@<REG_N>
2013-12-19 17:10:14 +00:00
sh4op(i0010_nnnn_mmmm_0000)
{
u32 n = GetN(op);
u32 m = GetM(op);
WriteMemU8(r[n],r[m] );
}
// mov.w <REG_M>,@<REG_N>
2013-12-19 17:10:14 +00:00
sh4op(i0010_nnnn_mmmm_0001)
{
u32 n = GetN(op);
u32 m = GetM(op);
WriteMemU16(r[n],r[m]);
}
// mov.l <REG_M>,@<REG_N>
2013-12-19 17:10:14 +00:00
sh4op(i0010_nnnn_mmmm_0010)
{
u32 n = GetN(op);
u32 m = GetM(op);
WriteMemU32(r[n],r[m]);
}
// mov.b <REG_M>,@-<REG_N>
2013-12-19 17:10:14 +00:00
sh4op(i0010_nnnn_mmmm_0100)
{
u32 n = GetN(op);
u32 m = GetM(op);
u32 addr = r[n] - 1;
WriteMemBOU8(r[n], (u32)-1, r[m]);
r[n] = addr;
2013-12-19 17:10:14 +00:00
}
//mov.w <REG_M>,@-<REG_N>
2013-12-19 17:10:14 +00:00
sh4op(i0010_nnnn_mmmm_0101)
{
u32 n = GetN(op);
u32 m = GetM(op);
u32 addr = r[n] - 2;
WriteMemU16(addr, r[m]);
r[n] = addr;
2013-12-19 17:10:14 +00:00
}
//mov.l <REG_M>,@-<REG_N>
2013-12-19 17:10:14 +00:00
sh4op(i0010_nnnn_mmmm_0110)
{
u32 n = GetN(op);
u32 m = GetM(op);
u32 addr = r[n] - 4;
WriteMemU32(addr, r[m]);
r[n] = addr;
2013-12-19 17:10:14 +00:00
}
//
// 4xxx
//sts.l FPUL,@-<REG_N>
2013-12-19 17:10:14 +00:00
sh4op(i0100_nnnn_0101_0010)
{
u32 n = GetN(op);
u32 addr = r[n] - 4;
WriteMemU32(addr, fpul);
r[n] = addr;
2013-12-19 17:10:14 +00:00
}
//sts.l MACH,@-<REG_N>
2013-12-19 17:10:14 +00:00
sh4op(i0100_nnnn_0000_0010)
{
u32 n = GetN(op);
u32 addr = r[n] - 4;
WriteMemU32(addr, mac.h);
r[n] = addr;
2013-12-19 17:10:14 +00:00
}
//sts.l MACL,@-<REG_N>
2013-12-19 17:10:14 +00:00
sh4op(i0100_nnnn_0001_0010)
{
u32 n = GetN(op);
u32 addr = r[n] - 4;
WriteMemU32(addr, mac.l);
r[n] = addr;
2013-12-19 17:10:14 +00:00
}
//sts.l PR,@-<REG_N>
2013-12-19 17:10:14 +00:00
sh4op(i0100_nnnn_0010_0010)
{
u32 n = GetN(op);
u32 addr = r[n] - 4;
WriteMemU32(addr,pr);
r[n] = addr;
2013-12-19 17:10:14 +00:00
}
//sts.l DBR,@-<REG_N>
2013-12-19 17:10:14 +00:00
sh4op(i0100_nnnn_1111_0010)
{
u32 n = GetN(op);
u32 addr = r[n] - 4;
WriteMemU32(addr,dbr);
r[n] = addr;
2013-12-19 17:10:14 +00:00
}
//stc.l GBR,@-<REG_N>
2013-12-19 17:10:14 +00:00
sh4op(i0100_nnnn_0001_0011)
{
u32 n = GetN(op);
u32 addr = r[n] - 4;
WriteMemU32(addr, gbr);
r[n] = addr;
2013-12-19 17:10:14 +00:00
}
//stc.l VBR,@-<REG_N>
2013-12-19 17:10:14 +00:00
sh4op(i0100_nnnn_0010_0011)
{
u32 n = GetN(op);
u32 addr = r[n] - 4;
WriteMemU32(addr, vbr);
r[n] = addr;
2013-12-19 17:10:14 +00:00
}
//stc.l SSR,@-<REG_N>
2013-12-19 17:10:14 +00:00
sh4op(i0100_nnnn_0011_0011)
{
u32 n = GetN(op);
u32 addr = r[n] - 4;
WriteMemU32(addr, ssr);
r[n] = addr;
2013-12-19 17:10:14 +00:00
}
//stc.l SGR,@-<REG_N>
2013-12-19 17:10:14 +00:00
sh4op(i0100_nnnn_0011_0010)
{
u32 n = GetN(op);
u32 addr = r[n] - 4;
WriteMemU32(addr, sgr);
r[n] = addr;
2013-12-19 17:10:14 +00:00
}
//stc.l SPC,@-<REG_N>
2013-12-19 17:10:14 +00:00
sh4op(i0100_nnnn_0100_0011)
{
u32 n = GetN(op);
u32 addr = r[n] - 4;
WriteMemU32(addr, spc);
r[n] = addr;
2013-12-19 17:10:14 +00:00
}
//stc RM_BANK,@-<REG_N>
2013-12-19 17:10:14 +00:00
sh4op(i0100_nnnn_1mmm_0011)
{
u32 n = GetN(op);
u32 m = GetM(op) & 0x07;
u32 addr = r[n] - 4;
WriteMemU32(addr, r_bank[m]);
r[n] = addr;
2013-12-19 17:10:14 +00:00
}
//lds.l @<REG_N>+,MACH
2013-12-19 17:10:14 +00:00
sh4op(i0100_nnnn_0000_0110)
{
u32 n = GetN(op);
ReadMemU32(mac.h,r[n]);
2013-12-19 17:10:14 +00:00
r[n] += 4;
}
//lds.l @<REG_N>+,MACL
2013-12-19 17:10:14 +00:00
sh4op(i0100_nnnn_0001_0110)
{
u32 n = GetN(op);
ReadMemU32(mac.l,r[n]);
2013-12-19 17:10:14 +00:00
r[n] += 4;
}
//lds.l @<REG_N>+,PR
2013-12-19 17:10:14 +00:00
sh4op(i0100_nnnn_0010_0110)
{
u32 n = GetN(op);
ReadMemU32(pr,r[n]);
2013-12-19 17:10:14 +00:00
r[n] += 4;
}
//lds.l @<REG_N>+,FPUL
2013-12-19 17:10:14 +00:00
sh4op(i0100_nnnn_0101_0110)
{
u32 n = GetN(op);
2013-12-19 17:10:14 +00:00
ReadMemU32(fpul,r[n]);
r[n] += 4;
}
//lds.l @<REG_N>+,DBR
2013-12-19 17:10:14 +00:00
sh4op(i0100_nnnn_1111_0110)
{
u32 n = GetN(op);
2013-12-19 17:10:14 +00:00
ReadMemU32(dbr,r[n]);
r[n] += 4;
}
//ldc.l @<REG_N>+,GBR
2013-12-19 17:10:14 +00:00
sh4op(i0100_nnnn_0001_0111)
{
u32 n = GetN(op);
2013-12-19 17:10:14 +00:00
ReadMemU32(gbr,r[n]);
r[n] += 4;
}
//ldc.l @<REG_N>+,VBR
2013-12-19 17:10:14 +00:00
sh4op(i0100_nnnn_0010_0111)
{
u32 n = GetN(op);
2013-12-19 17:10:14 +00:00
ReadMemU32(vbr,r[n]);
r[n] += 4;
}
//ldc.l @<REG_N>+,SSR
2013-12-19 17:10:14 +00:00
sh4op(i0100_nnnn_0011_0111)
{
u32 n = GetN(op);
2013-12-19 17:10:14 +00:00
ReadMemU32(ssr,r[n]);
r[n] += 4;
}
//ldc.l @<REG_N>+,SGR
2013-12-19 17:10:14 +00:00
sh4op(i0100_nnnn_0011_0110)
{
u32 n = GetN(op);
2013-12-19 17:10:14 +00:00
ReadMemU32(sgr,r[n]);
r[n] += 4;
}
//ldc.l @<REG_N>+,SPC
2013-12-19 17:10:14 +00:00
sh4op(i0100_nnnn_0100_0111)
{
u32 n = GetN(op);
2013-12-19 17:10:14 +00:00
ReadMemU32(spc,r[n]);
r[n] += 4;
}
//ldc.l @<REG_N>+,RM_BANK
2013-12-19 17:10:14 +00:00
sh4op(i0100_nnnn_1mmm_0111)
{
u32 n = GetN(op);
u32 m = GetM(op) & 7;
2013-12-19 17:10:14 +00:00
ReadMemU32(r_bank[m],r[n]);
r[n] += 4;
}
//lds <REG_N>,MACH
2013-12-19 17:10:14 +00:00
sh4op(i0100_nnnn_0000_1010)
{
u32 n = GetN(op);
mac.h = r[n];
}
//lds <REG_N>,MACL
2013-12-19 17:10:14 +00:00
sh4op(i0100_nnnn_0001_1010)
{
u32 n = GetN(op);
mac.l = r[n];
}
//lds <REG_N>,PR
2013-12-19 17:10:14 +00:00
sh4op(i0100_nnnn_0010_1010)
{
u32 n = GetN(op);
pr = r[n];
}
//lds <REG_N>,FPUL
2013-12-19 17:10:14 +00:00
sh4op(i0100_nnnn_0101_1010)
{
u32 n = GetN(op);
fpul =r[n];
}
//ldc <REG_N>,DBR
2013-12-19 17:10:14 +00:00
sh4op(i0100_nnnn_1111_1010)
{
u32 n = GetN(op);
dbr = r[n];
}
//ldc <REG_N>,GBR
2013-12-19 17:10:14 +00:00
sh4op(i0100_nnnn_0001_1110)
{
u32 n = GetN(op);
gbr = r[n];
}
//ldc <REG_N>,VBR
2013-12-19 17:10:14 +00:00
sh4op(i0100_nnnn_0010_1110)
{
u32 n = GetN(op);
vbr = r[n];
}
//ldc <REG_N>,SSR
2013-12-19 17:10:14 +00:00
sh4op(i0100_nnnn_0011_1110)
{
u32 n = GetN(op);
ssr = r[n];
}
//ldc <REG_N>,SGR
2013-12-19 17:10:14 +00:00
sh4op(i0100_nnnn_0011_1010)
{
u32 n = GetN(op);
sgr = r[n];
}
//ldc <REG_N>,SPC
2013-12-19 17:10:14 +00:00
sh4op(i0100_nnnn_0100_1110)
{
u32 n = GetN(op);
spc = r[n];
}
//ldc <REG_N>,RM_BANK
2013-12-19 17:10:14 +00:00
sh4op(i0100_nnnn_1mmm_1110)
{
u32 n = GetN(op);
u32 m = GetM(op) & 7;
r_bank[m] = r[n];
}
//
// 5xxx
//mov.l @(<disp>,<REG_M>),<REG_N>
sh4op(i0101_nnnn_mmmm_iiii)
{
u32 n = GetN(op);
u32 m = GetM(op);
u32 disp = GetImm4(op) << 2;
2013-12-19 17:10:14 +00:00
ReadMemBOU32(r[n],r[m],disp);
}
//
// 6xxx
//mov.b @<REG_M>,<REG_N>
2013-12-19 17:10:14 +00:00
sh4op(i0110_nnnn_mmmm_0000)
{
u32 n = GetN(op);
u32 m = GetM(op);
2013-12-19 17:10:14 +00:00
ReadMemS8(r[n],r[m]);
}
2013-12-19 17:10:14 +00:00
//mov.w @<REG_M>,<REG_N>
2013-12-19 17:10:14 +00:00
sh4op(i0110_nnnn_mmmm_0001)
{
u32 n = GetN(op);
u32 m = GetM(op);
//r[n] = (u32)(s32)(s16)ReadMem16(r[m]);
ReadMemS16(r[n] ,r[m]);
}
2013-12-19 17:10:14 +00:00
//mov.l @<REG_M>,<REG_N>
2013-12-19 17:10:14 +00:00
sh4op(i0110_nnnn_mmmm_0010)
{
u32 n = GetN(op);
u32 m = GetM(op);
2013-12-19 17:10:14 +00:00
ReadMemU32(r[n],r[m]);
}
2013-12-19 17:10:14 +00:00
//mov <REG_M>,<REG_N>
2013-12-19 17:10:14 +00:00
sh4op(i0110_nnnn_mmmm_0011)
{
u32 n = GetN(op);
u32 m = GetM(op);
r[n] = r[m];
}
//mov.b @<REG_M>+,<REG_N>
2013-12-19 17:10:14 +00:00
sh4op(i0110_nnnn_mmmm_0100)
{
u32 n = GetN(op);
u32 m = GetM(op);
//r[n] = (u32)(s32)(s8)ReadMem8(r[m]);
ReadMemS8(r[n],r[m]);
if (n != m)
2013-12-19 17:10:14 +00:00
r[m] += 1;
}
2013-12-19 17:10:14 +00:00
//mov.w @<REG_M>+,<REG_N>
2013-12-19 17:10:14 +00:00
sh4op(i0110_nnnn_mmmm_0101)
{
u32 n = GetN(op);
u32 m = GetM(op);
//r[n] = (u32)(s16)(u16)ReadMem16(r[m]);
ReadMemS16(r[n],r[m]);
if (n != m)
2013-12-19 17:10:14 +00:00
r[m] += 2;
}
2013-12-19 17:10:14 +00:00
//mov.l @<REG_M>+,<REG_N>
2013-12-19 17:10:14 +00:00
sh4op(i0110_nnnn_mmmm_0110)
{
u32 n = GetN(op);
u32 m = GetM(op);
2013-12-19 17:10:14 +00:00
ReadMemU32(r[n],r[m]);
if (n != m)
r[m] += 4;
}
2013-12-19 17:10:14 +00:00
//
//8xxx
// mov.b R0,@(<disp>,<REG_M>)
2013-12-19 17:10:14 +00:00
sh4op(i1000_0000_mmmm_iiii)
{
u32 n = GetM(op);
u32 disp = GetImm4(op);
WriteMemBOU8(r[n],disp,r[0]);
}
// mov.w R0,@(<disp>,<REG_M>)
2013-12-19 17:10:14 +00:00
sh4op(i1000_0001_mmmm_iiii)
{
u32 disp = GetImm4(op);
u32 m = GetM(op);
WriteMemBOU16(r[m] , (disp << 1),r[0]);
}
// mov.b @(<disp>,<REG_M>),R0
2013-12-19 17:10:14 +00:00
sh4op(i1000_0100_mmmm_iiii)
{
u32 disp = GetImm4(op);
u32 m = GetM(op);
//r[0] = (u32)(s8)ReadMem8(r[m] + disp);
ReadMemBOS8(r[0] ,r[m] , disp);
}
// mov.w @(<disp>,<REG_M>),R0
2013-12-19 17:10:14 +00:00
sh4op(i1000_0101_mmmm_iiii)
{
u32 disp = GetImm4(op);
u32 m = GetM(op);
//r[0] = (u32)(s16)ReadMem16(r[m] + (disp << 1));
ReadMemBOS16(r[0],r[m] , (disp << 1));
}
//
// 9xxx
//mov.w @(<disp>,PC),<REG_N>
2013-12-19 17:10:14 +00:00
sh4op(i1001_nnnn_iiii_iiii)
{
u32 n = GetN(op);
u32 disp = (GetImm8(op));
//r[n]=(u32)(s32)(s16)ReadMem16((disp<<1) + pc + 4);
ReadMemS16(r[n],(disp<<1) + next_pc + 2);
}
//
// Cxxx
// mov.b R0,@(<disp>,GBR)
2013-12-19 17:10:14 +00:00
sh4op(i1100_0000_iiii_iiii)
{
u32 disp = GetImm8(op);
2013-12-19 17:10:14 +00:00
WriteMemBOU8(gbr, disp, r[0]);
}
// mov.w R0,@(<disp>,GBR)
2013-12-19 17:10:14 +00:00
sh4op(i1100_0001_iiii_iiii)
{
u32 disp = GetImm8(op);
//Write_Word(GBR+(disp<<1),R[0]);
WriteMemBOU16(gbr , (disp << 1), r[0]);
}
// mov.l R0,@(<disp>,GBR)
2013-12-19 17:10:14 +00:00
sh4op(i1100_0010_iiii_iiii)
{
u32 disp = (GetImm8(op));
//u32 source = (disp << 2) + gbr;
WriteMemBOU32(gbr,(disp << 2), r[0]);
}
// mov.b @(<disp>,GBR),R0
2013-12-19 17:10:14 +00:00
sh4op(i1100_0100_iiii_iiii)
{
u32 disp = GetImm8(op);
//r[0] = (u32)(s8)ReadMem8(gbr+disp);
ReadMemBOS8(r[0],gbr,disp);
}
// mov.w @(<disp>,GBR),R0
2013-12-19 17:10:14 +00:00
sh4op(i1100_0101_iiii_iiii)
{
u32 disp = GetImm8(op);
//r[0] = (u32)(s16)ReadMem16(gbr+(disp<<1) );
ReadMemBOS16(r[0],gbr,(disp<<1));
}
// mov.l @(<disp>,GBR),R0
2013-12-19 17:10:14 +00:00
sh4op(i1100_0110_iiii_iiii)
{
u32 disp = GetImm8(op);
2013-12-19 17:10:14 +00:00
ReadMemBOU32(r[0],gbr,(disp<<2));
}
// mova @(<disp>,PC),R0
2013-12-19 17:10:14 +00:00
sh4op(i1100_0111_iiii_iiii)
{
//u32 disp = (() << 2) + ((pc + 4) & 0xFFFFFFFC);
r[0] = ((next_pc+2)&0xFFFFFFFC)+(GetImm8(op)<<2);
}
//
// Dxxx
// mov.l @(<disp>,PC),<REG_N>
2013-12-19 17:10:14 +00:00
sh4op(i1101_nnnn_iiii_iiii)
{
u32 n = GetN(op);
u32 disp = (GetImm8(op));
2013-12-19 17:10:14 +00:00
ReadMemU32(r[n],(disp<<2) + ((next_pc+2) & 0xFFFFFFFC));
}
//
// Exxx
// mov #<imm>,<REG_N>
sh4op(i1110_nnnn_iiii_iiii)
{
u32 n = GetN(op);
r[n] = (u32)(s32)(s8)(GetSImm8(op));//(u32)(s8)= signextend8 :)
}
//movca.l R0, @<REG_N>
2013-12-19 17:10:14 +00:00
sh4op(i0000_nnnn_1100_0011)
{
u32 n = GetN(op);
WriteMemU32(r[n],r[0]);//at r[n],r[0]
//iWarn(op, "movca.l R0, @<REG_N>");
}
2013-12-19 17:10:14 +00:00
//clrmac
2013-12-19 17:10:14 +00:00
sh4op(i0000_0000_0010_1000)
{
mac.full=0;
}
2013-12-19 17:10:14 +00:00
//braf <REG_N>
sh4op(i0000_nnnn_0010_0011)
{
u32 n = GetN(op);
u32 newpc = r[n] + next_pc + 2;//
ExecuteDelayslot(); //WARN : r[n] can change here
next_pc = newpc;
}
//bsrf <REG_N>
sh4op(i0000_nnnn_0000_0011)
{
//TODO: Check pr setting vs real h/w
2013-12-19 17:10:14 +00:00
u32 n = GetN(op);
u32 newpc = r[n] + next_pc +2;
u32 newpr = next_pc + 2;
ExecuteDelayslot(); //WARN : pr and r[n] can change here
pr = newpr;
2013-12-19 17:10:14 +00:00
next_pc = newpc;
}
//rte
sh4op(i0000_0000_0010_1011)
{
u32 newpc = spc;
// FIXME In an RTE delay slot, status register (SR) bits are referenced as follows.
// In instruction access, the MD bit is used before modification, and in data access,
// the MD bit is accessed after modification.
// The other bits—S, T, M, Q, FD, BL, and RB—after modification are used for delay slot
// instruction execution. The STC and STC.L SR instructions access all SR bits after modification.
sh4_sr_SetFull(ssr);
2013-12-19 17:10:14 +00:00
ExecuteDelayslot_RTE();
next_pc = newpc;
if (UpdateSR())
2013-12-19 17:10:14 +00:00
{
UpdateINTC();
}
}
//rts
sh4op(i0000_0000_0000_1011)
{
u32 newpc=pr;
ExecuteDelayslot(); //WARN : pr can change here
2013-12-19 17:10:14 +00:00
next_pc=newpc;
}
u32 branch_target_s8(u32 op)
{
return GetSImm8(op)*2 + 2 + next_pc;
}
// bf <bdisp8>
sh4op(i1000_1011_iiii_iiii)
{
//TODO : Check Me [26/4/05] | Check DELAY SLOT [28/1/06]
2013-12-19 17:10:14 +00:00
if (sr.T==0)
{
//direct jump
next_pc = branch_target_s8(op);
}
}
// bf.s <bdisp8>
sh4op(i1000_1111_iiii_iiii)
{
if (sr.T==0)
{
//delay 1 instruction
u32 newpc=branch_target_s8(op);
ExecuteDelayslot();
next_pc = newpc;
}
}
// bt <bdisp8>
sh4op(i1000_1001_iiii_iiii)
{
if (sr.T != 0)
{
//direct jump
next_pc = branch_target_s8(op);
}
}
// bt.s <bdisp8>
sh4op(i1000_1101_iiii_iiii)
{
if (sr.T != 0)
{
//delay 1 instruction
u32 newpc=branch_target_s8(op);
ExecuteDelayslot();
next_pc = newpc;
}
}
u32 branch_target_s12(u32 op)
{
return GetSImm12(op)*2 + 2 + next_pc;
}
// bra <bdisp12>
sh4op(i1010_iiii_iiii_iiii)
{
u32 newpc = branch_target_s12(op);//(u32) (( ((s16)((GetImm12(op))<<4)) >>3) + pc + 4);//(s16<<4,>>4(-1*2))
ExecuteDelayslot();
next_pc=newpc;
}
// bsr <bdisp12>
sh4op(i1011_iiii_iiii_iiii)
{
//TODO: check pr vs real h/w
u32 newpr = next_pc + 2; //return after delayslot
2013-12-19 17:10:14 +00:00
u32 newpc = branch_target_s12(op);
ExecuteDelayslot();
pr = newpr;
2013-12-19 17:10:14 +00:00
next_pc=newpc;
}
// trapa #<imm>
sh4op(i1100_0011_iiii_iiii)
{
//printf("trapa 0x%X\n",(GetImm8(op) << 2));
CCN_TRA = (GetImm8(op) << 2);
2015-09-17 19:31:25 +00:00
Do_Exception(next_pc,0x160,0x100);
2013-12-19 17:10:14 +00:00
}
//jmp @<REG_N>
sh4op(i0100_nnnn_0010_1011)
{
u32 n = GetN(op);
u32 newpc=r[n];
ExecuteDelayslot(); //r[n] can change here
2013-12-19 17:10:14 +00:00
next_pc=newpc;
}
//jsr @<REG_N>
sh4op(i0100_nnnn_0000_1011)
{
u32 n = GetN(op);
//TODO: check pr vs real h/w
u32 newpr = next_pc + 2; //return after delayslot
2013-12-19 17:10:14 +00:00
u32 newpc= r[n];
ExecuteDelayslot(); //r[n]/pr can change here
pr = newpr;
2013-12-19 17:10:14 +00:00
next_pc=newpc;
}
//sleep
sh4op(i0000_0000_0001_1011)
{
//just wait for an Interrupt
int i=0,s=1;
while (!UpdateSystem_INTC())//448
{
if (i++>1000)
{
s=0;
break;
}
}
//if not Interrupted , we must rexecute the sleep
if (s==0)
next_pc-=2;// re execute sleep
}
// sub <REG_M>,<REG_N>
2013-12-19 17:10:14 +00:00
sh4op(i0011_nnnn_mmmm_1000)
{
u32 n = GetN(op);
u32 m = GetM(op);
//rn=(s32)r[n];
//rm=(s32)r[m];
r[n] -=r[m];
//r[n]=(u32)rn;
}
//add <REG_M>,<REG_N>
2013-12-19 17:10:14 +00:00
sh4op(i0011_nnnn_mmmm_1100)
{
u32 n = GetN(op);
u32 m = GetM(op);
r[n] +=r[m];
}
//
// 7xxx
//add #<imm>,<REG_N>
sh4op(i0111_nnnn_iiii_iiii)
{
u32 n = GetN(op);
s32 stmp1 = GetSImm8(op);
r[n] +=stmp1;
}
//Bitwise logical operations
//
//and <REG_M>,<REG_N>
2013-12-19 17:10:14 +00:00
sh4op(i0010_nnnn_mmmm_1001)
{
u32 n = GetN(op);
u32 m = GetM(op);
r[n] &= r[m];
}
//xor <REG_M>,<REG_N>
2013-12-19 17:10:14 +00:00
sh4op(i0010_nnnn_mmmm_1010)
{
u32 n = GetN(op);
u32 m = GetM(op);
r[n] ^= r[m];
}
//or <REG_M>,<REG_N>
2013-12-19 17:10:14 +00:00
sh4op(i0010_nnnn_mmmm_1011)
{
u32 n = GetN(op);
u32 m = GetM(op);
2013-12-19 17:10:14 +00:00
r[n] |= r[m];
}
//shll2 <REG_N>
2013-12-19 17:10:14 +00:00
sh4op(i0100_nnnn_0000_1000)
{
u32 n = GetN(op);
r[n] <<= 2;
}
//shll8 <REG_N>
2013-12-19 17:10:14 +00:00
sh4op(i0100_nnnn_0001_1000)
{//ToDo : Check This [26/4/05]
u32 n = GetN(op);
r[n] <<= 8;
}
//shll16 <REG_N>
2013-12-19 17:10:14 +00:00
sh4op(i0100_nnnn_0010_1000)
{//ToDo : Check This [26/4/05]
u32 n = GetN(op);
r[n] <<= 16;
}
//shlr2 <REG_N>
2013-12-19 17:10:14 +00:00
sh4op(i0100_nnnn_0000_1001)
{//ToDo : Check This [26/4/05]
u32 n = GetN(op);
r[n] >>= 2;
}
//shlr8 <REG_N>
2013-12-19 17:10:14 +00:00
sh4op(i0100_nnnn_0001_1001)
{
u32 n = GetN(op);
r[n] >>= 8;
}
//shlr16 <REG_N>
2013-12-19 17:10:14 +00:00
sh4op(i0100_nnnn_0010_1001)
{
u32 n = GetN(op);
r[n] >>= 16;
}
// and #<imm>,R0
2013-12-19 17:10:14 +00:00
sh4op(i1100_1001_iiii_iiii)
{//ToDo : Check This [26/4/05]
u32 imm = GetImm8(op);
r[0] &= imm;
}
// xor #<imm>,R0
2013-12-19 17:10:14 +00:00
sh4op(i1100_1010_iiii_iiii)
{
u32 imm = GetImm8(op);
r[0] ^= imm;
}
// or #<imm>,R0
2013-12-19 17:10:14 +00:00
sh4op(i1100_1011_iiii_iiii)
{//ToDo : Check This [26/4/05]
u32 imm = GetImm8(op);
r[0] |= imm;
}
2013-12-19 17:10:14 +00:00
//TODO : move it somewhere better
//nop
2013-12-19 17:10:14 +00:00
sh4op(i0000_0000_0000_1001)
{
//no operation xD XD .. i just love this opcode ..
//what ? you expected something fancy or smth ?
}
2013-12-19 17:10:14 +00:00
//************************ TLB/Cache ************************
//ldtlb
sh4op(i0000_0000_0011_1000)
{
2015-09-17 19:31:25 +00:00
//printf("ldtlb %d/%d\n",CCN_MMUCR.URC,CCN_MMUCR.URB);
2013-12-19 17:10:14 +00:00
UTLB[CCN_MMUCR.URC].Data=CCN_PTEL;
UTLB[CCN_MMUCR.URC].Address=CCN_PTEH;
UTLB[CCN_MMUCR.URC].Assistance=CCN_PTEA;
2013-12-19 17:10:14 +00:00
UTLB_Sync(CCN_MMUCR.URC);
}
//ocbi @<REG_N>
sh4op(i0000_nnnn_1001_0011)
{
u32 n = GetN(op);
//printf("ocbi @0x%08X \n",r[n]);
}
//ocbp @<REG_N>
sh4op(i0000_nnnn_1010_0011)
{
u32 n = GetN(op);
//printf("ocbp @0x%08X \n",r[n]);
}
//ocbwb @<REG_N>
sh4op(i0000_nnnn_1011_0011)
{
u32 n = GetN(op);
//printf("ocbwb @0x%08X \n",r[n]);
}
//pref @<REG_N>
template<bool mmu_on>
INLINE void DYNACALL do_sqw(u32 Dest)
{
//TODO : Check for enabled store queues ?
u32 Address;
//Translate the SQ addresses as needed
if (mmu_on)
{
if (!mmu_TranslateSQW(Dest, &Address))
return;
2013-12-19 17:10:14 +00:00
}
else
{
#if HOST_CPU ==CPU_X86
//sanity/optimisation check
verify(CCN_QACR_TR[0]==CCN_QACR_TR[1]);
#endif
u32 QACR = CCN_QACR_TR[0];
/*
//sq1 ? if so use QACR1
//(QACR1==QACR0 in all stuff i've tested)
if (Dest& 0x20)
QACR = CCN_QACR_TR[1];
*/
//QACR has already 0xE000_0000
Address= QACR+(Dest&~0x1f);
}
if (((Address >> 26) & 0x7) != 4)//Area 4
{
u8* sq=&sq_both[Dest& 0x20];
WriteMemBlock_nommu_sq(Address,(u32*)sq);
}
else
{
TAWriteSQ(Address,sq_both);
}
}
void DYNACALL do_sqw_mmu(u32 dst) { do_sqw<true>(dst); }
#if HOST_CPU != CPU_ARM && HOST_CPU != CPU_ARM64
//yes, this micro optimization makes a difference
extern "C" void DYNACALL do_sqw_nommu_area_3(u32 dst,u8* sqb)
{
2013-12-19 17:10:14 +00:00
u8* pmem=sqb+512+0x0C000000;
memcpy((u64*)&pmem[dst&(RAM_MASK-0x1F)],(u64*)&sqb[dst & 0x20],32);
}
#endif
extern "C" void DYNACALL do_sqw_nommu_area_3_nonvmem(u32 dst,u8* sqb)
{
u8* pmem = mem_b.data;
memcpy((u64*)&pmem[dst&(RAM_MASK-0x1F)],(u64*)&sqb[dst & 0x20],32);
}
2013-12-19 17:10:14 +00:00
void DYNACALL do_sqw_nommu_full(u32 dst, u8* sqb) { do_sqw<false>(dst); }
sh4op(i0000_nnnn_1000_0011)
{
u32 n = GetN(op);
u32 Dest = r[n];
if ((Dest>>26) == 0x38) //Store Queue
{
if (CCN_MMUCR.AT)
do_sqw<true>(Dest);
else
do_sqw<false>(Dest);
}
}
//************************ Set/Get T/S ************************
//sets
sh4op(i0000_0000_0101_1000)
{
sr.S = 1;
}
//clrs
sh4op(i0000_0000_0100_1000)
{
sr.S = 0;
}
//sett
sh4op(i0000_0000_0001_1000)
{
sr.T = 1;
}
//clrt
sh4op(i0000_0000_0000_1000)
{
sr.T = 0;
}
//movt <REG_N>
sh4op(i0000_nnnn_0010_1001)
{
u32 n = GetN(op);
r[n] = sr.T;
}
//************************ Reg Compares ************************
//cmp/pz <REG_N>
sh4op(i0100_nnnn_0001_0001)
{
u32 n = GetN(op);
if (((s32)r[n]) >= 0)
sr.T = 1;
else
sr.T = 0;
}
//cmp/pl <REG_N>
sh4op(i0100_nnnn_0001_0101)
{
u32 n = GetN(op);
if ((s32)r[n] > 0)
sr.T = 1;
else
sr.T = 0;
}
//cmp/eq #<imm>,R0
sh4op(i1000_1000_iiii_iiii)
{
u32 imm = (u32)(s32)(GetSImm8(op));
if (r[0] == imm)
sr.T =1;
else
sr.T =0;
}
//cmp/eq <REG_M>,<REG_N>
sh4op(i0011_nnnn_mmmm_0000)
{
u32 n = GetN(op);
u32 m = GetM(op);
if (r[m] == r[n])
sr.T = 1;
else
sr.T = 0;
}
//cmp/hs <REG_M>,<REG_N>
sh4op(i0011_nnnn_mmmm_0010)
{//ToDo : Check Me [26/4/05]
u32 n = GetN(op);
u32 m = GetM(op);
if (r[n] >= r[m])
sr.T=1;
else
sr.T=0;
}
//cmp/ge <REG_M>,<REG_N>
sh4op(i0011_nnnn_mmmm_0011)
{
u32 n = GetN(op);
u32 m = GetM(op);
if ((s32)r[n] >= (s32)r[m])
sr.T = 1;
else
sr.T = 0;
}
//cmp/hi <REG_M>,<REG_N>
sh4op(i0011_nnnn_mmmm_0110)
{
u32 n = GetN(op);
u32 m = GetM(op);
if (r[n] > r[m])
sr.T=1;
else
sr.T=0;
}
//cmp/gt <REG_M>,<REG_N>
sh4op(i0011_nnnn_mmmm_0111)
{
u32 n = GetN(op);
u32 m = GetM(op);
if (((s32)r[n]) > ((s32)r[m]))
sr.T = 1;
else
sr.T = 0;
}
//cmp/str <REG_M>,<REG_N>
sh4op(i0010_nnnn_mmmm_1100)
{
//T -> 1 if -any- bytes are equal
u32 n = GetN(op);
u32 m = GetM(op);
u32 temp;
u32 HH, HL, LH, LL;
temp = r[n] ^ r[m];
HH=(temp&0xFF000000)>>24;
HL=(temp&0x00FF0000)>>16;
LH=(temp&0x0000FF00)>>8;
LL=temp&0x000000FF;
HH=HH&&HL&&LH&&LL;
if (HH==0)
sr.T=1;
else
sr.T=0;
}
//tst #<imm>,R0
sh4op(i1100_1000_iiii_iiii)
{
u32 utmp1 = r[0] & GetImm8(op);
if (utmp1 == 0)
sr.T = 1;
else
sr.T = 0;
}
//tst <REG_M>,<REG_N>
sh4op(i0010_nnnn_mmmm_1000)
{//ToDo : Check This [26/4/05]
u32 n = GetN(op);
u32 m = GetM(op);
if ((r[n] & r[m])!=0)
sr.T=0;
else
sr.T=1;
}
//************************ mulls! ************************
//mulu.w <REG_M>,<REG_N>
sh4op(i0010_nnnn_mmmm_1110)
{
//check ++
2013-12-19 17:10:14 +00:00
u32 n = GetN(op);
u32 m = GetM(op);
mac.l=((u16)r[n])*
((u16)r[m]);
}
//muls.w <REG_M>,<REG_N>
sh4op(i0010_nnnn_mmmm_1111)
{
u32 n = GetN(op);
u32 m = GetM(op);
mac.l = (u32)(((s16)(u16)r[n]) * ((s16)(u16)r[m]));
}
//dmulu.l <REG_M>,<REG_N>
sh4op(i0011_nnnn_mmmm_0101)
{
u32 n = GetN(op);
u32 m = GetM(op);
mac.full = (u64)r[n] * (u64)r[m];
}
//dmuls.l <REG_M>,<REG_N>
sh4op(i0011_nnnn_mmmm_1101)
{
//check ++
2013-12-19 17:10:14 +00:00
u32 n = GetN(op);
u32 m = GetM(op);
mac.full = (s64)(s32)r[n] * (s64)(s32)r[m];
}
//mac.w @<REG_M>+,@<REG_N>+
sh4op(i0100_nnnn_mmmm_1111)
{
u32 n = GetN(op);
u32 m = GetM(op);
if (sr.S!=0)
{
die("mac.w @<REG_M>+,@<REG_N>+ : S=1");
2013-12-19 17:10:14 +00:00
}
else
{
s32 rm,rn;
rn = (s32)(s16)ReadMem16(r[n]);
//if (n==m)
//{
// r[n]+=2;
// r[m]+=2;
//}
rm = (s32)(s16)ReadMem16(r[m] + (n == m ? 2 : 0));
2013-12-19 17:10:14 +00:00
r[n]+=2;
2013-12-19 17:10:14 +00:00
r[m]+=2;
s32 mul=rm * rn;
mac.full+=(s64)mul;
}
}
//mac.l @<REG_M>+,@<REG_N>+
sh4op(i0000_nnnn_mmmm_1111)
{
u32 n = GetN(op);
u32 m = GetM(op);
s32 rm, rn;
verify(sr.S==0);
ReadMemS32(rm,r[m]);
ReadMemS32(rn,r[n] + (n == m ? 4 : 0));
2013-12-19 17:10:14 +00:00
r[m] += 4;
r[n] += 4;
mac.full += (s64)rm * (s64)rn;
2013-12-19 17:10:14 +00:00
//printf("%I64u %I64u | %d %d | %d %d\n",mac,mul,macl,mach,rm,rn);
}
//mul.l <REG_M>,<REG_N>
sh4op(i0000_nnnn_mmmm_0111)
{
u32 n = GetN(op);
u32 m = GetM(op);
mac.l = (u32)((((s32)r[n]) * ((s32)r[m])));
}
//************************ Div ! ************************
//div0u
sh4op(i0000_0000_0001_1001)
{
sr.Q = 0;
sr.M = 0;
sr.T = 0;
}
//div0s <REG_M>,<REG_N>
sh4op(i0010_nnnn_mmmm_0111)
{//ToDo : Check This [26/4/05]
u32 n = GetN(op);
u32 m = GetM(op);
//new implementation
sr.Q=r[n]>>31;
sr.M=r[m]>>31;
sr.T=sr.M^sr.Q;
return;
/*
if ((r[n] & 0x80000000)!=0)
sr.Q = 1;
else
sr.Q = 0;
if ((r[m] & 0x80000000)!=0)
sr.M = 1;
else
sr.M = 0;
if (sr.Q == sr.M)
sr.T = 0;
else
sr.T = 1;
*/
}
//div1 <REG_M>,<REG_N>
sh4op(i0011_nnnn_mmmm_0100)
{
u32 n=GetN(op);
u32 m=GetM(op);
2019-07-01 13:22:04 +00:00
u32 tmp0, tmp2;
2013-12-19 17:10:14 +00:00
unsigned char old_q, tmp1;
old_q = sr.Q;
sr.Q = (u8)((0x80000000 & r[n]) !=0);
r[n] <<= 1;
r[n] |= (unsigned long)sr.T;
tmp0 = r[n]; // this need only be done once here ..
2019-07-01 13:22:04 +00:00
// Old implementation
// tmp2 = r[m];
//
// if( 0 == old_q )
// {
// if( 0 == sr.M )
// {
// r[n] -= tmp2;
// tmp1 = (r[n]>tmp0);
// sr.Q = (sr.Q==0) ? tmp1 : (u8)(tmp1==0) ;
// }
// else
// {
// r[n] += tmp2;
// tmp1 =(r[n]<tmp0);
// sr.Q = (sr.Q==0) ? (u8)(tmp1==0) : tmp1 ;
// }
// }
// else
// {
// if( 0 == sr.M )
// {
// r[n] += tmp2;
// tmp1 =(r[n]<tmp0);
// sr.Q = (sr.Q==0) ? tmp1 : (u8)(tmp1==0) ;
// }
// else
// {
// r[n] -= tmp2;
// tmp1 =(r[n]>tmp0);
// sr.Q = (sr.Q==0) ? (u8)(tmp1==0) : tmp1 ;
// }
// }
r[n] += (2 * (old_q ^ sr.M) - 1) * r[m];
sr.Q ^= old_q ^ (sr.M ? r[n] > tmp0 : r[n] >= tmp0);
sr.T = (sr.Q == sr.M);
2013-12-19 17:10:14 +00:00
}
//************************ Simple maths ************************
//addc <REG_M>,<REG_N>
sh4op(i0011_nnnn_mmmm_1110)
{
u32 n = GetN(op);
u32 m = GetM(op);
u32 tmp1 = r[n] + r[m];
u32 tmp0 = r[n];
r[n] = tmp1 + sr.T;
if (tmp0 > tmp1)
sr.T = 1;
else
sr.T = 0;
if (tmp1 > r[n])
sr.T = 1;
}
// addv <REG_M>,<REG_N>
sh4op(i0011_nnnn_mmmm_1111)
{
2018-08-17 16:30:54 +00:00
//printf("WARN: addv <REG_M>,<REG_N> used, %04X\n",op);
2013-12-19 17:10:14 +00:00
//Retail game "Twinkle Star Sprites" "uses" this opcode.
u32 n = GetN(op);
u32 m = GetM(op);
s64 br=(s64)(s32)r[n]+(s64)(s32)r[m];
//u32 rm=r[m];
//u32 rn=r[n];
/*__asm
{
mov eax,rm;
mov ecx,rn;
add eax,ecx;
seto sr.T;
mov rn,eax;
};*/
//r[n]=rn;
2013-12-19 17:10:14 +00:00
if (br >=0x80000000)
sr.T=1;
else if (br < (s64) (0xFFFFFFFF80000000u))
sr.T=1;
else
sr.T=0;
/*if (br>>32)
sr.T=1;
else
sr.T=0;*/
r[n]+=r[m];
}
//subc <REG_M>,<REG_N>
sh4op(i0011_nnnn_mmmm_1010)
{//ToDo : Check This [26/4/05]
u32 n = GetN(op);
u32 m = GetM(op);
u32 tmp1 = r[n] - r[m];
u32 tmp0 = r[n];
r[n] = tmp1 - sr.T;
if (tmp0 < tmp1)
sr.T=1;
else
sr.T=0;
if (tmp1 < r[n])
sr.T=1;
}
//subv <REG_M>,<REG_N>
sh4op(i0011_nnnn_mmmm_1011)
{
2018-08-17 16:30:54 +00:00
//printf("WARN: subv <REG_M>,<REG_N> used, %04X\n",op);
2013-12-19 17:10:14 +00:00
//Retail game "Twinkle Star Sprites" "uses" this opcode.
u32 n = GetN(op);
u32 m = GetM(op);
s64 br=(s64)(s32)r[n]-(s64)(s32)r[m];
if (br >=0x80000000)
sr.T=1;
else if (br < (s64) (0xFFFFFFFF80000000u))
sr.T=1;
else
sr.T=0;
/*if (br>>32)
sr.T=1;
else
sr.T=0;*/
r[n]-=r[m];
//u32 rm=r[m];
//u32 rn=r[n];
/*__asm
{
mov eax,rm;
mov ecx,rn;
sub eax,ecx;
seto sr.T;
mov rn,eax;
};*/
//r[n]=rn;
}
//dt <REG_N>
sh4op(i0100_nnnn_0001_0000)
{
u32 n = GetN(op);
r[n]-=1;
if (r[n] == 0)
sr.T=1;
else
sr.T=0;
}
//negc <REG_M>,<REG_N>
sh4op(i0110_nnnn_mmmm_1010)
{
u32 n = GetN(op);
u32 m = GetM(op);
//r[n]=-r[m]-sr.T;
u32 tmp=0 - r[m];
r[n]=tmp - sr.T;
if (0<tmp)
sr.T=1;
else
sr.T=0;
if (tmp<r[n])
sr.T=1;
}
//neg <REG_M>,<REG_N>
sh4op(i0110_nnnn_mmmm_1011)
{//ToDo : Check This [26/4/05]
u32 n = GetN(op);
u32 m = GetM(op);
r[n] = -r[m];
}
//not <REG_M>,<REG_N>
sh4op(i0110_nnnn_mmmm_0111)
{
u32 n = GetN(op);
u32 m = GetM(op);
r[n] = ~r[m];
}
//************************ shifts/rotates ************************
//shll <REG_N>
sh4op(i0100_nnnn_0000_0000)
{//ToDo : Check This [26/4/05]
u32 n = GetN(op);
sr.T = r[n] >> 31;
r[n] <<= 1;
}
//shal <REG_N>
sh4op(i0100_nnnn_0010_0000)
{
u32 n=GetN(op);
sr.T=r[n]>>31;
r[n]=((s32)r[n])<<1;
}
//shlr <REG_N>
sh4op(i0100_nnnn_0000_0001)
{//ToDo : Check This [26/4/05]
u32 n = GetN(op);
sr.T = r[n] & 0x1;
r[n] >>= 1;
}
//shar <REG_N>
sh4op(i0100_nnnn_0010_0001)
{//ToDo : Check This [26/4/05] x2
u32 n = GetN(op);
sr.T=r[n] & 1;
r[n]=((s32)r[n])>>1;
}
//shad <REG_M>,<REG_N>
sh4op(i0100_nnnn_mmmm_1100)
{
u32 n = GetN(op);
u32 m = GetM(op);
u32 sgn = r[m] & 0x80000000;
if (sgn == 0)
r[n] <<= (r[m] & 0x1F);
else if ((r[m] & 0x1F) == 0)
{
r[n]=((s32)r[n])>>31;
}
else
r[n] = ((s32)r[n]) >> ((~r[m] & 0x1F) + 1);
}
//shld <REG_M>,<REG_N>
sh4op(i0100_nnnn_mmmm_1101)
{
u32 n = GetN(op);
u32 m = GetM(op);
u32 sgn = r[m] & 0x80000000;
if (sgn == 0)
r[n] <<= (r[m] & 0x1F);
else if ((r[m] & 0x1F) == 0)
{
r[n] = 0;
}
else
r[n] = ((u32)r[n]) >> ((~r[m] & 0x1F) + 1); //isn't this the same as -r[m] ?
}
//rotcl <REG_N>
sh4op(i0100_nnnn_0010_0100)
{
u32 n = GetN(op);
u32 t;
t = sr.T;
sr.T = r[n] >> 31;
r[n] <<= 1;
r[n]|=t;
}
//rotl <REG_N>
sh4op(i0100_nnnn_0000_0100)
{
u32 n = GetN(op);
sr.T=r[n]>>31;
r[n] <<= 1;
r[n]|=sr.T;
}
//rotcr <REG_N>
sh4op(i0100_nnnn_0010_0101)
{
u32 n = GetN(op);
u32 t;
t = r[n] & 0x1;
r[n] >>= 1;
r[n] |=(sr.T)<<31;
sr.T = t;
}
//rotr <REG_N>
sh4op(i0100_nnnn_0000_0101)
{
u32 n = GetN(op);
sr.T = r[n] & 0x1;
r[n] >>= 1;
r[n] |= ((sr.T) << 31);
}
//************************ byte reorder/sign ************************
//swap.b <REG_M>,<REG_N>
sh4op(i0110_nnnn_mmmm_1000)
{
u32 m = GetM(op);
u32 n = GetN(op);
u32 rg=r[m];
r[n] = (rg & 0xFFFF0000) | ((rg&0xFF)<<8) | ((rg>>8)&0xFF);
}
//swap.w <REG_M>,<REG_N>
sh4op(i0110_nnnn_mmmm_1001)
{
u32 n = GetN(op);
u32 m = GetM(op);
u16 t = (u16)(r[m]>>16);
r[n] = (r[m] << 16) | t;
}
//extu.b <REG_M>,<REG_N>
sh4op(i0110_nnnn_mmmm_1100)
{
u32 n = GetN(op);
u32 m = GetM(op);
r[n] = (u32)(u8)r[m];
}
//extu.w <REG_M>,<REG_N>
sh4op(i0110_nnnn_mmmm_1101)
{
u32 n = GetN(op);
u32 m = GetM(op);
r[n] =(u32)(u16) r[m];
}
//exts.b <REG_M>,<REG_N>
sh4op(i0110_nnnn_mmmm_1110)
{
u32 n = GetN(op);
u32 m = GetM(op);
r[n] = (u32)(s32)(s8)(u8)(r[m]);
}
//exts.w <REG_M>,<REG_N>
sh4op(i0110_nnnn_mmmm_1111)
{
u32 n = GetN(op);
u32 m = GetM(op);
r[n] = (u32)(s32)(s16)(u16)(r[m]);
}
//xtrct <REG_M>,<REG_N>
sh4op(i0010_nnnn_mmmm_1101)
{
u32 n = GetN(op);
u32 m = GetM(op);
r[n] = ((r[n] >> 16) & 0xFFFF) | ((r[m] << 16) & 0xFFFF0000);
}
//************************ xxx.b #<imm>,@(R0,GBR) ************************
//tst.b #<imm>,@(R0,GBR)
sh4op(i1100_1100_iiii_iiii)
{
2018-08-17 16:30:54 +00:00
//printf("WARN: tst.b #<imm>,@(R0,GBR) used, %04X\n",op);
2013-12-19 17:10:14 +00:00
//Retail game "Twinkle Star Sprites" "uses" this opcode.
u32 imm=GetImm8(op);
u32 temp = (u8)ReadMem8(gbr+r[0]);
temp &= imm;
if (temp==0)
sr.T=1;
else
sr.T=0;
}
//and.b #<imm>,@(R0,GBR)
sh4op(i1100_1101_iiii_iiii)
{
u8 temp = (u8)ReadMem8(gbr+r[0]);
2013-12-19 17:10:14 +00:00
temp &= GetImm8(op);
WriteMem8(gbr +r[0], temp);
}
//xor.b #<imm>,@(R0,GBR)
sh4op(i1100_1110_iiii_iiii)
{
u8 temp = (u8)ReadMem8(gbr+r[0]);
temp ^= GetImm8(op);
WriteMem8(gbr +r[0], temp);
}
//or.b #<imm>,@(R0,GBR)
sh4op(i1100_1111_iiii_iiii)
{
u8 temp = (u8)ReadMem8(gbr+r[0]);
2013-12-19 17:10:14 +00:00
temp |= GetImm8(op);
WriteMem8(gbr+r[0], temp);
}
//tas.b @<REG_N>
sh4op(i0100_nnnn_0001_1011)
{
u32 n = GetN(op);
u8 val;
val=(u8)ReadMem8(r[n]);
u32 srT;
if (val == 0)
srT = 1;
else
srT = 0;
val |= 0x80;
WriteMem8(r[n], val);
sr.T=srT;
}
//************************ Opcodes that read/write the status registers ************************
//stc SR,<REG_N>
sh4op(i0000_nnnn_0000_0010)//0002
{
u32 n = GetN(op);
2018-09-02 13:49:23 +00:00
r[n] = sh4_sr_GetFull();
2013-12-19 17:10:14 +00:00
}
//sts FPSCR,<REG_N>
sh4op(i0000_nnnn_0110_1010)
{
u32 n = GetN(op);
r[n] = fpscr.full;
//UpdateFPSCR();
2013-12-19 17:10:14 +00:00
}
//sts.l FPSCR,@-<REG_N>
sh4op(i0100_nnnn_0110_0010)
{
u32 n = GetN(op);
WriteMemU32(r[n] - 4, fpscr.full);
2013-12-19 17:10:14 +00:00
r[n] -= 4;
}
//stc.l SR,@-<REG_N>
sh4op(i0100_nnnn_0000_0011)
{
u32 n = GetN(op);
WriteMemU32(r[n] - 4, sh4_sr_GetFull());
2013-12-19 17:10:14 +00:00
r[n] -= 4;
}
//lds.l @<REG_N>+,FPSCR
sh4op(i0100_nnnn_0110_0110)
{
u32 n = GetN(op);
ReadMemU32(fpscr.full,r[n]);
UpdateFPSCR();
r[n] += 4;
}
//ldc.l @<REG_N>+,SR
sh4op(i0100_nnnn_0000_0111)
{
u32 n = GetN(op);
u32 sr_t;
ReadMemU32(sr_t,r[n]);
2018-09-02 13:49:23 +00:00
sh4_sr_SetFull(sr_t);
2013-12-19 17:10:14 +00:00
r[n] += 4;
if (UpdateSR())
{
UpdateINTC();
}
}
//lds <REG_N>,FPSCR
sh4op(i0100_nnnn_0110_1010)
{
u32 n = GetN(op);
fpscr.full = r[n];
UpdateFPSCR();
}
//ldc <REG_N>,SR
sh4op(i0100_nnnn_0000_1110)
{
u32 n = GetN(op);
2018-09-02 13:49:23 +00:00
sh4_sr_SetFull(r[n]);
2013-12-19 17:10:14 +00:00
if (UpdateSR())
{
UpdateINTC();
}
}
//Not implt
sh4op(iNotImplemented)
{
#ifndef NO_MMU
2019-07-01 13:22:04 +00:00
INFO_LOG(INTERPRETER, "iNimp %04X", op);
SH4ThrownException ex = { next_pc - 2, 0x180, 0x100 };
throw ex;
#else
cpu_iNimp(op, "Unknown opcode");
#endif
2013-12-19 17:10:14 +00:00
}