mirror of
https://github.com/dolphin-emu/dolphin.git
synced 2025-01-25 07:21:14 +01:00
Removed an extraneous FlushLockX, further optimized simultaneous handling of carry/overflow.
This commit is contained in:
parent
4cb1af0f13
commit
a6d041bfa9
@ -149,7 +149,7 @@ public:
|
|||||||
|
|
||||||
void GenerateConstantOverflow(bool overflow);
|
void GenerateConstantOverflow(bool overflow);
|
||||||
void GenerateOverflow();
|
void GenerateOverflow();
|
||||||
void GenerateOverflowFinalizeCarry(bool oe, bool inv = false);
|
void FinalizeCarryOverflow(bool oe, bool inv = false);
|
||||||
void GenerateCarry();
|
void GenerateCarry();
|
||||||
void GenerateRC();
|
void GenerateRC();
|
||||||
void ComputeRC(const Gen::OpArg & arg);
|
void ComputeRC(const Gen::OpArg & arg);
|
||||||
|
@ -50,8 +50,8 @@ void Jit64::GenerateOverflow()
|
|||||||
SetJumpTarget(exit);
|
SetJumpTarget(exit);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Assumes CA is clear
|
// Assumes CA,OV are clear
|
||||||
void Jit64::GenerateOverflowFinalizeCarry(bool oe, bool inv)
|
void Jit64::FinalizeCarryOverflow(bool oe, bool inv)
|
||||||
{
|
{
|
||||||
// USES_XER
|
// USES_XER
|
||||||
if (oe)
|
if (oe)
|
||||||
@ -69,8 +69,6 @@ void Jit64::GenerateOverflowFinalizeCarry(bool oe, bool inv)
|
|||||||
FixupBranch carry2 = J_CC(inv ? CC_C : CC_NC);
|
FixupBranch carry2 = J_CC(inv ? CC_C : CC_NC);
|
||||||
JitSetCA();
|
JitSetCA();
|
||||||
SetJumpTarget(carry2);
|
SetJumpTarget(carry2);
|
||||||
//XER[OV] = 0
|
|
||||||
AND(32, M(&PowerPC::ppcState.spr[SPR_XER]), Imm32(~XER_OV_MASK));
|
|
||||||
SetJumpTarget(exit);
|
SetJumpTarget(exit);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
@ -878,7 +876,7 @@ void Jit64::subfcx(UGeckoInstruction inst)
|
|||||||
gpr.Lock(a, b, d);
|
gpr.Lock(a, b, d);
|
||||||
gpr.BindToRegister(d, (d == a || d == b), true);
|
gpr.BindToRegister(d, (d == a || d == b), true);
|
||||||
|
|
||||||
JitClearCA();
|
JitClearCAOV(inst.OE);
|
||||||
if (d == b)
|
if (d == b)
|
||||||
{
|
{
|
||||||
SUB(32, gpr.R(d), gpr.R(a));
|
SUB(32, gpr.R(d), gpr.R(a));
|
||||||
@ -897,7 +895,7 @@ void Jit64::subfcx(UGeckoInstruction inst)
|
|||||||
if (inst.Rc) {
|
if (inst.Rc) {
|
||||||
GenerateRC();
|
GenerateRC();
|
||||||
}
|
}
|
||||||
GenerateOverflowFinalizeCarry(inst.OE, true);
|
FinalizeCarryOverflow(inst.OE, true);
|
||||||
|
|
||||||
gpr.UnlockAll();
|
gpr.UnlockAll();
|
||||||
}
|
}
|
||||||
@ -907,13 +905,12 @@ void Jit64::subfex(UGeckoInstruction inst)
|
|||||||
INSTRUCTION_START;
|
INSTRUCTION_START;
|
||||||
JITDISABLE(Integer)
|
JITDISABLE(Integer)
|
||||||
int a = inst.RA, b = inst.RB, d = inst.RD;
|
int a = inst.RA, b = inst.RB, d = inst.RD;
|
||||||
gpr.FlushLockX(ECX);
|
|
||||||
gpr.Lock(a, b, d);
|
gpr.Lock(a, b, d);
|
||||||
gpr.BindToRegister(d, (d == a || d == b), true);
|
gpr.BindToRegister(d, (d == a || d == b), true);
|
||||||
|
|
||||||
// Get CA and clear it
|
// Get CA and clear it (along with OV if applicable)
|
||||||
MOV(32, R(EAX), M(&PowerPC::ppcState.spr[SPR_XER]));
|
MOV(32, R(EAX), M(&PowerPC::ppcState.spr[SPR_XER]));
|
||||||
JitClearCA();
|
JitClearCAOV(inst.OE);
|
||||||
SHR(32, R(EAX), Imm8(30));
|
SHR(32, R(EAX), Imm8(30));
|
||||||
|
|
||||||
// Convert carry to borrow
|
// Convert carry to borrow
|
||||||
@ -937,10 +934,9 @@ void Jit64::subfex(UGeckoInstruction inst)
|
|||||||
if (inst.Rc) {
|
if (inst.Rc) {
|
||||||
GenerateRC();
|
GenerateRC();
|
||||||
}
|
}
|
||||||
GenerateOverflowFinalizeCarry(inst.OE, true);
|
FinalizeCarryOverflow(inst.OE, true);
|
||||||
|
|
||||||
gpr.UnlockAll();
|
gpr.UnlockAll();
|
||||||
gpr.UnlockAllX();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void Jit64::subfmex(UGeckoInstruction inst)
|
void Jit64::subfmex(UGeckoInstruction inst)
|
||||||
@ -953,7 +949,7 @@ void Jit64::subfmex(UGeckoInstruction inst)
|
|||||||
gpr.BindToRegister(d, d == a);
|
gpr.BindToRegister(d, d == a);
|
||||||
|
|
||||||
MOV(32, R(EAX), M(&PowerPC::ppcState.spr[SPR_XER]));
|
MOV(32, R(EAX), M(&PowerPC::ppcState.spr[SPR_XER]));
|
||||||
JitClearCA();
|
JitClearCAOV(inst.OE);
|
||||||
SHR(32, R(EAX), Imm8(30)); // shift the carry flag out into the x86 carry flag
|
SHR(32, R(EAX), Imm8(30)); // shift the carry flag out into the x86 carry flag
|
||||||
if (d != a)
|
if (d != a)
|
||||||
{
|
{
|
||||||
@ -965,7 +961,7 @@ void Jit64::subfmex(UGeckoInstruction inst)
|
|||||||
{
|
{
|
||||||
GenerateRC();
|
GenerateRC();
|
||||||
}
|
}
|
||||||
GenerateOverflowFinalizeCarry(inst.OE);
|
FinalizeCarryOverflow(inst.OE);
|
||||||
gpr.UnlockAll();
|
gpr.UnlockAll();
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -979,7 +975,7 @@ void Jit64::subfzex(UGeckoInstruction inst)
|
|||||||
gpr.Lock(a, d);
|
gpr.Lock(a, d);
|
||||||
gpr.BindToRegister(d, d == a);
|
gpr.BindToRegister(d, d == a);
|
||||||
MOV(32, R(EAX), M(&PowerPC::ppcState.spr[SPR_XER]));
|
MOV(32, R(EAX), M(&PowerPC::ppcState.spr[SPR_XER]));
|
||||||
JitClearCA();
|
JitClearCAOV(inst.OE);
|
||||||
SHR(32, R(EAX), Imm8(30)); // shift the carry flag out into the x86 carry flag
|
SHR(32, R(EAX), Imm8(30)); // shift the carry flag out into the x86 carry flag
|
||||||
if (d != a)
|
if (d != a)
|
||||||
{
|
{
|
||||||
@ -991,7 +987,7 @@ void Jit64::subfzex(UGeckoInstruction inst)
|
|||||||
{
|
{
|
||||||
GenerateRC();
|
GenerateRC();
|
||||||
}
|
}
|
||||||
GenerateOverflowFinalizeCarry(inst.OE);
|
FinalizeCarryOverflow(inst.OE);
|
||||||
|
|
||||||
gpr.UnlockAll();
|
gpr.UnlockAll();
|
||||||
}
|
}
|
||||||
@ -1269,14 +1265,14 @@ void Jit64::addex(UGeckoInstruction inst)
|
|||||||
gpr.Lock(a, b, d);
|
gpr.Lock(a, b, d);
|
||||||
gpr.BindToRegister(d, true);
|
gpr.BindToRegister(d, true);
|
||||||
MOV(32, R(EAX), M(&PowerPC::ppcState.spr[SPR_XER]));
|
MOV(32, R(EAX), M(&PowerPC::ppcState.spr[SPR_XER]));
|
||||||
JitClearCA();
|
JitClearCAOV(inst.OE);
|
||||||
SHR(32, R(EAX), Imm8(30)); // shift the carry flag out into the x86 carry flag
|
SHR(32, R(EAX), Imm8(30)); // shift the carry flag out into the x86 carry flag
|
||||||
ADC(32, gpr.R(d), gpr.R((d == a) ? b : a));
|
ADC(32, gpr.R(d), gpr.R((d == a) ? b : a));
|
||||||
if (inst.Rc)
|
if (inst.Rc)
|
||||||
{
|
{
|
||||||
GenerateRC();
|
GenerateRC();
|
||||||
}
|
}
|
||||||
GenerateOverflowFinalizeCarry(inst.OE);
|
FinalizeCarryOverflow(inst.OE);
|
||||||
gpr.UnlockAll();
|
gpr.UnlockAll();
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
@ -1284,7 +1280,7 @@ void Jit64::addex(UGeckoInstruction inst)
|
|||||||
gpr.Lock(a, b, d);
|
gpr.Lock(a, b, d);
|
||||||
gpr.BindToRegister(d, false);
|
gpr.BindToRegister(d, false);
|
||||||
MOV(32, R(EAX), M(&PowerPC::ppcState.spr[SPR_XER]));
|
MOV(32, R(EAX), M(&PowerPC::ppcState.spr[SPR_XER]));
|
||||||
JitClearCA();
|
JitClearCAOV(inst.OE);
|
||||||
SHR(32, R(EAX), Imm8(30)); // shift the carry flag out into the x86 carry flag
|
SHR(32, R(EAX), Imm8(30)); // shift the carry flag out into the x86 carry flag
|
||||||
MOV(32, gpr.R(d), gpr.R(a));
|
MOV(32, gpr.R(d), gpr.R(a));
|
||||||
ADC(32, gpr.R(d), gpr.R(b));
|
ADC(32, gpr.R(d), gpr.R(b));
|
||||||
@ -1292,7 +1288,7 @@ void Jit64::addex(UGeckoInstruction inst)
|
|||||||
{
|
{
|
||||||
GenerateRC();
|
GenerateRC();
|
||||||
}
|
}
|
||||||
GenerateOverflowFinalizeCarry(inst.OE);
|
FinalizeCarryOverflow(inst.OE);
|
||||||
gpr.UnlockAll();
|
gpr.UnlockAll();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -1308,27 +1304,27 @@ void Jit64::addcx(UGeckoInstruction inst)
|
|||||||
int operand = ((d == a) ? b : a);
|
int operand = ((d == a) ? b : a);
|
||||||
gpr.Lock(a, b, d);
|
gpr.Lock(a, b, d);
|
||||||
gpr.BindToRegister(d, true);
|
gpr.BindToRegister(d, true);
|
||||||
JitClearCA();
|
JitClearCAOV(inst.OE);
|
||||||
ADD(32, gpr.R(d), gpr.R(operand));
|
ADD(32, gpr.R(d), gpr.R(operand));
|
||||||
if (inst.Rc)
|
if (inst.Rc)
|
||||||
{
|
{
|
||||||
GenerateRC();
|
GenerateRC();
|
||||||
}
|
}
|
||||||
GenerateOverflowFinalizeCarry(inst.OE);
|
FinalizeCarryOverflow(inst.OE);
|
||||||
gpr.UnlockAll();
|
gpr.UnlockAll();
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
gpr.Lock(a, b, d);
|
gpr.Lock(a, b, d);
|
||||||
gpr.BindToRegister(d, false);
|
gpr.BindToRegister(d, false);
|
||||||
JitClearCA();
|
JitClearCAOV(inst.OE);
|
||||||
MOV(32, gpr.R(d), gpr.R(a));
|
MOV(32, gpr.R(d), gpr.R(a));
|
||||||
ADD(32, gpr.R(d), gpr.R(b));
|
ADD(32, gpr.R(d), gpr.R(b));
|
||||||
if (inst.Rc)
|
if (inst.Rc)
|
||||||
{
|
{
|
||||||
GenerateRC();
|
GenerateRC();
|
||||||
}
|
}
|
||||||
GenerateOverflowFinalizeCarry(inst.OE);
|
FinalizeCarryOverflow(inst.OE);
|
||||||
gpr.UnlockAll();
|
gpr.UnlockAll();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -1345,14 +1341,14 @@ void Jit64::addmex(UGeckoInstruction inst)
|
|||||||
gpr.Lock(d);
|
gpr.Lock(d);
|
||||||
gpr.BindToRegister(d, true);
|
gpr.BindToRegister(d, true);
|
||||||
MOV(32, R(EAX), M(&PowerPC::ppcState.spr[SPR_XER]));
|
MOV(32, R(EAX), M(&PowerPC::ppcState.spr[SPR_XER]));
|
||||||
JitClearCA();
|
JitClearCAOV(inst.OE);
|
||||||
SHR(32, R(EAX), Imm8(30)); // shift the carry flag out into the x86 carry flag
|
SHR(32, R(EAX), Imm8(30)); // shift the carry flag out into the x86 carry flag
|
||||||
ADC(32, gpr.R(d), Imm32(0xFFFFFFFF));
|
ADC(32, gpr.R(d), Imm32(0xFFFFFFFF));
|
||||||
if (inst.Rc)
|
if (inst.Rc)
|
||||||
{
|
{
|
||||||
GenerateRC();
|
GenerateRC();
|
||||||
}
|
}
|
||||||
GenerateOverflowFinalizeCarry(inst.OE);
|
FinalizeCarryOverflow(inst.OE);
|
||||||
gpr.UnlockAll();
|
gpr.UnlockAll();
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
@ -1360,7 +1356,7 @@ void Jit64::addmex(UGeckoInstruction inst)
|
|||||||
gpr.Lock(a, d);
|
gpr.Lock(a, d);
|
||||||
gpr.BindToRegister(d, false);
|
gpr.BindToRegister(d, false);
|
||||||
MOV(32, R(EAX), M(&PowerPC::ppcState.spr[SPR_XER]));
|
MOV(32, R(EAX), M(&PowerPC::ppcState.spr[SPR_XER]));
|
||||||
JitClearCA();
|
JitClearCAOV(inst.OE);
|
||||||
SHR(32, R(EAX), Imm8(30)); // shift the carry flag out into the x86 carry flag
|
SHR(32, R(EAX), Imm8(30)); // shift the carry flag out into the x86 carry flag
|
||||||
MOV(32, gpr.R(d), gpr.R(a));
|
MOV(32, gpr.R(d), gpr.R(a));
|
||||||
ADC(32, gpr.R(d), Imm32(0xFFFFFFFF));
|
ADC(32, gpr.R(d), Imm32(0xFFFFFFFF));
|
||||||
@ -1368,7 +1364,7 @@ void Jit64::addmex(UGeckoInstruction inst)
|
|||||||
{
|
{
|
||||||
GenerateRC();
|
GenerateRC();
|
||||||
}
|
}
|
||||||
GenerateOverflowFinalizeCarry(inst.OE);
|
FinalizeCarryOverflow(inst.OE);
|
||||||
gpr.UnlockAll();
|
gpr.UnlockAll();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -1385,14 +1381,14 @@ void Jit64::addzex(UGeckoInstruction inst)
|
|||||||
gpr.Lock(d);
|
gpr.Lock(d);
|
||||||
gpr.BindToRegister(d, true);
|
gpr.BindToRegister(d, true);
|
||||||
MOV(32, R(EAX), M(&PowerPC::ppcState.spr[SPR_XER]));
|
MOV(32, R(EAX), M(&PowerPC::ppcState.spr[SPR_XER]));
|
||||||
JitClearCA();
|
JitClearCAOV(inst.OE);
|
||||||
SHR(32, R(EAX), Imm8(30)); // shift the carry flag out into the x86 carry flag
|
SHR(32, R(EAX), Imm8(30)); // shift the carry flag out into the x86 carry flag
|
||||||
ADC(32, gpr.R(d), Imm8(0));
|
ADC(32, gpr.R(d), Imm8(0));
|
||||||
if (inst.Rc)
|
if (inst.Rc)
|
||||||
{
|
{
|
||||||
GenerateRC();
|
GenerateRC();
|
||||||
}
|
}
|
||||||
GenerateOverflowFinalizeCarry(inst.OE);
|
FinalizeCarryOverflow(inst.OE);
|
||||||
gpr.UnlockAll();
|
gpr.UnlockAll();
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
@ -1400,7 +1396,7 @@ void Jit64::addzex(UGeckoInstruction inst)
|
|||||||
gpr.Lock(a, d);
|
gpr.Lock(a, d);
|
||||||
gpr.BindToRegister(d, false);
|
gpr.BindToRegister(d, false);
|
||||||
MOV(32, R(EAX), M(&PowerPC::ppcState.spr[SPR_XER]));
|
MOV(32, R(EAX), M(&PowerPC::ppcState.spr[SPR_XER]));
|
||||||
JitClearCA();
|
JitClearCAOV(inst.OE);
|
||||||
SHR(32, R(EAX), Imm8(30)); // shift the carry flag out into the x86 carry flag
|
SHR(32, R(EAX), Imm8(30)); // shift the carry flag out into the x86 carry flag
|
||||||
MOV(32, gpr.R(d), gpr.R(a));
|
MOV(32, gpr.R(d), gpr.R(a));
|
||||||
ADC(32, gpr.R(d), Imm8(0));
|
ADC(32, gpr.R(d), Imm8(0));
|
||||||
@ -1408,7 +1404,7 @@ void Jit64::addzex(UGeckoInstruction inst)
|
|||||||
{
|
{
|
||||||
GenerateRC();
|
GenerateRC();
|
||||||
}
|
}
|
||||||
GenerateOverflowFinalizeCarry(inst.OE);
|
FinalizeCarryOverflow(inst.OE);
|
||||||
gpr.UnlockAll();
|
gpr.UnlockAll();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -352,3 +352,11 @@ void EmuCodeBlock::JitSetCA()
|
|||||||
{
|
{
|
||||||
OR(32, M(&PowerPC::ppcState.spr[SPR_XER]), Imm32(XER_CA_MASK)); //XER.CA = 1
|
OR(32, M(&PowerPC::ppcState.spr[SPR_XER]), Imm32(XER_CA_MASK)); //XER.CA = 1
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void EmuCodeBlock::JitClearCAOV(bool oe)
|
||||||
|
{
|
||||||
|
if (oe)
|
||||||
|
AND(32, M(&PowerPC::ppcState.spr[SPR_XER]), Imm32(~XER_CA_MASK & ~XER_OV_MASK)); //XER.CA, XER.OV = 0
|
||||||
|
else
|
||||||
|
AND(32, M(&PowerPC::ppcState.spr[SPR_XER]), Imm32(~XER_CA_MASK)); //XER.CA = 0
|
||||||
|
}
|
@ -38,6 +38,7 @@ public:
|
|||||||
void WriteFloatToConstRamAddress(const Gen::X64Reg& xmm_reg, u32 address);
|
void WriteFloatToConstRamAddress(const Gen::X64Reg& xmm_reg, u32 address);
|
||||||
void JitClearCA();
|
void JitClearCA();
|
||||||
void JitSetCA();
|
void JitSetCA();
|
||||||
|
void JitClearCAOV(bool oe);
|
||||||
|
|
||||||
void ForceSinglePrecisionS(Gen::X64Reg xmm);
|
void ForceSinglePrecisionS(Gen::X64Reg xmm);
|
||||||
void ForceSinglePrecisionP(Gen::X64Reg xmm);
|
void ForceSinglePrecisionP(Gen::X64Reg xmm);
|
||||||
|
Loading…
x
Reference in New Issue
Block a user