2012-11-01 16:19:01 +01:00
|
|
|
// Copyright (c) 2012- PPSSPP Project.
|
|
|
|
|
|
|
|
// 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
|
2012-11-04 23:58:25 +01:00
|
|
|
// the Free Software Foundation, version 2.0 or later versions.
|
2012-11-01 16:19:01 +01:00
|
|
|
|
|
|
|
// 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 2.0 for more details.
|
|
|
|
|
|
|
|
// A copy of the GPL 2.0 should have been included with the program.
|
|
|
|
// If not, see http://www.gnu.org/licenses/
|
|
|
|
|
|
|
|
// Official git repository and contact information can be found at
|
|
|
|
// https://github.com/hrydgard/ppsspp and http://www.ppsspp.org/.
|
|
|
|
|
2013-01-07 22:33:09 +01:00
|
|
|
#include "ArmJit.h"
|
|
|
|
#include "ArmRegCache.h"
|
|
|
|
#include "ArmEmitter.h"
|
2012-11-01 16:19:01 +01:00
|
|
|
|
|
|
|
using namespace MIPSAnalyst;
|
|
|
|
#define _RS ((op>>21) & 0x1F)
|
|
|
|
#define _RT ((op>>16) & 0x1F)
|
|
|
|
#define _RD ((op>>11) & 0x1F)
|
|
|
|
#define _FS ((op>>11) & 0x1F)
|
|
|
|
#define _FT ((op>>16) & 0x1F)
|
|
|
|
#define _FD ((op>>6 ) & 0x1F)
|
|
|
|
#define _SA ((op>>6 ) & 0x1F)
|
|
|
|
#define _POS ((op>>6 ) & 0x1F)
|
|
|
|
#define _SIZE ((op>>11 ) & 0x1F)
|
|
|
|
|
|
|
|
#define OLDD Comp_Generic(op); return;
|
|
|
|
|
|
|
|
namespace MIPSComp
|
|
|
|
{
|
|
|
|
/*
|
2013-01-08 13:49:52 +01:00
|
|
|
void Jit::CompImmLogic(u32 op, void (ARMXEmitter::*arith)(ARMReg dst, ARMReg src, Operand2 op2))
|
2012-11-01 16:19:01 +01:00
|
|
|
{
|
|
|
|
u32 uimm = (u16)(op & 0xFFFF);
|
2013-01-08 13:49:52 +01:00
|
|
|
gpr.SpillLock()
|
2012-11-01 16:19:01 +01:00
|
|
|
int rt = _RT;
|
|
|
|
int rs = _RS;
|
|
|
|
gpr.Lock(rt, rs);
|
|
|
|
gpr.BindToRegister(rt, rt == rs, true);
|
|
|
|
if (rt != rs)
|
|
|
|
MOV(32, gpr.R(rt), gpr.R(rs));
|
|
|
|
(this->*arith)(32, gpr.R(rt), Imm32(uimm));
|
|
|
|
gpr.UnlockAll();
|
2013-01-08 13:49:52 +01:00
|
|
|
}*/
|
2012-11-01 16:19:01 +01:00
|
|
|
|
|
|
|
void Jit::Comp_IType(u32 op)
|
|
|
|
{
|
|
|
|
s32 simm = (s16)(op & 0xFFFF);
|
|
|
|
u32 uimm = (u16)(op & 0xFFFF);
|
|
|
|
|
|
|
|
int rt = _RT;
|
|
|
|
int rs = _RS;
|
|
|
|
|
|
|
|
switch (op >> 26)
|
|
|
|
{
|
2013-01-08 17:03:17 +01:00
|
|
|
/*
|
2012-11-01 16:19:01 +01:00
|
|
|
case 8: // same as addiu?
|
|
|
|
case 9: //R(rt) = R(rs) + simm; break; //addiu
|
|
|
|
{
|
2013-01-08 13:49:52 +01:00
|
|
|
if (gpr.IsImm(rs))
|
2012-11-01 16:19:01 +01:00
|
|
|
{
|
2013-01-08 13:49:52 +01:00
|
|
|
gpr.SetImm(rt, gpr.GetImm(rs) + simm);
|
2012-11-01 16:19:01 +01:00
|
|
|
break;
|
2013-01-08 13:49:52 +01:00
|
|
|
} else if (rs == 0) {
|
|
|
|
gpr.SetImm(rt, simm);
|
|
|
|
} else {
|
|
|
|
gpr.SpillLock(rs, rt);
|
|
|
|
gpr.MapReg(rs, MAP_INITVAL);
|
|
|
|
gpr.MapReg(rt, MAP_INITVAL | MAP_DIRTY);
|
|
|
|
ARMABI_MOVI2R(R0, (u32)simm);
|
|
|
|
ADD(gpr.R(rt), gpr.R(rs), R0);
|
|
|
|
gpr.ReleaseSpillLocks();
|
2012-11-01 16:19:01 +01:00
|
|
|
}
|
2013-01-08 13:49:52 +01:00
|
|
|
break;
|
2013-01-08 17:03:17 +01:00
|
|
|
}*/
|
2013-01-08 13:49:52 +01:00
|
|
|
/*
|
|
|
|
case 13: // OR
|
|
|
|
{
|
|
|
|
if (gpr.IsImm(rs))
|
2012-11-01 16:19:01 +01:00
|
|
|
{
|
2013-01-08 13:49:52 +01:00
|
|
|
gpr.SetImm(rt, gpr.GetImm(rs) | uimm);
|
|
|
|
break;
|
|
|
|
} else if (rs == 0) {
|
|
|
|
gpr.SetImm(rt, uimm);
|
|
|
|
} else {
|
|
|
|
gpr.SpillLock(rs, rt);
|
|
|
|
gpr.MapReg(rs, MAP_INITVAL);
|
|
|
|
gpr.MapReg(rt, MAP_INITVAL | MAP_DIRTY);
|
|
|
|
ARMABI_MOVI2R(R0, (u32)uimm);
|
|
|
|
ORR(gpr.R(rt), gpr.R(rs), R0);
|
|
|
|
gpr.ReleaseSpillLocks();
|
2012-11-01 16:19:01 +01:00
|
|
|
}
|
2013-01-08 13:49:52 +01:00
|
|
|
break;
|
2012-11-01 16:19:01 +01:00
|
|
|
}
|
|
|
|
break;
|
2013-01-08 13:49:52 +01:00
|
|
|
*/
|
|
|
|
//case 12: CompImmLogic(op, &XEmitter::AND); break;
|
|
|
|
//case 14: CompImmLogic(op, &XEmitter::XOR); break;
|
2012-11-01 16:19:01 +01:00
|
|
|
|
2013-01-08 13:49:52 +01:00
|
|
|
/*
|
2012-11-01 16:19:01 +01:00
|
|
|
case 10: // R(rt) = (s32)R(rs) < simm; break; //slti
|
|
|
|
gpr.Lock(rt, rs);
|
|
|
|
gpr.BindToRegister(rt, rt == rs, true);
|
|
|
|
XOR(32, R(EAX), R(EAX));
|
|
|
|
CMP(32, gpr.R(rs), Imm32(simm));
|
|
|
|
SETcc(CC_L, R(EAX));
|
|
|
|
MOV(32, gpr.R(rt), R(EAX));
|
|
|
|
gpr.UnlockAll();
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 11: // R(rt) = R(rs) < uimm; break; //sltiu
|
|
|
|
gpr.Lock(rt, rs);
|
|
|
|
gpr.BindToRegister(rt, rt == rs, true);
|
|
|
|
XOR(32, R(EAX), R(EAX));
|
|
|
|
CMP(32, gpr.R(rs), Imm32((u32)simm));
|
|
|
|
SETcc(CC_B, R(EAX));
|
|
|
|
MOV(32, gpr.R(rt), R(EAX));
|
|
|
|
gpr.UnlockAll();
|
|
|
|
break;
|
|
|
|
|
2013-01-09 00:12:38 +01:00
|
|
|
*/
|
2012-11-01 16:19:01 +01:00
|
|
|
|
|
|
|
case 15: //R(rt) = uimm << 16; break; //lui
|
2013-01-08 13:49:52 +01:00
|
|
|
gpr.SetImm(rt, uimm << 16);
|
2012-11-01 16:19:01 +01:00
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
Comp_Generic(op);
|
|
|
|
break;
|
2013-01-08 13:49:52 +01:00
|
|
|
}
|
2012-11-01 16:19:01 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
//rd = rs X rt
|
|
|
|
/*
|
|
|
|
void Jit::CompTriArith(u32 op, void (XEmitter::*arith)(int, const OpArg &, const OpArg &))
|
|
|
|
{
|
|
|
|
int rt = _RT;
|
|
|
|
int rs = _RS;
|
|
|
|
int rd = _RD;
|
|
|
|
|
|
|
|
gpr.Lock(rt, rs, rd);
|
|
|
|
MOV(32, R(EAX), gpr.R(rs));
|
|
|
|
MOV(32, R(EBX), gpr.R(rt));
|
|
|
|
gpr.BindToRegister(rd, true, true);
|
|
|
|
(this->*arith)(32, R(EAX), R(EBX));
|
|
|
|
MOV(32, gpr.R(rd), R(EAX));
|
|
|
|
gpr.UnlockAll();
|
|
|
|
}
|
|
|
|
*/
|
|
|
|
|
|
|
|
void Jit::Comp_RType3(u32 op)
|
|
|
|
{
|
|
|
|
OLDD
|
|
|
|
|
|
|
|
int rt = _RT;
|
|
|
|
int rs = _RS;
|
|
|
|
int rd = _RD;
|
|
|
|
|
2013-01-08 13:49:52 +01:00
|
|
|
gpr.SpillLock(rt, rs, rd);
|
|
|
|
gpr.MapReg(rt, MAP_INITVAL);
|
|
|
|
gpr.MapReg(rs, MAP_INITVAL);
|
|
|
|
gpr.MapReg(rd, MAP_INITVAL | MAP_DIRTY); // can get rid of INITVAL in some cases
|
|
|
|
gpr.ReleaseSpillLocks();
|
2012-11-01 16:19:01 +01:00
|
|
|
|
|
|
|
switch (op & 63)
|
|
|
|
{
|
|
|
|
//case 10: if (!R(rt)) R(rd) = R(rs); break; //movz
|
|
|
|
//case 11: if (R(rt)) R(rd) = R(rs); break; //movn
|
|
|
|
|
|
|
|
// case 32: //R(rd) = R(rs) + R(rt); break; //add
|
|
|
|
case 33: //R(rd) = R(rs) + R(rt); break; //addu
|
2012-11-23 19:41:35 +01:00
|
|
|
ADD(gpr.R(rd), gpr.R(rs), gpr.R(rt));
|
2012-11-01 16:19:01 +01:00
|
|
|
break;
|
|
|
|
case 134: //R(rd) = R(rs) - R(rt); break; //sub
|
|
|
|
case 135:
|
2012-11-23 19:41:35 +01:00
|
|
|
SUB(gpr.R(rd), gpr.R(rs), gpr.R(rt));
|
2012-11-01 16:19:01 +01:00
|
|
|
break;
|
|
|
|
case 136: //R(rd) = R(rs) & R(rt); break; //and
|
2012-11-23 19:41:35 +01:00
|
|
|
AND(gpr.R(rd), gpr.R(rs), gpr.R(rt));
|
2012-11-01 16:19:01 +01:00
|
|
|
break;
|
|
|
|
case 137: //R(rd) = R(rs) | R(rt); break; //or
|
2012-11-23 19:41:35 +01:00
|
|
|
ORR(gpr.R(rd), gpr.R(rs), gpr.R(rt));
|
2012-11-01 16:19:01 +01:00
|
|
|
break;
|
|
|
|
case 138: //R(rd) = R(rs) ^ R(rt); break; //xor/eor
|
2012-11-23 19:41:35 +01:00
|
|
|
EOR(gpr.R(rd), gpr.R(rs), gpr.R(rt));
|
2012-11-01 16:19:01 +01:00
|
|
|
break;
|
|
|
|
|
|
|
|
case 39: // R(rd) = ~(R(rs) | R(rt)); //nor
|
2012-11-23 19:41:35 +01:00
|
|
|
ORR(gpr.R(rd), gpr.R(rs), gpr.R(rt));
|
|
|
|
MVN(gpr.R(rd), gpr.R(rd));
|
2012-11-01 16:19:01 +01:00
|
|
|
break;
|
|
|
|
|
|
|
|
case 42: //R(rd) = (int)R(rs) < (int)R(rt); break; //slt
|
2012-11-23 19:41:35 +01:00
|
|
|
CMP(gpr.R(rs), gpr.R(rt));
|
2012-11-01 16:19:01 +01:00
|
|
|
SetCC(CC_LT);
|
2012-11-23 19:41:35 +01:00
|
|
|
ARMABI_MOVI2R(gpr.R(rd), 1);
|
2012-11-01 16:19:01 +01:00
|
|
|
SetCC(CC_GE);
|
2012-11-23 19:41:35 +01:00
|
|
|
ARMABI_MOVI2R(gpr.R(rd), 0);
|
2012-11-01 16:19:01 +01:00
|
|
|
SetCC(CC_AL);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 43: //R(rd) = R(rs) < R(rt); break; //sltu
|
2012-11-23 19:41:35 +01:00
|
|
|
CMP(gpr.R(rs), gpr.R(rt));
|
2012-11-01 16:19:01 +01:00
|
|
|
SetCC(CC_LO);
|
2012-11-23 19:41:35 +01:00
|
|
|
ARMABI_MOVI2R(gpr.R(rd), 1);
|
2012-11-01 16:19:01 +01:00
|
|
|
SetCC(CC_HS);
|
2012-11-23 19:41:35 +01:00
|
|
|
ARMABI_MOVI2R(gpr.R(rd), 0);
|
2012-11-01 16:19:01 +01:00
|
|
|
SetCC(CC_AL);
|
|
|
|
break;
|
|
|
|
|
|
|
|
// case 44: R(rd) = (R(rs) > R(rt)) ? R(rs) : R(rt); break; //max
|
|
|
|
// CMP(a,b); CMOVLT(a,b)
|
|
|
|
|
|
|
|
// case 45: R(rd) = (R(rs) < R(rt)) ? R(rs) : R(rt); break; //min
|
|
|
|
// CMP(a,b); CMOVGT(a,b)
|
|
|
|
|
|
|
|
default:
|
2012-11-23 19:41:35 +01:00
|
|
|
// gpr.UnlockAll();
|
2012-11-01 16:19:01 +01:00
|
|
|
Comp_Generic(op);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
|
|
|
void Jit::CompShiftImm(u32 op, void (XEmitter::*shift)(int, OpArg, OpArg))
|
|
|
|
{
|
|
|
|
int rd = _RD;
|
|
|
|
int rt = _RT;
|
|
|
|
gpr.Lock(rd, rt);
|
|
|
|
int sa = _SA;
|
|
|
|
gpr.BindToRegister(rd, rd == rt, true);
|
|
|
|
if (rd != rt)
|
|
|
|
MOV(32, gpr.R(rd), gpr.R(rt));
|
|
|
|
(this->*shift)(32, gpr.R(rd), Imm8(sa));
|
|
|
|
gpr.UnlockAll();
|
|
|
|
}
|
|
|
|
*/
|
|
|
|
// "over-shifts" work the same as on x86 - only bottom 5 bits are used to get the shift value
|
|
|
|
/*
|
|
|
|
void Jit::CompShiftVar(u32 op, void (XEmitter::*shift)(int, OpArg, OpArg))
|
|
|
|
{
|
|
|
|
int rd = _RD;
|
|
|
|
int rt = _RT;
|
|
|
|
int rs = _RS;
|
|
|
|
gpr.FlushLockX(ECX);
|
|
|
|
gpr.Lock(rd, rt, rs);
|
|
|
|
gpr.BindToRegister(rd, true, true);
|
|
|
|
if (rd != rt)
|
|
|
|
MOV(32, gpr.R(rd), gpr.R(rt));
|
|
|
|
MOV(32, R(ECX), gpr.R(rs)); // Only ECX can be used for variable shifts.
|
|
|
|
AND(32, R(ECX), Imm32(0x1f));
|
|
|
|
(this->*shift)(32, gpr.R(rd), R(ECX));
|
|
|
|
gpr.UnlockAll();
|
|
|
|
gpr.UnlockAllX();
|
|
|
|
}
|
|
|
|
*/
|
|
|
|
void Jit::Comp_ShiftType(u32 op)
|
|
|
|
{
|
|
|
|
// WARNIGN : ROTR
|
|
|
|
OLDD
|
|
|
|
switch (op & 0x3f)
|
|
|
|
{
|
|
|
|
//case 0: CompShiftImm(op, &ARMXEmitter::SHL); break;
|
|
|
|
//case 2: CompShiftImm(op, &XEmitter::SHR); break; // srl
|
|
|
|
//case 3: CompShiftImm(op, &XEmitter::SAR); break; // sra
|
|
|
|
|
|
|
|
// case 4: CompShiftVar(op, &XEmitter::SHL); break; // R(rd) = R(rt) << R(rs); break; //sllv
|
|
|
|
// case 6: CompShiftVar(op, &XEmitter::SHR); break; // R(rd) = R(rt) >> R(rs); break; //srlv
|
|
|
|
// case 7: CompShiftVar(op, &XEmitter::SAR); break; // R(rd) = ((s32)R(rt)) >> R(rs); break; //srav
|
|
|
|
|
|
|
|
default:
|
|
|
|
Comp_Generic(op);
|
|
|
|
//_dbg_assert_msg_(CPU,0,"Trying to interpret instruction that can't be interpreted");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Jit::Comp_Allegrex(u32 op)
|
|
|
|
{
|
|
|
|
OLDD
|
|
|
|
int rt = _RT;
|
|
|
|
int rd = _RD;
|
|
|
|
switch ((op >> 6) & 31)
|
|
|
|
{
|
|
|
|
case 16: // seb // R(rd) = (u32)(s32)(s8)(u8)R(rt);
|
|
|
|
/*
|
|
|
|
gpr.Lock(rd, rt);
|
|
|
|
gpr.BindToRegister(rd, true, true);
|
|
|
|
MOV(32, R(EAX), gpr.R(rt)); // work around the byte-register addressing problem
|
|
|
|
MOVSX(32, 8, gpr.RX(rd), R(EAX));
|
|
|
|
gpr.UnlockAll();*/
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 24: // seh
|
|
|
|
/*
|
|
|
|
gpr.Lock(rd, rt);
|
|
|
|
gpr.BindToRegister(rd, true, true);
|
|
|
|
MOVSX(32, 16, gpr.RX(rd), gpr.R(rt));
|
|
|
|
gpr.UnlockAll();*/
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 20: //bitrev
|
|
|
|
default:
|
|
|
|
Comp_Generic(op);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|