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:
JosJuice 2023-02-14 18:39:00 +01:00
parent 9d139ea84e
commit 0049a76775
1 changed files with 4 additions and 4 deletions

View File

@ -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))
{ {