JitArm64: Fix special cases of cmp
This fixes a regression from 592ba31
. When `a` was a constant 0 and `b`
was a non-constant 0x80000000, the 32-bit negation operation would
overflow, causing an incorrect result. The sign extension needs to happen
before the negation to avoid overflow.
Note that I can't merge the SXTW and NEG into one instruction.
NEG is an alias for SUB with the first operand being set to ZR,
but "SUB (extended register)" treats register 31 as SP instead of ZR.
I've also changed the order for the case where `a` is a constant
0xFFFFFFFF. I don't think the order actually affects correctness here,
but let's use the same order for all the cases since it makes the code
easier to reason about.
This commit is contained in:
parent
9d139ea84e
commit
0049a76775
|
@ -584,13 +584,13 @@ void JitArm64::cmp(UGeckoInstruction inst)
|
||||||
}
|
}
|
||||||
else if (gpr.IsImm(a) && !gpr.GetImm(a))
|
else if (gpr.IsImm(a) && !gpr.GetImm(a))
|
||||||
{
|
{
|
||||||
NEG(EncodeRegTo32(CR), gpr.R(b));
|
SXTW(CR, gpr.R(b));
|
||||||
SXTW(CR, EncodeRegTo32(CR));
|
NEG(CR, CR);
|
||||||
}
|
}
|
||||||
else if (gpr.IsImm(a) && gpr.GetImm(a) == 0xFFFFFFFF)
|
else if (gpr.IsImm(a) && gpr.GetImm(a) == 0xFFFFFFFF)
|
||||||
{
|
{
|
||||||
MVN(EncodeRegTo32(CR), gpr.R(b));
|
SXTW(CR, gpr.R(b));
|
||||||
SXTW(CR, EncodeRegTo32(CR));
|
MVN(CR, CR);
|
||||||
}
|
}
|
||||||
else if (gpr.IsImm(b) && !gpr.GetImm(b))
|
else if (gpr.IsImm(b) && !gpr.GetImm(b))
|
||||||
{
|
{
|
||||||
|
|
Loading…
Reference in New Issue