mirror of
https://github.com/yuzu-emu/unicorn.git
synced 2025-01-22 12:41:01 +00:00
tcg/arm: Fix double-word comparisons
The code sequence we were generating was only good for unsigned comparisons. For signed comparisions, use the sequence from gcc. Fixes booting of ppc64 firmware, with a patch changing the code sequence for ppc comparisons. Backports commit 7170ac33135e6ecf89752d3949bcecf9b9766d1c from qemu
This commit is contained in:
parent
4a680da17d
commit
9cd6985799
|
@ -1103,6 +1103,56 @@ static inline void tcg_out_mb(TCGContext *s, TCGArg a0)
|
|||
}
|
||||
}
|
||||
|
||||
static TCGCond tcg_out_cmp2(TCGContext *s, const TCGArg *args,
|
||||
const int *const_args)
|
||||
{
|
||||
TCGReg al = args[0];
|
||||
TCGReg ah = args[1];
|
||||
TCGArg bl = args[2];
|
||||
TCGArg bh = args[3];
|
||||
TCGCond cond = args[4];
|
||||
int const_bl = const_args[2];
|
||||
int const_bh = const_args[3];
|
||||
|
||||
switch (cond) {
|
||||
case TCG_COND_EQ:
|
||||
case TCG_COND_NE:
|
||||
case TCG_COND_LTU:
|
||||
case TCG_COND_LEU:
|
||||
case TCG_COND_GTU:
|
||||
case TCG_COND_GEU:
|
||||
/* We perform a conditional comparision. If the high half is
|
||||
equal, then overwrite the flags with the comparison of the
|
||||
low half. The resulting flags cover the whole. */
|
||||
tcg_out_dat_rI(s, COND_AL, ARITH_CMP, 0, ah, bh, const_bh);
|
||||
tcg_out_dat_rI(s, COND_EQ, ARITH_CMP, 0, al, bl, const_bl);
|
||||
return cond;
|
||||
|
||||
case TCG_COND_LT:
|
||||
case TCG_COND_GE:
|
||||
/* We perform a double-word subtraction and examine the result.
|
||||
We do not actually need the result of the subtract, so the
|
||||
low part "subtract" is a compare. For the high half we have
|
||||
no choice but to compute into a temporary. */
|
||||
tcg_out_dat_rI(s, COND_AL, ARITH_CMP, 0, al, bl, const_bl);
|
||||
tcg_out_dat_rI(s, COND_AL, ARITH_SBC | TO_CPSR,
|
||||
TCG_REG_TMP, ah, bh, const_bh);
|
||||
return cond;
|
||||
|
||||
case TCG_COND_LE:
|
||||
case TCG_COND_GT:
|
||||
/* Similar, but with swapped arguments, via reversed subtract. */
|
||||
tcg_out_dat_rI(s, COND_AL, ARITH_RSB | TO_CPSR,
|
||||
TCG_REG_TMP, al, bl, const_bl);
|
||||
tcg_out_dat_rI(s, COND_AL, ARITH_RSC | TO_CPSR,
|
||||
TCG_REG_TMP, ah, bh, const_bh);
|
||||
return tcg_swap_cond(cond);
|
||||
|
||||
default:
|
||||
g_assert_not_reached();
|
||||
}
|
||||
}
|
||||
|
||||
#ifdef CONFIG_SOFTMMU
|
||||
#include "tcg-ldst.inc.c"
|
||||
|
||||
|
@ -1964,22 +2014,6 @@ static inline void tcg_out_op(TCGContext *s, TCGOpcode opc,
|
|||
tcg_out_goto_label(s, tcg_cond_to_arm_cond[args[2]],
|
||||
arg_label(args[3]));
|
||||
break;
|
||||
case INDEX_op_brcond2_i32:
|
||||
/* The resulting conditions are:
|
||||
* TCG_COND_EQ --> a0 == a2 && a1 == a3,
|
||||
* TCG_COND_NE --> (a0 != a2 && a1 == a3) || a1 != a3,
|
||||
* TCG_COND_LT(U) --> (a0 < a2 && a1 == a3) || a1 < a3,
|
||||
* TCG_COND_GE(U) --> (a0 >= a2 && a1 == a3) || (a1 >= a3 && a1 != a3),
|
||||
* TCG_COND_LE(U) --> (a0 <= a2 && a1 == a3) || (a1 <= a3 && a1 != a3),
|
||||
* TCG_COND_GT(U) --> (a0 > a2 && a1 == a3) || a1 > a3,
|
||||
*/
|
||||
tcg_out_dat_rIN(s, COND_AL, ARITH_CMP, ARITH_CMN, 0,
|
||||
args[1], args[3], const_args[3]);
|
||||
tcg_out_dat_rIN(s, COND_EQ, ARITH_CMP, ARITH_CMN, 0,
|
||||
args[0], args[2], const_args[2]);
|
||||
tcg_out_goto_label(s, tcg_cond_to_arm_cond[args[4]],
|
||||
arg_label(args[5]));
|
||||
break;
|
||||
case INDEX_op_setcond_i32:
|
||||
tcg_out_dat_rIN(s, COND_AL, ARITH_CMP, ARITH_CMN, 0,
|
||||
args[1], args[2], const_args[2]);
|
||||
|
@ -1988,15 +2022,14 @@ static inline void tcg_out_op(TCGContext *s, TCGOpcode opc,
|
|||
tcg_out_dat_imm(s, tcg_cond_to_arm_cond[tcg_invert_cond(args[3])],
|
||||
ARITH_MOV, args[0], 0, 0);
|
||||
break;
|
||||
case INDEX_op_brcond2_i32:
|
||||
c = tcg_out_cmp2(s, args, const_args);
|
||||
tcg_out_goto_label(s, tcg_cond_to_arm_cond[c], arg_label(args[5]));
|
||||
break;
|
||||
case INDEX_op_setcond2_i32:
|
||||
/* See brcond2_i32 comment */
|
||||
tcg_out_dat_rIN(s, COND_AL, ARITH_CMP, ARITH_CMN, 0,
|
||||
args[2], args[4], const_args[4]);
|
||||
tcg_out_dat_rIN(s, COND_EQ, ARITH_CMP, ARITH_CMN, 0,
|
||||
args[1], args[3], const_args[3]);
|
||||
tcg_out_dat_imm(s, tcg_cond_to_arm_cond[args[5]],
|
||||
ARITH_MOV, args[0], 0, 1);
|
||||
tcg_out_dat_imm(s, tcg_cond_to_arm_cond[tcg_invert_cond(args[5])],
|
||||
c = tcg_out_cmp2(s, args + 1, const_args + 1);
|
||||
tcg_out_dat_imm(s, tcg_cond_to_arm_cond[c], ARITH_MOV, args[0], 0, 1);
|
||||
tcg_out_dat_imm(s, tcg_cond_to_arm_cond[tcg_invert_cond(c)],
|
||||
ARITH_MOV, args[0], 0, 0);
|
||||
break;
|
||||
|
||||
|
@ -2082,8 +2115,8 @@ static const TCGTargetOpDef *tcg_target_op_def(TCGOpcode op)
|
|||
static const TCGTargetOpDef movc = { 0, { "r", "r", "rIN", "rIK", "0" } };
|
||||
static const TCGTargetOpDef add2 = { 0, { "r", "r", "r", "r", "rIN", "rIK" } };
|
||||
static const TCGTargetOpDef sub2 = { 0, { "r", "r", "rI", "rI", "rIN", "rIK" } };
|
||||
static const TCGTargetOpDef br2 = { 0, { "r", "r", "rIN", "rIN" } };
|
||||
static const TCGTargetOpDef setc2 = { 0, { "r", "r", "r", "rIN", "rIN" } };
|
||||
static const TCGTargetOpDef br2 = { 0, { "r", "r", "rI", "rI" } };
|
||||
static const TCGTargetOpDef setc2 = { 0, { "r", "r", "r", "rI", "rI" } };
|
||||
|
||||
switch (op) {
|
||||
case INDEX_op_goto_ptr:
|
||||
|
|
Loading…
Reference in a new issue