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"
|
2013-03-03 16:40:58 +01:00
|
|
|
#include "Common/CPUDetect.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)
|
|
|
|
|
2013-03-01 10:34:49 -08:00
|
|
|
// All functions should have CONDITIONAL_DISABLE, so we can narrow things down to a file quickly.
|
|
|
|
// Currently known non working ones should have DISABLE.
|
|
|
|
|
|
|
|
//#define CONDITIONAL_DISABLE { Comp_Generic(op); return; }
|
|
|
|
#define CONDITIONAL_DISABLE ;
|
|
|
|
#define DISABLE { Comp_Generic(op); return; }
|
2012-11-01 16:19:01 +01:00
|
|
|
|
|
|
|
namespace MIPSComp
|
|
|
|
{
|
2013-01-10 01:08:24 +01:00
|
|
|
static u32 EvalOr(u32 a, u32 b) { return a | b; }
|
|
|
|
static u32 EvalXor(u32 a, u32 b) { return a ^ b; }
|
|
|
|
static u32 EvalAnd(u32 a, u32 b) { return a & b; }
|
2013-01-09 11:20:48 +01:00
|
|
|
|
|
|
|
void Jit::CompImmLogic(int rs, int rt, u32 uimm, void (ARMXEmitter::*arith)(ARMReg dst, ARMReg src, Operand2 op2), u32 (*eval)(u32 a, u32 b))
|
2012-11-01 16:19:01 +01:00
|
|
|
{
|
2013-01-09 11:20:48 +01:00
|
|
|
if (gpr.IsImm(rs)) {
|
|
|
|
gpr.SetImm(rt, (*eval)(gpr.GetImm(rs), uimm));
|
|
|
|
} else {
|
2013-01-11 17:25:54 +01:00
|
|
|
gpr.MapDirtyIn(rt, rs);
|
2013-01-09 11:20:48 +01:00
|
|
|
// TODO: Special case when uimm can be represented as an Operand2
|
2013-01-10 01:08:24 +01:00
|
|
|
Operand2 op2;
|
|
|
|
if (TryMakeOperand2(uimm, op2)) {
|
|
|
|
(this->*arith)(gpr.R(rt), gpr.R(rs), op2);
|
|
|
|
} else {
|
2013-01-30 21:49:20 +01:00
|
|
|
MOVI2R(R0, (u32)uimm);
|
2013-01-10 01:08:24 +01:00
|
|
|
(this->*arith)(gpr.R(rt), gpr.R(rs), R0);
|
|
|
|
}
|
2013-01-09 11:20:48 +01:00
|
|
|
}
|
|
|
|
}
|
2012-11-01 16:19:01 +01:00
|
|
|
|
|
|
|
void Jit::Comp_IType(u32 op)
|
|
|
|
{
|
2013-03-01 10:34:49 -08:00
|
|
|
CONDITIONAL_DISABLE;
|
2013-01-10 00:03:51 +01:00
|
|
|
s32 simm = (s32)(s16)(op & 0xFFFF); // sign extension
|
2013-01-11 23:42:58 +01:00
|
|
|
u32 uimm = op & 0xFFFF;
|
2013-02-06 23:35:24 +01:00
|
|
|
u32 suimm = (u32)(s32)simm;
|
2012-11-01 16:19:01 +01:00
|
|
|
|
|
|
|
int rt = _RT;
|
|
|
|
int rs = _RS;
|
|
|
|
|
|
|
|
switch (op >> 26)
|
|
|
|
{
|
|
|
|
case 8: // same as addiu?
|
2013-01-10 01:08:24 +01:00
|
|
|
case 9: // R(rt) = R(rs) + simm; break; //addiu
|
2012-11-01 16:19:01 +01:00
|
|
|
{
|
2013-01-09 11:20:48 +01:00
|
|
|
if (gpr.IsImm(rs)) {
|
2013-01-08 13:49:52 +01:00
|
|
|
gpr.SetImm(rt, gpr.GetImm(rs) + simm);
|
2013-01-09 13:38:44 +01:00
|
|
|
} else if (rs == 0) { // add to zero register = immediate
|
|
|
|
gpr.SetImm(rt, (u32)simm);
|
2013-01-08 13:49:52 +01:00
|
|
|
} else {
|
2013-01-11 17:25:54 +01:00
|
|
|
gpr.MapDirtyIn(rt, rs);
|
2013-01-09 11:20:48 +01:00
|
|
|
Operand2 op2;
|
2013-01-11 17:25:54 +01:00
|
|
|
bool negated;
|
|
|
|
if (TryMakeOperand2_AllowNegation(simm, op2, &negated)) {
|
|
|
|
if (!negated)
|
|
|
|
ADD(gpr.R(rt), gpr.R(rs), op2);
|
|
|
|
else
|
|
|
|
SUB(gpr.R(rt), gpr.R(rs), op2);
|
2013-01-09 11:20:48 +01:00
|
|
|
} else {
|
2013-01-30 21:49:20 +01:00
|
|
|
MOVI2R(R0, (u32)simm);
|
2013-01-09 11:20:48 +01:00
|
|
|
ADD(gpr.R(rt), gpr.R(rs), R0);
|
|
|
|
}
|
2013-01-10 00:03:51 +01:00
|
|
|
}
|
2013-01-08 13:49:52 +01:00
|
|
|
break;
|
2012-11-01 16:19:01 +01:00
|
|
|
}
|
|
|
|
|
2013-01-10 01:08:24 +01:00
|
|
|
case 12: CompImmLogic(rs, rt, uimm, &ARMXEmitter::AND, &EvalAnd); break;
|
|
|
|
case 13: CompImmLogic(rs, rt, uimm, &ARMXEmitter::ORR, &EvalOr); break;
|
|
|
|
case 14: CompImmLogic(rs, rt, uimm, &ARMXEmitter::EOR, &EvalXor); break;
|
|
|
|
|
2012-11-01 16:19:01 +01:00
|
|
|
case 10: // R(rt) = (s32)R(rs) < simm; break; //slti
|
2013-01-10 01:08:24 +01:00
|
|
|
{
|
2013-01-11 23:42:58 +01:00
|
|
|
gpr.MapDirtyIn(rt, rs);
|
2013-01-10 01:08:24 +01:00
|
|
|
Operand2 op2;
|
2013-01-30 00:02:04 +01:00
|
|
|
bool negated;
|
|
|
|
if (TryMakeOperand2_AllowNegation(simm, op2, &negated)) {
|
|
|
|
if (!negated)
|
|
|
|
CMP(gpr.R(rs), op2);
|
|
|
|
else
|
|
|
|
CMN(gpr.R(rs), op2);
|
2013-01-10 01:08:24 +01:00
|
|
|
} else {
|
2013-01-30 21:49:20 +01:00
|
|
|
MOVI2R(R0, simm);
|
2013-01-10 01:08:24 +01:00
|
|
|
CMP(gpr.R(rs), R0);
|
|
|
|
}
|
2013-01-11 23:42:58 +01:00
|
|
|
SetCC(CC_LT);
|
2013-01-30 21:49:20 +01:00
|
|
|
MOVI2R(gpr.R(rt), 1);
|
2013-01-11 23:42:58 +01:00
|
|
|
SetCC(CC_GE);
|
2013-01-30 21:49:20 +01:00
|
|
|
MOVI2R(gpr.R(rt), 0);
|
2013-01-11 23:42:58 +01:00
|
|
|
SetCC(CC_AL);
|
2013-01-10 01:08:24 +01:00
|
|
|
}
|
2012-11-01 16:19:01 +01:00
|
|
|
break;
|
2013-01-30 00:02:04 +01:00
|
|
|
|
2013-01-10 01:08:24 +01:00
|
|
|
case 11: // R(rt) = R(rs) < uimm; break; //sltiu
|
|
|
|
{
|
2013-01-11 23:42:58 +01:00
|
|
|
gpr.MapDirtyIn(rt, rs);
|
2013-01-10 01:08:24 +01:00
|
|
|
Operand2 op2;
|
2013-01-30 00:02:04 +01:00
|
|
|
bool negated;
|
2013-02-06 23:35:24 +01:00
|
|
|
if (TryMakeOperand2_AllowNegation(suimm, op2, &negated)) {
|
2013-01-30 00:02:04 +01:00
|
|
|
if (!negated)
|
|
|
|
CMP(gpr.R(rs), op2);
|
|
|
|
else
|
|
|
|
CMN(gpr.R(rs), op2);
|
2013-01-10 01:08:24 +01:00
|
|
|
} else {
|
2013-02-06 23:35:24 +01:00
|
|
|
MOVI2R(R0, suimm);
|
2013-01-10 01:08:24 +01:00
|
|
|
CMP(gpr.R(rs), R0);
|
|
|
|
}
|
2013-01-11 23:42:58 +01:00
|
|
|
SetCC(CC_LO);
|
2013-02-06 23:35:24 +01:00
|
|
|
MOVI2R(gpr.R(rt), 1);
|
2013-01-11 23:42:58 +01:00
|
|
|
SetCC(CC_HS);
|
2013-02-06 23:35:24 +01:00
|
|
|
MOVI2R(gpr.R(rt), 0);
|
2013-01-11 23:42:58 +01:00
|
|
|
SetCC(CC_AL);
|
2013-01-10 01:08:24 +01:00
|
|
|
}
|
|
|
|
break;
|
2013-01-30 00:02:04 +01:00
|
|
|
|
2013-01-09 13:38:44 +01:00
|
|
|
case 15: // R(rt) = uimm << 16; //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
|
|
|
}
|
2013-02-21 00:52:37 -08:00
|
|
|
|
|
|
|
void Jit::Comp_RType2(u32 op)
|
|
|
|
{
|
2013-03-05 14:16:35 +10:00
|
|
|
CONDITIONAL_DISABLE;
|
|
|
|
int rs = _RS;
|
|
|
|
int rd = _RD;
|
|
|
|
|
|
|
|
// Don't change $zr.
|
|
|
|
if (rd == 0)
|
|
|
|
return;
|
|
|
|
|
|
|
|
switch (op & 63)
|
|
|
|
{
|
|
|
|
case 22: //clz
|
|
|
|
gpr.MapDirtyIn(rd, rs);
|
|
|
|
CLZ(gpr.R(rd), gpr.R(rs));
|
|
|
|
break;
|
|
|
|
case 23: //clo
|
2013-03-05 15:14:22 +10:00
|
|
|
gpr.MapDirtyIn(rd, rs);
|
|
|
|
RSB(R0, gpr.R(rs), Operand2(0));
|
2013-03-05 15:20:14 +10:00
|
|
|
CLZ(gpr.R(rd), R0);
|
2013-03-05 14:16:35 +10:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
DISABLE;
|
2013-03-05 15:20:14 +10:00
|
|
|
}
|
2013-03-05 14:16:35 +10:00
|
|
|
}
|
2013-02-21 00:52:37 -08:00
|
|
|
|
2012-11-01 16:19:01 +01:00
|
|
|
void Jit::Comp_RType3(u32 op)
|
|
|
|
{
|
2013-03-01 10:34:49 -08:00
|
|
|
CONDITIONAL_DISABLE;
|
2012-11-01 16:19:01 +01:00
|
|
|
int rt = _RT;
|
|
|
|
int rs = _RS;
|
|
|
|
int rd = _RD;
|
2013-01-11 15:22:31 +01:00
|
|
|
|
2012-11-01 16:19:01 +01:00
|
|
|
switch (op & 63)
|
|
|
|
{
|
2013-03-01 02:17:39 +10:00
|
|
|
//case 10: if (!R(rt)) R(rd) = R(rs); break; //movz
|
|
|
|
//case 11: if (R(rt)) R(rd) = R(rs); break; //movn
|
2012-11-01 16:19:01 +01:00
|
|
|
|
2013-03-01 02:17:39 +10:00
|
|
|
// case 32: //R(rd) = R(rs) + R(rt); break; //add
|
|
|
|
case 33: //R(rd) = R(rs) + R(rt); break; //addu
|
2013-01-11 15:22:31 +01:00
|
|
|
// Some optimized special cases
|
|
|
|
if (rs == 0) {
|
|
|
|
gpr.MapDirtyIn(rd, rt);
|
|
|
|
MOV(gpr.R(rd), gpr.R(rt));
|
|
|
|
} else if (rt == 0) {
|
|
|
|
gpr.MapDirtyIn(rd, rs);
|
|
|
|
MOV(gpr.R(rd), gpr.R(rs));
|
|
|
|
} else {
|
|
|
|
gpr.MapDirtyInIn(rd, rs, rt);
|
|
|
|
ADD(gpr.R(rd), gpr.R(rs), gpr.R(rt));
|
|
|
|
}
|
2012-11-01 16:19:01 +01:00
|
|
|
break;
|
2013-03-01 02:17:39 +10:00
|
|
|
case 34: //R(rd) = R(rs) - R(rt); break; //sub
|
2013-01-11 15:22:31 +01:00
|
|
|
case 35:
|
|
|
|
gpr.MapDirtyInIn(rd, rs, rt);
|
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;
|
2013-03-01 02:17:39 +10:00
|
|
|
case 36: //R(rd) = R(rs) & R(rt); break; //and
|
2013-01-11 15:22:31 +01:00
|
|
|
gpr.MapDirtyInIn(rd, rs, rt);
|
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;
|
2013-03-01 02:17:39 +10:00
|
|
|
case 37: //R(rd) = R(rs) | R(rt); break; //or
|
2013-01-11 15:22:31 +01:00
|
|
|
gpr.MapDirtyInIn(rd, rs, rt);
|
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;
|
2013-03-01 02:17:39 +10:00
|
|
|
case 38: //R(rd) = R(rs) ^ R(rt); break; //xor/eor
|
2013-01-11 15:22:31 +01:00
|
|
|
gpr.MapDirtyInIn(rd, rs, rt);
|
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;
|
|
|
|
|
2013-03-01 02:17:39 +10:00
|
|
|
case 39: // R(rd) = ~(R(rs) | R(rt)); break; //nor
|
2013-01-11 15:22:31 +01:00
|
|
|
gpr.MapDirtyInIn(rd, rs, rt);
|
2013-02-14 00:02:09 +01:00
|
|
|
if (rt == 0) {
|
|
|
|
MVN(gpr.R(rd), gpr.R(rs));
|
|
|
|
} else {
|
|
|
|
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
|
2013-01-11 15:22:31 +01:00
|
|
|
gpr.MapDirtyInIn(rd, rs, rt);
|
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);
|
2013-01-30 21:49:20 +01:00
|
|
|
MOVI2R(gpr.R(rd), 1);
|
2012-11-01 16:19:01 +01:00
|
|
|
SetCC(CC_GE);
|
2013-01-30 21:49:20 +01:00
|
|
|
MOVI2R(gpr.R(rd), 0);
|
2012-11-01 16:19:01 +01:00
|
|
|
SetCC(CC_AL);
|
|
|
|
break;
|
|
|
|
|
2013-03-01 02:17:39 +10:00
|
|
|
case 43: //R(rd) = R(rs) < R(rt); break; //sltu
|
2013-01-11 15:22:31 +01:00
|
|
|
gpr.MapDirtyInIn(rd, rs, rt);
|
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);
|
2013-01-30 21:49:20 +01:00
|
|
|
MOVI2R(gpr.R(rd), 1);
|
2012-11-01 16:19:01 +01:00
|
|
|
SetCC(CC_HS);
|
2013-01-30 21:49:20 +01:00
|
|
|
MOVI2R(gpr.R(rd), 0);
|
2012-11-01 16:19:01 +01:00
|
|
|
SetCC(CC_AL);
|
|
|
|
break;
|
|
|
|
|
2013-03-01 02:17:39 +10:00
|
|
|
case 44: //R(rd) = max(R(rs), R(rt); break; //max
|
|
|
|
gpr.MapDirtyInIn(rd, rs, rt);
|
|
|
|
CMP(gpr.R(rs), gpr.R(rt));
|
|
|
|
SetCC(CC_GT);
|
|
|
|
MOV(gpr.R(rd), gpr.R(rs));
|
|
|
|
SetCC(CC_LE);
|
|
|
|
MOV(gpr.R(rd), gpr.R(rt));
|
|
|
|
SetCC(CC_AL);
|
|
|
|
break;
|
2012-11-01 16:19:01 +01:00
|
|
|
|
2013-03-01 02:17:39 +10:00
|
|
|
case 45: //R(rd) = min(R(rs), R(rt)); break; //min
|
|
|
|
gpr.MapDirtyInIn(rd, rs, rt);
|
|
|
|
CMP(gpr.R(rs), gpr.R(rt));
|
|
|
|
SetCC(CC_LT);
|
|
|
|
MOV(gpr.R(rd), gpr.R(rs));
|
|
|
|
SetCC(CC_GE);
|
|
|
|
MOV(gpr.R(rd), gpr.R(rt));
|
|
|
|
SetCC(CC_AL);
|
|
|
|
break;
|
2012-11-01 16:19:01 +01:00
|
|
|
|
|
|
|
default:
|
2012-11-23 19:41:35 +01:00
|
|
|
// gpr.UnlockAll();
|
2012-11-01 16:19:01 +01:00
|
|
|
Comp_Generic(op);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-01-10 01:08:24 +01:00
|
|
|
void Jit::CompShiftImm(u32 op, ArmGen::ShiftType shiftType)
|
2012-11-01 16:19:01 +01:00
|
|
|
{
|
|
|
|
int rd = _RD;
|
|
|
|
int rt = _RT;
|
|
|
|
int sa = _SA;
|
2013-01-10 01:08:24 +01:00
|
|
|
|
2013-01-11 17:25:54 +01:00
|
|
|
gpr.MapDirtyIn(rd, rt);
|
2013-01-10 01:08:24 +01:00
|
|
|
MOV(gpr.R(rd), Operand2(sa, shiftType, gpr.R(rt)));
|
2012-11-01 16:19:01 +01:00
|
|
|
}
|
2013-01-10 01:08:24 +01:00
|
|
|
|
2013-03-05 13:52:03 +10:00
|
|
|
void Jit::CompShiftVar(u32 op, ArmGen::ShiftType shiftType)
|
2012-11-01 16:19:01 +01:00
|
|
|
{
|
|
|
|
int rd = _RD;
|
|
|
|
int rt = _RT;
|
|
|
|
int rs = _RS;
|
2013-03-05 13:52:03 +10:00
|
|
|
if (gpr.IsImm(rs))
|
|
|
|
{
|
2013-03-05 14:55:33 +10:00
|
|
|
gpr.MapDirtyIn(rd, rt);
|
|
|
|
int sa = gpr.GetImm(rs) & 0x1F;
|
2013-03-05 13:52:03 +10:00
|
|
|
MOV(gpr.R(rd), Operand2(sa, shiftType, gpr.R(rt)));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
gpr.MapDirtyInIn(rd, rs, rt);
|
|
|
|
AND(R0, gpr.R(rs), Operand2(0x1F));
|
|
|
|
MOV(gpr.R(rd), Operand2(gpr.R(rt), shiftType, R0));
|
2012-11-01 16:19:01 +01:00
|
|
|
}
|
2013-03-05 13:52:03 +10:00
|
|
|
|
2012-11-01 16:19:01 +01:00
|
|
|
void Jit::Comp_ShiftType(u32 op)
|
|
|
|
{
|
2013-03-01 10:34:49 -08:00
|
|
|
CONDITIONAL_DISABLE;
|
2013-02-24 22:58:31 -08:00
|
|
|
int rs = _RS;
|
|
|
|
int fd = _FD;
|
2013-03-05 13:52:03 +10:00
|
|
|
// WARNING : ROTR
|
2012-11-01 16:19:01 +01:00
|
|
|
switch (op & 0x3f)
|
|
|
|
{
|
2013-03-05 13:52:03 +10:00
|
|
|
case 0: CompShiftImm(op, ST_LSL); break; //sll
|
|
|
|
case 2: CompShiftImm(op, rs == 1 ? ST_ROR : ST_LSR); break; //srl
|
|
|
|
case 3: CompShiftImm(op, ST_ASR); break; //sra
|
|
|
|
case 4: CompShiftVar(op, ST_LSL); break; //sllv
|
|
|
|
case 6: CompShiftVar(op, rs == 1 ? ST_ROR : ST_LSR); break; //srlv
|
|
|
|
case 7: CompShiftVar(op, ST_ASR); break; //srav
|
2012-11-01 16:19:01 +01:00
|
|
|
|
|
|
|
default:
|
|
|
|
Comp_Generic(op);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-02-10 12:14:55 +01:00
|
|
|
void Jit::Comp_Special3(u32 op)
|
|
|
|
{
|
2013-03-03 15:29:13 +01:00
|
|
|
CONDITIONAL_DISABLE;
|
2013-03-03 16:40:58 +01:00
|
|
|
|
2013-03-03 23:17:21 +01:00
|
|
|
bool useUBFXandBFI = false;
|
2013-03-03 22:26:20 +01:00
|
|
|
|
2013-03-03 16:40:58 +01:00
|
|
|
if (!cpu_info.bArmV7) {
|
2013-03-03 23:17:21 +01:00
|
|
|
// useUBFXandBFI = true;
|
2013-03-03 16:40:58 +01:00
|
|
|
}
|
|
|
|
|
2013-03-03 15:29:13 +01:00
|
|
|
int rs = _RS;
|
|
|
|
int rt = _RT;
|
|
|
|
|
2013-03-03 16:40:58 +01:00
|
|
|
int pos = _POS;
|
2013-03-03 15:29:13 +01:00
|
|
|
int size = _SIZE + 1;
|
|
|
|
u32 mask = 0xFFFFFFFFUL >> (32 - size);
|
|
|
|
|
|
|
|
// Don't change $zr.
|
|
|
|
if (rt == 0)
|
|
|
|
return;
|
|
|
|
|
|
|
|
switch (op & 0x3f)
|
|
|
|
{
|
|
|
|
case 0x0: //ext
|
|
|
|
if (gpr.IsImm(rs))
|
|
|
|
{
|
|
|
|
gpr.SetImm(rt, (gpr.GetImm(rs) >> pos) & mask);
|
|
|
|
return;
|
|
|
|
}
|
2013-03-03 23:17:21 +01:00
|
|
|
|
2013-03-03 16:40:58 +01:00
|
|
|
gpr.MapDirtyIn(rt, rs, false);
|
2013-03-03 23:17:21 +01:00
|
|
|
if (useUBFXandBFI) {
|
|
|
|
UBFX(gpr.R(rt), gpr.R(rs), pos, size);
|
|
|
|
} else {
|
|
|
|
MOV(gpr.R(rt), Operand2(pos, ST_LSR, gpr.R(rs)));
|
|
|
|
ANDI2R(gpr.R(rt), gpr.R(rt), mask, R0);
|
|
|
|
}
|
2013-03-03 15:29:13 +01:00
|
|
|
break;
|
|
|
|
|
|
|
|
case 0x4: //ins
|
2013-03-03 16:40:58 +01:00
|
|
|
{
|
2013-03-04 22:09:45 +10:00
|
|
|
DISABLE;
|
2013-03-03 16:40:58 +01:00
|
|
|
u32 sourcemask = mask >> pos;
|
|
|
|
u32 destmask = ~(sourcemask << pos);
|
2013-03-03 23:17:21 +01:00
|
|
|
if (gpr.IsImm(rs))
|
|
|
|
{
|
|
|
|
u32 inserted = (gpr.GetImm(rs) & sourcemask) << pos;
|
|
|
|
if (gpr.IsImm(rt))
|
|
|
|
{
|
|
|
|
gpr.SetImm(rt, (gpr.GetImm(rt) & destmask) | inserted);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2013-03-04 00:06:33 +01:00
|
|
|
gpr.MapReg(rt, MAP_DIRTY);
|
2013-03-03 23:17:21 +01:00
|
|
|
ANDI2R(gpr.R(rt), gpr.R(rt), destmask, R0);
|
|
|
|
ORI2R(gpr.R(rt), gpr.R(rt), inserted, R0);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (useUBFXandBFI) {
|
|
|
|
gpr.MapDirtyIn(rt, rs, false);
|
|
|
|
BFI(gpr.R(rt), gpr.R(rs), pos, size);
|
|
|
|
} else {
|
|
|
|
gpr.MapDirtyIn(rt, rs, false);
|
|
|
|
ANDI2R(R0, gpr.R(rs), sourcemask, R1);
|
2013-03-04 22:09:45 +10:00
|
|
|
MOV(R0, Operand2(pos, ST_LSL, R0));
|
2013-03-03 23:17:21 +01:00
|
|
|
ANDI2R(gpr.R(rt), gpr.R(rt), destmask, R1);
|
|
|
|
ORR(gpr.R(rt), gpr.R(rt), R0);
|
|
|
|
}
|
|
|
|
}
|
2013-03-03 16:40:58 +01:00
|
|
|
}
|
2013-03-03 15:29:13 +01:00
|
|
|
break;
|
|
|
|
}
|
2013-02-10 12:14:55 +01:00
|
|
|
}
|
|
|
|
|
2012-11-01 16:19:01 +01:00
|
|
|
void Jit::Comp_Allegrex(u32 op)
|
|
|
|
{
|
2013-03-01 10:34:49 -08:00
|
|
|
CONDITIONAL_DISABLE;
|
2012-11-01 16:19:01 +01:00
|
|
|
int rt = _RT;
|
|
|
|
int rd = _RD;
|
2013-03-03 15:29:13 +01:00
|
|
|
// Don't change $zr.
|
|
|
|
if (rd == 0)
|
|
|
|
return;
|
|
|
|
|
2012-11-01 16:19:01 +01:00
|
|
|
switch ((op >> 6) & 31)
|
|
|
|
{
|
|
|
|
case 16: // seb // R(rd) = (u32)(s32)(s8)(u8)R(rt);
|
2013-03-03 15:29:13 +01:00
|
|
|
if (gpr.IsImm(rt))
|
|
|
|
{
|
|
|
|
gpr.SetImm(rd, (s32)(s8)(u8)gpr.GetImm(rt));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
gpr.MapDirtyIn(rd, rt);
|
|
|
|
SXTB(gpr.R(rd), gpr.R(rt));
|
2012-11-01 16:19:01 +01:00
|
|
|
break;
|
|
|
|
|
|
|
|
case 24: // seh
|
2013-03-03 15:29:13 +01:00
|
|
|
if (gpr.IsImm(rt))
|
|
|
|
{
|
|
|
|
gpr.SetImm(rd, (s32)(s16)(u16)gpr.GetImm(rt));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
gpr.MapDirtyIn(rd, rt);
|
|
|
|
SXTH(gpr.R(rd), gpr.R(rt));
|
2012-11-01 16:19:01 +01:00
|
|
|
break;
|
2013-03-03 15:29:13 +01:00
|
|
|
|
2012-11-01 16:19:01 +01:00
|
|
|
case 20: //bitrev
|
2013-02-28 13:59:33 +10:00
|
|
|
if (gpr.IsImm(rt))
|
|
|
|
{
|
|
|
|
// http://graphics.stanford.edu/~seander/bithacks.html#ReverseParallel
|
|
|
|
u32 v = gpr.GetImm(rt);
|
2013-03-05 02:58:51 +10:00
|
|
|
v = ((v >> 1) & 0x55555555) | ((v & 0x55555555) << 1); // odd<->even
|
|
|
|
v = ((v >> 2) & 0x33333333) | ((v & 0x33333333) << 2); // pair<->pair
|
|
|
|
v = ((v >> 4) & 0x0F0F0F0F) | ((v & 0x0F0F0F0F) << 4); // nibb<->nibb
|
|
|
|
v = ((v >> 8) & 0x00FF00FF) | ((v & 0x00FF00FF) << 8); // byte<->byte
|
|
|
|
v = ( v >> 16 ) | ( v << 16); // hword<->hword
|
2013-03-03 15:29:13 +01:00
|
|
|
gpr.SetImm(rd, v);
|
2013-03-05 02:58:51 +10:00
|
|
|
return;
|
2013-02-28 13:59:33 +10:00
|
|
|
}
|
2013-03-05 02:58:51 +10:00
|
|
|
|
|
|
|
if (cpu_info.bArmV7) {
|
|
|
|
gpr.MapDirtyIn(rd, rt);
|
2013-03-05 03:13:33 +10:00
|
|
|
RBIT(gpr.R(rd), gpr.R(rt));
|
2013-03-05 02:58:51 +10:00
|
|
|
} else {
|
|
|
|
Comp_Generic(op);
|
|
|
|
}
|
|
|
|
break;
|
2012-11-01 16:19:01 +01:00
|
|
|
default:
|
|
|
|
Comp_Generic(op);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-01-29 00:48:42 +01:00
|
|
|
void Jit::Comp_MulDivType(u32 op)
|
|
|
|
{
|
2013-03-01 10:34:49 -08:00
|
|
|
CONDITIONAL_DISABLE;
|
2013-01-30 01:06:14 +01:00
|
|
|
int rt = _RT;
|
|
|
|
int rs = _RS;
|
|
|
|
int rd = _RD;
|
|
|
|
|
|
|
|
switch (op & 63)
|
|
|
|
{
|
|
|
|
case 16: // R(rd) = HI; //mfhi
|
|
|
|
gpr.MapDirtyIn(rd, MIPSREG_HI);
|
|
|
|
MOV(gpr.R(rd), gpr.R(MIPSREG_HI));
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 17: // HI = R(rs); //mthi
|
|
|
|
gpr.MapDirtyIn(MIPSREG_HI, rs);
|
|
|
|
MOV(gpr.R(MIPSREG_HI), gpr.R(rs));
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 18: // R(rd) = LO; break; //mflo
|
|
|
|
gpr.MapDirtyIn(rd, MIPSREG_LO);
|
|
|
|
MOV(gpr.R(rd), gpr.R(MIPSREG_LO));
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 19: // LO = R(rs); break; //mtlo
|
|
|
|
gpr.MapDirtyIn(MIPSREG_LO, rs);
|
|
|
|
MOV(gpr.R(MIPSREG_LO), gpr.R(rs));
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 24: //mult (the most popular one). lo,hi = signed mul (rs * rt)
|
|
|
|
gpr.MapDirtyDirtyInIn(MIPSREG_LO, MIPSREG_HI, rs, rt);
|
|
|
|
SMULL(gpr.R(MIPSREG_LO), gpr.R(MIPSREG_HI), gpr.R(rs), gpr.R(rt));
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 25: //multu (2nd) lo,hi = unsigned mul (rs * rt)
|
|
|
|
gpr.MapDirtyDirtyInIn(MIPSREG_LO, MIPSREG_HI, rs, rt);
|
|
|
|
UMULL(gpr.R(MIPSREG_LO), gpr.R(MIPSREG_HI), gpr.R(rs), gpr.R(rt));
|
2013-02-28 13:59:33 +10:00
|
|
|
break;
|
|
|
|
|
2013-03-05 13:16:08 +10:00
|
|
|
case 26: //div
|
|
|
|
DISABLE;
|
|
|
|
gpr.MapDirtyDirtyInIn(MIPSREG_LO, MIPSREG_HI, rs, rt);
|
|
|
|
if (cpu_info.bIDIVa)
|
|
|
|
{
|
|
|
|
SDIV(gpr.R(MIPSREG_LO), gpr.R(rs), gpr.R(rt));
|
2013-03-05 15:14:22 +10:00
|
|
|
MUL(R0, gpr.R(rt), gpr.R(MIPSREG_LO));
|
|
|
|
SUB(gpr.R(MIPSREG_HI), gpr.R(rs), Operand2(R0));
|
2013-03-05 13:16:08 +10:00
|
|
|
} else {
|
|
|
|
VMOV(S0, gpr.R(rs));
|
|
|
|
VMOV(S1, gpr.R(rt));
|
|
|
|
VCVT(D0, S0, TO_FLOAT | IS_SIGNED);
|
|
|
|
VCVT(D1, S1, TO_FLOAT | IS_SIGNED);
|
|
|
|
VDIV(D0, D0, D1);
|
|
|
|
VCVT(gpr.R(MIPSREG_LO), D0, TO_INT | IS_SIGNED);
|
2013-03-05 15:14:22 +10:00
|
|
|
MUL(R0, gpr.R(rt), gpr.R(MIPSREG_LO));
|
|
|
|
SUB(gpr.R(MIPSREG_HI), gpr.R(rs), Operand2(R0));
|
2013-03-05 13:16:08 +10:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 27: //divu
|
|
|
|
DISABLE;
|
|
|
|
gpr.MapDirtyDirtyInIn(MIPSREG_LO, MIPSREG_HI, rs, rt);
|
|
|
|
if (cpu_info.bIDIVa)
|
|
|
|
{
|
|
|
|
UDIV(gpr.R(MIPSREG_LO), gpr.R(rs), gpr.R(rt));
|
2013-03-05 15:14:22 +10:00
|
|
|
MUL(R0, gpr.R(rt), gpr.R(MIPSREG_LO));
|
|
|
|
SUB(gpr.R(MIPSREG_HI), gpr.R(rs), Operand2(R0));
|
2013-03-05 13:16:08 +10:00
|
|
|
} else {
|
|
|
|
VMOV(S0, gpr.R(rs));
|
|
|
|
VMOV(S1, gpr.R(rt));
|
|
|
|
VCVT(D0, S0, TO_FLOAT);
|
|
|
|
VCVT(D1, S1, TO_FLOAT);
|
|
|
|
VDIV(D0, D0, D1);
|
|
|
|
VCVT(gpr.R(MIPSREG_LO), D0, TO_INT);
|
2013-03-05 15:14:22 +10:00
|
|
|
MUL(R0, gpr.R(rt), gpr.R(MIPSREG_LO));
|
|
|
|
SUB(gpr.R(MIPSREG_HI), gpr.R(rs), Operand2(R0));
|
2013-03-05 13:16:08 +10:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2013-02-28 13:59:33 +10:00
|
|
|
case 28: //madd
|
2013-02-28 23:44:40 +01:00
|
|
|
gpr.MapDirtyDirtyInIn(MIPSREG_LO, MIPSREG_HI, rs, rt, false);
|
2013-02-28 13:59:33 +10:00
|
|
|
SMLAL(gpr.R(MIPSREG_LO), gpr.R(MIPSREG_HI), gpr.R(rs), gpr.R(rt));
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 29: //maddu
|
2013-02-28 23:44:40 +01:00
|
|
|
gpr.MapDirtyDirtyInIn(MIPSREG_LO, MIPSREG_HI, rs, rt, false);
|
2013-02-28 13:59:33 +10:00
|
|
|
UMLAL(gpr.R(MIPSREG_LO), gpr.R(MIPSREG_HI), gpr.R(rs), gpr.R(rt));
|
|
|
|
break;
|
2013-01-30 01:06:14 +01:00
|
|
|
|
2013-03-05 13:16:08 +10:00
|
|
|
case 46: // msub
|
|
|
|
DISABLE;
|
|
|
|
gpr.MapDirtyDirtyInIn(MIPSREG_LO, MIPSREG_HI, rs, rt, false);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 47: // msubu
|
|
|
|
DISABLE;
|
|
|
|
gpr.MapDirtyDirtyInIn(MIPSREG_LO, MIPSREG_HI, rs, rt, false);
|
|
|
|
break;
|
|
|
|
|
2013-01-30 01:06:14 +01:00
|
|
|
default:
|
|
|
|
DISABLE;
|
|
|
|
}
|
2013-01-29 00:48:42 +01:00
|
|
|
}
|
|
|
|
|
2012-11-01 16:19:01 +01:00
|
|
|
}
|