mirror of
https://github.com/Motorhead1991/qemu.git
synced 2025-08-11 03:24:58 -06:00
tcg: Convert clz to TCGOutOpBinary
Reviewed-by: Pierrick Bouvier <pierrick.bouvier@linaro.org> Signed-off-by: Richard Henderson <richard.henderson@linaro.org>
This commit is contained in:
parent
005a87e148
commit
8b915879b0
25 changed files with 365 additions and 264 deletions
|
@ -15,7 +15,6 @@
|
||||||
/* optional instructions */
|
/* optional instructions */
|
||||||
#define TCG_TARGET_HAS_bswap16_i32 1
|
#define TCG_TARGET_HAS_bswap16_i32 1
|
||||||
#define TCG_TARGET_HAS_bswap32_i32 1
|
#define TCG_TARGET_HAS_bswap32_i32 1
|
||||||
#define TCG_TARGET_HAS_clz_i32 1
|
|
||||||
#define TCG_TARGET_HAS_ctz_i32 1
|
#define TCG_TARGET_HAS_ctz_i32 1
|
||||||
#define TCG_TARGET_HAS_ctpop_i32 0
|
#define TCG_TARGET_HAS_ctpop_i32 0
|
||||||
#define TCG_TARGET_HAS_extract2_i32 1
|
#define TCG_TARGET_HAS_extract2_i32 1
|
||||||
|
@ -30,7 +29,6 @@
|
||||||
#define TCG_TARGET_HAS_bswap16_i64 1
|
#define TCG_TARGET_HAS_bswap16_i64 1
|
||||||
#define TCG_TARGET_HAS_bswap32_i64 1
|
#define TCG_TARGET_HAS_bswap32_i64 1
|
||||||
#define TCG_TARGET_HAS_bswap64_i64 1
|
#define TCG_TARGET_HAS_bswap64_i64 1
|
||||||
#define TCG_TARGET_HAS_clz_i64 1
|
|
||||||
#define TCG_TARGET_HAS_ctz_i64 1
|
#define TCG_TARGET_HAS_ctz_i64 1
|
||||||
#define TCG_TARGET_HAS_ctpop_i64 0
|
#define TCG_TARGET_HAS_ctpop_i64 0
|
||||||
#define TCG_TARGET_HAS_extract2_i64 1
|
#define TCG_TARGET_HAS_extract2_i64 1
|
||||||
|
|
|
@ -1618,37 +1618,6 @@ static inline void tcg_out_mb(TCGContext *s, TCGArg a0)
|
||||||
tcg_out32(s, sync[a0 & TCG_MO_ALL]);
|
tcg_out32(s, sync[a0 & TCG_MO_ALL]);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void tcg_out_cltz(TCGContext *s, TCGType ext, TCGReg d,
|
|
||||||
TCGReg a0, TCGArg b, bool const_b, bool is_ctz)
|
|
||||||
{
|
|
||||||
TCGReg a1 = a0;
|
|
||||||
if (is_ctz) {
|
|
||||||
a1 = TCG_REG_TMP0;
|
|
||||||
tcg_out_insn(s, 3507, RBIT, ext, a1, a0);
|
|
||||||
}
|
|
||||||
if (const_b && b == (ext ? 64 : 32)) {
|
|
||||||
tcg_out_insn(s, 3507, CLZ, ext, d, a1);
|
|
||||||
} else {
|
|
||||||
AArch64Insn sel = I3506_CSEL;
|
|
||||||
|
|
||||||
tcg_out_cmp(s, ext, TCG_COND_NE, a0, 0, 1);
|
|
||||||
tcg_out_insn(s, 3507, CLZ, ext, TCG_REG_TMP0, a1);
|
|
||||||
|
|
||||||
if (const_b) {
|
|
||||||
if (b == -1) {
|
|
||||||
b = TCG_REG_XZR;
|
|
||||||
sel = I3506_CSINV;
|
|
||||||
} else if (b == 0) {
|
|
||||||
b = TCG_REG_XZR;
|
|
||||||
} else {
|
|
||||||
tcg_out_movi(s, ext, d, b);
|
|
||||||
b = d;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
tcg_out_insn_3506(s, sel, ext, d, TCG_REG_TMP0, b, TCG_COND_NE);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
typedef struct {
|
typedef struct {
|
||||||
TCGReg base;
|
TCGReg base;
|
||||||
TCGReg index;
|
TCGReg index;
|
||||||
|
@ -2121,6 +2090,45 @@ static const TCGOutOpBinary outop_andc = {
|
||||||
.out_rrr = tgen_andc,
|
.out_rrr = tgen_andc,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static void tgen_clz(TCGContext *s, TCGType type,
|
||||||
|
TCGReg a0, TCGReg a1, TCGReg a2)
|
||||||
|
{
|
||||||
|
tcg_out_cmp(s, type, TCG_COND_NE, a1, 0, true);
|
||||||
|
tcg_out_insn(s, 3507, CLZ, type, TCG_REG_TMP0, a1);
|
||||||
|
tcg_out_insn(s, 3506, CSEL, type, a0, TCG_REG_TMP0, a2, TCG_COND_NE);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void tgen_clzi(TCGContext *s, TCGType type,
|
||||||
|
TCGReg a0, TCGReg a1, tcg_target_long a2)
|
||||||
|
{
|
||||||
|
if (a2 == (type == TCG_TYPE_I32 ? 32 : 64)) {
|
||||||
|
tcg_out_insn(s, 3507, CLZ, type, a0, a1);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
tcg_out_cmp(s, type, TCG_COND_NE, a1, 0, true);
|
||||||
|
tcg_out_insn(s, 3507, CLZ, type, a0, a1);
|
||||||
|
|
||||||
|
switch (a2) {
|
||||||
|
case -1:
|
||||||
|
tcg_out_insn(s, 3506, CSINV, type, a0, a0, TCG_REG_XZR, TCG_COND_NE);
|
||||||
|
break;
|
||||||
|
case 0:
|
||||||
|
tcg_out_insn(s, 3506, CSEL, type, a0, a0, TCG_REG_XZR, TCG_COND_NE);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
tcg_out_movi(s, type, TCG_REG_TMP0, a2);
|
||||||
|
tcg_out_insn(s, 3506, CSEL, type, a0, a0, TCG_REG_TMP0, TCG_COND_NE);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static const TCGOutOpBinary outop_clz = {
|
||||||
|
.base.static_constraint = C_O1_I2(r, r, rAL),
|
||||||
|
.out_rrr = tgen_clz,
|
||||||
|
.out_rri = tgen_clzi,
|
||||||
|
};
|
||||||
|
|
||||||
static void tgen_divs(TCGContext *s, TCGType type,
|
static void tgen_divs(TCGContext *s, TCGType type,
|
||||||
TCGReg a0, TCGReg a1, TCGReg a2)
|
TCGReg a0, TCGReg a1, TCGReg a2)
|
||||||
{
|
{
|
||||||
|
@ -2460,13 +2468,14 @@ static void tcg_out_op(TCGContext *s, TCGOpcode opc, TCGType ext,
|
||||||
tcg_out_ldst(s, I3312_STRX, a0, a1, a2, 3);
|
tcg_out_ldst(s, I3312_STRX, a0, a1, a2, 3);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case INDEX_op_clz_i64:
|
|
||||||
case INDEX_op_clz_i32:
|
|
||||||
tcg_out_cltz(s, ext, a0, a1, a2, c2, false);
|
|
||||||
break;
|
|
||||||
case INDEX_op_ctz_i64:
|
case INDEX_op_ctz_i64:
|
||||||
case INDEX_op_ctz_i32:
|
case INDEX_op_ctz_i32:
|
||||||
tcg_out_cltz(s, ext, a0, a1, a2, c2, true);
|
tcg_out_insn(s, 3507, RBIT, ext, TCG_REG_TMP0, a1);
|
||||||
|
if (c2) {
|
||||||
|
tgen_clzi(s, ext, a0, TCG_REG_TMP0, a2);
|
||||||
|
} else {
|
||||||
|
tgen_clz(s, ext, a0, TCG_REG_TMP0, a2);
|
||||||
|
}
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case INDEX_op_brcond_i32:
|
case INDEX_op_brcond_i32:
|
||||||
|
@ -3089,9 +3098,7 @@ tcg_target_op_def(TCGOpcode op, TCGType type, unsigned flags)
|
||||||
case INDEX_op_negsetcond_i64:
|
case INDEX_op_negsetcond_i64:
|
||||||
return C_O1_I2(r, r, rC);
|
return C_O1_I2(r, r, rC);
|
||||||
|
|
||||||
case INDEX_op_clz_i32:
|
|
||||||
case INDEX_op_ctz_i32:
|
case INDEX_op_ctz_i32:
|
||||||
case INDEX_op_clz_i64:
|
|
||||||
case INDEX_op_ctz_i64:
|
case INDEX_op_ctz_i64:
|
||||||
return C_O1_I2(r, r, rAL);
|
return C_O1_I2(r, r, rAL);
|
||||||
|
|
||||||
|
|
|
@ -26,7 +26,6 @@ extern bool use_neon_instructions;
|
||||||
/* optional instructions */
|
/* optional instructions */
|
||||||
#define TCG_TARGET_HAS_bswap16_i32 1
|
#define TCG_TARGET_HAS_bswap16_i32 1
|
||||||
#define TCG_TARGET_HAS_bswap32_i32 1
|
#define TCG_TARGET_HAS_bswap32_i32 1
|
||||||
#define TCG_TARGET_HAS_clz_i32 1
|
|
||||||
#define TCG_TARGET_HAS_ctz_i32 use_armv7_instructions
|
#define TCG_TARGET_HAS_ctz_i32 use_armv7_instructions
|
||||||
#define TCG_TARGET_HAS_ctpop_i32 0
|
#define TCG_TARGET_HAS_ctpop_i32 0
|
||||||
#define TCG_TARGET_HAS_extract2_i32 1
|
#define TCG_TARGET_HAS_extract2_i32 1
|
||||||
|
|
|
@ -1862,6 +1862,32 @@ static const TCGOutOpBinary outop_andc = {
|
||||||
.out_rrr = tgen_andc,
|
.out_rrr = tgen_andc,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static void tgen_clz(TCGContext *s, TCGType type,
|
||||||
|
TCGReg a0, TCGReg a1, TCGReg a2)
|
||||||
|
{
|
||||||
|
tcg_out_dat_imm(s, COND_AL, ARITH_CMP, 0, a1, 0);
|
||||||
|
tcg_out_dat_reg(s, COND_NE, INSN_CLZ, a0, 0, a1, 0);
|
||||||
|
tcg_out_mov_reg(s, COND_EQ, a0, a2);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void tgen_clzi(TCGContext *s, TCGType type,
|
||||||
|
TCGReg a0, TCGReg a1, tcg_target_long a2)
|
||||||
|
{
|
||||||
|
if (a2 == 32) {
|
||||||
|
tcg_out_dat_reg(s, COND_AL, INSN_CLZ, a0, 0, a1, 0);
|
||||||
|
} else {
|
||||||
|
tcg_out_dat_imm(s, COND_AL, ARITH_CMP, 0, a1, 0);
|
||||||
|
tcg_out_dat_reg(s, COND_NE, INSN_CLZ, a0, 0, a1, 0);
|
||||||
|
tcg_out_movi32(s, COND_EQ, a0, a2);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static const TCGOutOpBinary outop_clz = {
|
||||||
|
.base.static_constraint = C_O1_I2(r, r, rIK),
|
||||||
|
.out_rrr = tgen_clz,
|
||||||
|
.out_rri = tgen_clzi,
|
||||||
|
};
|
||||||
|
|
||||||
static TCGConstraintSetIndex cset_idiv(TCGType type, unsigned flags)
|
static TCGConstraintSetIndex cset_idiv(TCGType type, unsigned flags)
|
||||||
{
|
{
|
||||||
return use_idiv_instructions ? C_O1_I2(r, r, r) : C_NotImplemented;
|
return use_idiv_instructions ? C_O1_I2(r, r, r) : C_NotImplemented;
|
||||||
|
@ -2196,23 +2222,10 @@ static void tcg_out_op(TCGContext *s, TCGOpcode opc, TCGType type,
|
||||||
|
|
||||||
case INDEX_op_ctz_i32:
|
case INDEX_op_ctz_i32:
|
||||||
tcg_out_dat_reg(s, COND_AL, INSN_RBIT, TCG_REG_TMP, 0, args[1], 0);
|
tcg_out_dat_reg(s, COND_AL, INSN_RBIT, TCG_REG_TMP, 0, args[1], 0);
|
||||||
a1 = TCG_REG_TMP;
|
if (const_args[2]) {
|
||||||
goto do_clz;
|
tgen_clzi(s, TCG_TYPE_I32, args[0], TCG_REG_TMP, args[2]);
|
||||||
|
} else {
|
||||||
case INDEX_op_clz_i32:
|
tgen_clz(s, TCG_TYPE_I32, args[0], TCG_REG_TMP, args[2]);
|
||||||
a1 = args[1];
|
|
||||||
do_clz:
|
|
||||||
a0 = args[0];
|
|
||||||
a2 = args[2];
|
|
||||||
c = const_args[2];
|
|
||||||
if (c && a2 == 32) {
|
|
||||||
tcg_out_dat_reg(s, COND_AL, INSN_CLZ, a0, 0, a1, 0);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
tcg_out_dat_imm(s, COND_AL, ARITH_CMP, 0, a1, 0);
|
|
||||||
tcg_out_dat_reg(s, COND_NE, INSN_CLZ, a0, 0, a1, 0);
|
|
||||||
if (c || a0 != a2) {
|
|
||||||
tcg_out_dat_rIK(s, COND_EQ, ARITH_MOV, ARITH_MVN, a0, 0, a2, c);
|
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
|
|
@ -28,7 +28,6 @@
|
||||||
/* optional instructions */
|
/* optional instructions */
|
||||||
#define TCG_TARGET_HAS_bswap16_i32 1
|
#define TCG_TARGET_HAS_bswap16_i32 1
|
||||||
#define TCG_TARGET_HAS_bswap32_i32 1
|
#define TCG_TARGET_HAS_bswap32_i32 1
|
||||||
#define TCG_TARGET_HAS_clz_i32 1
|
|
||||||
#define TCG_TARGET_HAS_ctz_i32 1
|
#define TCG_TARGET_HAS_ctz_i32 1
|
||||||
#define TCG_TARGET_HAS_ctpop_i32 have_popcnt
|
#define TCG_TARGET_HAS_ctpop_i32 have_popcnt
|
||||||
#define TCG_TARGET_HAS_extract2_i32 1
|
#define TCG_TARGET_HAS_extract2_i32 1
|
||||||
|
@ -44,7 +43,6 @@
|
||||||
#define TCG_TARGET_HAS_bswap16_i64 1
|
#define TCG_TARGET_HAS_bswap16_i64 1
|
||||||
#define TCG_TARGET_HAS_bswap32_i64 1
|
#define TCG_TARGET_HAS_bswap32_i64 1
|
||||||
#define TCG_TARGET_HAS_bswap64_i64 1
|
#define TCG_TARGET_HAS_bswap64_i64 1
|
||||||
#define TCG_TARGET_HAS_clz_i64 1
|
|
||||||
#define TCG_TARGET_HAS_ctz_i64 1
|
#define TCG_TARGET_HAS_ctz_i64 1
|
||||||
#define TCG_TARGET_HAS_ctpop_i64 have_popcnt
|
#define TCG_TARGET_HAS_ctpop_i64 have_popcnt
|
||||||
#define TCG_TARGET_HAS_extract2_i64 1
|
#define TCG_TARGET_HAS_extract2_i64 1
|
||||||
|
|
|
@ -1869,32 +1869,6 @@ static void tcg_out_ctz(TCGContext *s, int rexw, TCGReg dest, TCGReg arg1,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static void tcg_out_clz(TCGContext *s, int rexw, TCGReg dest, TCGReg arg1,
|
|
||||||
TCGArg arg2, bool const_a2)
|
|
||||||
{
|
|
||||||
if (have_lzcnt) {
|
|
||||||
tcg_out_modrm(s, OPC_LZCNT + rexw, dest, arg1);
|
|
||||||
if (const_a2) {
|
|
||||||
tcg_debug_assert(arg2 == (rexw ? 64 : 32));
|
|
||||||
} else {
|
|
||||||
tcg_debug_assert(dest != arg2);
|
|
||||||
tcg_out_cmov(s, JCC_JB, rexw, dest, arg2);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
tcg_debug_assert(!const_a2);
|
|
||||||
tcg_debug_assert(dest != arg1);
|
|
||||||
tcg_debug_assert(dest != arg2);
|
|
||||||
|
|
||||||
/* Recall that the output of BSR is the index not the count. */
|
|
||||||
tcg_out_modrm(s, OPC_BSR + rexw, dest, arg1);
|
|
||||||
tgen_arithi(s, ARITH_XOR + rexw, dest, rexw ? 63 : 31, 0);
|
|
||||||
|
|
||||||
/* Since we have destroyed the flags from BSR, we have to re-test. */
|
|
||||||
int jcc = tcg_out_cmp(s, TCG_COND_EQ, arg1, 0, 1, rexw);
|
|
||||||
tcg_out_cmov(s, jcc, rexw, dest, arg2);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static void tcg_out_branch(TCGContext *s, int call, const tcg_insn_unit *dest)
|
static void tcg_out_branch(TCGContext *s, int call, const tcg_insn_unit *dest)
|
||||||
{
|
{
|
||||||
intptr_t disp = tcg_pcrel_diff(s, dest) - 5;
|
intptr_t disp = tcg_pcrel_diff(s, dest) - 5;
|
||||||
|
@ -2633,6 +2607,45 @@ static const TCGOutOpBinary outop_andc = {
|
||||||
.out_rrr = tgen_andc,
|
.out_rrr = tgen_andc,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static void tgen_clz(TCGContext *s, TCGType type,
|
||||||
|
TCGReg a0, TCGReg a1, TCGReg a2)
|
||||||
|
{
|
||||||
|
int rexw = type == TCG_TYPE_I32 ? 0 : P_REXW;
|
||||||
|
int jcc;
|
||||||
|
|
||||||
|
if (have_lzcnt) {
|
||||||
|
tcg_out_modrm(s, OPC_LZCNT + rexw, a0, a1);
|
||||||
|
jcc = JCC_JB;
|
||||||
|
} else {
|
||||||
|
/* Recall that the output of BSR is the index not the count. */
|
||||||
|
tcg_out_modrm(s, OPC_BSR + rexw, a0, a1);
|
||||||
|
tgen_arithi(s, ARITH_XOR + rexw, a0, rexw ? 63 : 31, 0);
|
||||||
|
|
||||||
|
/* Since we have destroyed the flags from BSR, we have to re-test. */
|
||||||
|
jcc = tcg_out_cmp(s, TCG_COND_EQ, a1, 0, 1, rexw);
|
||||||
|
}
|
||||||
|
tcg_out_cmov(s, jcc, rexw, a0, a2);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void tgen_clzi(TCGContext *s, TCGType type,
|
||||||
|
TCGReg a0, TCGReg a1, tcg_target_long a2)
|
||||||
|
{
|
||||||
|
int rexw = type == TCG_TYPE_I32 ? 0 : P_REXW;
|
||||||
|
tcg_out_modrm(s, OPC_LZCNT + rexw, a0, a1);
|
||||||
|
}
|
||||||
|
|
||||||
|
static TCGConstraintSetIndex cset_clz(TCGType type, unsigned flags)
|
||||||
|
{
|
||||||
|
return have_lzcnt ? C_N1_I2(r, r, rW) : C_N1_I2(r, r, r);
|
||||||
|
}
|
||||||
|
|
||||||
|
static const TCGOutOpBinary outop_clz = {
|
||||||
|
.base.static_constraint = C_Dynamic,
|
||||||
|
.base.dynamic_constraint = cset_clz,
|
||||||
|
.out_rrr = tgen_clz,
|
||||||
|
.out_rri = tgen_clzi,
|
||||||
|
};
|
||||||
|
|
||||||
static const TCGOutOpBinary outop_divs = {
|
static const TCGOutOpBinary outop_divs = {
|
||||||
.base.static_constraint = C_NotImplemented,
|
.base.static_constraint = C_NotImplemented,
|
||||||
};
|
};
|
||||||
|
@ -3019,9 +3032,6 @@ static void tcg_out_op(TCGContext *s, TCGOpcode opc, TCGType type,
|
||||||
OP_32_64(ctz):
|
OP_32_64(ctz):
|
||||||
tcg_out_ctz(s, rexw, args[0], args[1], args[2], const_args[2]);
|
tcg_out_ctz(s, rexw, args[0], args[1], args[2], const_args[2]);
|
||||||
break;
|
break;
|
||||||
OP_32_64(clz):
|
|
||||||
tcg_out_clz(s, rexw, args[0], args[1], args[2], const_args[2]);
|
|
||||||
break;
|
|
||||||
OP_32_64(ctpop):
|
OP_32_64(ctpop):
|
||||||
tcg_out_modrm(s, OPC_POPCNT + rexw, a0, a1);
|
tcg_out_modrm(s, OPC_POPCNT + rexw, a0, a1);
|
||||||
break;
|
break;
|
||||||
|
@ -3907,10 +3917,6 @@ tcg_target_op_def(TCGOpcode op, TCGType type, unsigned flags)
|
||||||
case INDEX_op_ctz_i64:
|
case INDEX_op_ctz_i64:
|
||||||
return have_bmi1 ? C_N1_I2(r, r, rW) : C_N1_I2(r, r, r);
|
return have_bmi1 ? C_N1_I2(r, r, rW) : C_N1_I2(r, r, r);
|
||||||
|
|
||||||
case INDEX_op_clz_i32:
|
|
||||||
case INDEX_op_clz_i64:
|
|
||||||
return have_lzcnt ? C_N1_I2(r, r, rW) : C_N1_I2(r, r, r);
|
|
||||||
|
|
||||||
case INDEX_op_qemu_ld_i32:
|
case INDEX_op_qemu_ld_i32:
|
||||||
return C_O1_I1(r, L);
|
return C_O1_I1(r, L);
|
||||||
|
|
||||||
|
|
|
@ -18,7 +18,6 @@
|
||||||
#define TCG_TARGET_HAS_muls2_i32 0
|
#define TCG_TARGET_HAS_muls2_i32 0
|
||||||
#define TCG_TARGET_HAS_bswap16_i32 1
|
#define TCG_TARGET_HAS_bswap16_i32 1
|
||||||
#define TCG_TARGET_HAS_bswap32_i32 1
|
#define TCG_TARGET_HAS_bswap32_i32 1
|
||||||
#define TCG_TARGET_HAS_clz_i32 1
|
|
||||||
#define TCG_TARGET_HAS_ctz_i32 1
|
#define TCG_TARGET_HAS_ctz_i32 1
|
||||||
#define TCG_TARGET_HAS_ctpop_i32 0
|
#define TCG_TARGET_HAS_ctpop_i32 0
|
||||||
#define TCG_TARGET_HAS_qemu_st8_i32 0
|
#define TCG_TARGET_HAS_qemu_st8_i32 0
|
||||||
|
@ -30,7 +29,6 @@
|
||||||
#define TCG_TARGET_HAS_bswap16_i64 1
|
#define TCG_TARGET_HAS_bswap16_i64 1
|
||||||
#define TCG_TARGET_HAS_bswap32_i64 1
|
#define TCG_TARGET_HAS_bswap32_i64 1
|
||||||
#define TCG_TARGET_HAS_bswap64_i64 1
|
#define TCG_TARGET_HAS_bswap64_i64 1
|
||||||
#define TCG_TARGET_HAS_clz_i64 1
|
|
||||||
#define TCG_TARGET_HAS_ctz_i64 1
|
#define TCG_TARGET_HAS_ctz_i64 1
|
||||||
#define TCG_TARGET_HAS_ctpop_i64 0
|
#define TCG_TARGET_HAS_ctpop_i64 0
|
||||||
#define TCG_TARGET_HAS_add2_i64 0
|
#define TCG_TARGET_HAS_add2_i64 0
|
||||||
|
|
|
@ -1328,6 +1328,33 @@ static const TCGOutOpBinary outop_andc = {
|
||||||
.out_rrr = tgen_andc,
|
.out_rrr = tgen_andc,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static void tgen_clzi(TCGContext *s, TCGType type,
|
||||||
|
TCGReg a0, TCGReg a1, tcg_target_long a2)
|
||||||
|
{
|
||||||
|
/* a2 is constrained to exactly the type width. */
|
||||||
|
if (type == TCG_TYPE_I32) {
|
||||||
|
tcg_out_opc_clz_w(s, a0, a1);
|
||||||
|
} else {
|
||||||
|
tcg_out_opc_clz_d(s, a0, a1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void tgen_clz(TCGContext *s, TCGType type,
|
||||||
|
TCGReg a0, TCGReg a1, TCGReg a2)
|
||||||
|
{
|
||||||
|
tgen_clzi(s, type, TCG_REG_TMP0, a1, /* ignored */ 0);
|
||||||
|
/* a0 = a1 ? REG_TMP0 : a2 */
|
||||||
|
tcg_out_opc_maskeqz(s, TCG_REG_TMP0, TCG_REG_TMP0, a1);
|
||||||
|
tcg_out_opc_masknez(s, a0, a2, a1);
|
||||||
|
tcg_out_opc_or(s, a0, a0, TCG_REG_TMP0);
|
||||||
|
}
|
||||||
|
|
||||||
|
static const TCGOutOpBinary outop_clz = {
|
||||||
|
.base.static_constraint = C_O1_I2(r, r, rW),
|
||||||
|
.out_rrr = tgen_clz,
|
||||||
|
.out_rri = tgen_clzi,
|
||||||
|
};
|
||||||
|
|
||||||
static void tgen_divs(TCGContext *s, TCGType type,
|
static void tgen_divs(TCGContext *s, TCGType type,
|
||||||
TCGReg a0, TCGReg a1, TCGReg a2)
|
TCGReg a0, TCGReg a1, TCGReg a2)
|
||||||
{
|
{
|
||||||
|
@ -1754,13 +1781,6 @@ static void tcg_out_op(TCGContext *s, TCGOpcode opc, TCGType type,
|
||||||
tcg_out_opc_revb_d(s, a0, a1);
|
tcg_out_opc_revb_d(s, a0, a1);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case INDEX_op_clz_i32:
|
|
||||||
tcg_out_clzctz(s, OPC_CLZ_W, a0, a1, a2, c2, true);
|
|
||||||
break;
|
|
||||||
case INDEX_op_clz_i64:
|
|
||||||
tcg_out_clzctz(s, OPC_CLZ_D, a0, a1, a2, c2, false);
|
|
||||||
break;
|
|
||||||
|
|
||||||
case INDEX_op_ctz_i32:
|
case INDEX_op_ctz_i32:
|
||||||
tcg_out_clzctz(s, OPC_CTZ_W, a0, a1, a2, c2, true);
|
tcg_out_clzctz(s, OPC_CTZ_W, a0, a1, a2, c2, true);
|
||||||
break;
|
break;
|
||||||
|
@ -2398,8 +2418,6 @@ tcg_target_op_def(TCGOpcode op, TCGType type, unsigned flags)
|
||||||
case INDEX_op_qemu_ld_i64:
|
case INDEX_op_qemu_ld_i64:
|
||||||
return C_O1_I1(r, r);
|
return C_O1_I1(r, r);
|
||||||
|
|
||||||
case INDEX_op_clz_i32:
|
|
||||||
case INDEX_op_clz_i64:
|
|
||||||
case INDEX_op_ctz_i32:
|
case INDEX_op_ctz_i32:
|
||||||
case INDEX_op_ctz_i64:
|
case INDEX_op_ctz_i64:
|
||||||
return C_O1_I2(r, r, rW);
|
return C_O1_I2(r, r, rW);
|
||||||
|
|
|
@ -60,7 +60,6 @@ extern bool use_mips32r2_instructions;
|
||||||
|
|
||||||
/* optional instructions detected at runtime */
|
/* optional instructions detected at runtime */
|
||||||
#define TCG_TARGET_HAS_extract2_i32 0
|
#define TCG_TARGET_HAS_extract2_i32 0
|
||||||
#define TCG_TARGET_HAS_clz_i32 use_mips32r2_instructions
|
|
||||||
#define TCG_TARGET_HAS_ctz_i32 0
|
#define TCG_TARGET_HAS_ctz_i32 0
|
||||||
#define TCG_TARGET_HAS_ctpop_i32 0
|
#define TCG_TARGET_HAS_ctpop_i32 0
|
||||||
#define TCG_TARGET_HAS_qemu_st8_i32 0
|
#define TCG_TARGET_HAS_qemu_st8_i32 0
|
||||||
|
@ -70,7 +69,6 @@ extern bool use_mips32r2_instructions;
|
||||||
#define TCG_TARGET_HAS_bswap32_i64 1
|
#define TCG_TARGET_HAS_bswap32_i64 1
|
||||||
#define TCG_TARGET_HAS_bswap64_i64 1
|
#define TCG_TARGET_HAS_bswap64_i64 1
|
||||||
#define TCG_TARGET_HAS_extract2_i64 0
|
#define TCG_TARGET_HAS_extract2_i64 0
|
||||||
#define TCG_TARGET_HAS_clz_i64 use_mips32r2_instructions
|
|
||||||
#define TCG_TARGET_HAS_ctz_i64 0
|
#define TCG_TARGET_HAS_ctz_i64 0
|
||||||
#define TCG_TARGET_HAS_ctpop_i64 0
|
#define TCG_TARGET_HAS_ctpop_i64 0
|
||||||
#endif
|
#endif
|
||||||
|
|
|
@ -1563,33 +1563,6 @@ static void tcg_out_mb(TCGContext *s, TCGArg a0)
|
||||||
tcg_out32(s, sync[a0 & TCG_MO_ALL]);
|
tcg_out32(s, sync[a0 & TCG_MO_ALL]);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void tcg_out_clz(TCGContext *s, MIPSInsn opcv2, MIPSInsn opcv6,
|
|
||||||
int width, TCGReg a0, TCGReg a1, TCGArg a2)
|
|
||||||
{
|
|
||||||
if (use_mips32r6_instructions) {
|
|
||||||
if (a2 == width) {
|
|
||||||
tcg_out_opc_reg(s, opcv6, a0, a1, 0);
|
|
||||||
} else {
|
|
||||||
tcg_out_opc_reg(s, opcv6, TCG_TMP0, a1, 0);
|
|
||||||
tcg_out_movcond(s, TCG_COND_EQ, a0, a1, 0, a2, TCG_TMP0);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
if (a2 == width) {
|
|
||||||
tcg_out_opc_reg(s, opcv2, a0, a1, a1);
|
|
||||||
} else if (a0 == a2) {
|
|
||||||
tcg_out_opc_reg(s, opcv2, TCG_TMP0, a1, a1);
|
|
||||||
tcg_out_opc_reg(s, OPC_MOVN, a0, TCG_TMP0, a1);
|
|
||||||
} else if (a0 != a1) {
|
|
||||||
tcg_out_opc_reg(s, opcv2, a0, a1, a1);
|
|
||||||
tcg_out_opc_reg(s, OPC_MOVZ, a0, a2, a1);
|
|
||||||
} else {
|
|
||||||
tcg_out_opc_reg(s, opcv2, TCG_TMP0, a1, a1);
|
|
||||||
tcg_out_opc_reg(s, OPC_MOVZ, TCG_TMP0, a2, a1);
|
|
||||||
tcg_out_mov(s, TCG_TYPE_REG, a0, TCG_TMP0);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static void tcg_out_exit_tb(TCGContext *s, uintptr_t a0)
|
static void tcg_out_exit_tb(TCGContext *s, uintptr_t a0)
|
||||||
{
|
{
|
||||||
TCGReg base = TCG_REG_ZERO;
|
TCGReg base = TCG_REG_ZERO;
|
||||||
|
@ -1712,6 +1685,55 @@ static const TCGOutOpBinary outop_andc = {
|
||||||
.base.static_constraint = C_NotImplemented,
|
.base.static_constraint = C_NotImplemented,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static void tgen_clz(TCGContext *s, TCGType type,
|
||||||
|
TCGReg a0, TCGReg a1, TCGReg a2)
|
||||||
|
{
|
||||||
|
if (use_mips32r6_instructions) {
|
||||||
|
MIPSInsn opcv6 = type == TCG_TYPE_I32 ? OPC_CLZ_R6 : OPC_DCLZ_R6;
|
||||||
|
tcg_out_opc_reg(s, opcv6, TCG_TMP0, a1, 0);
|
||||||
|
tcg_out_movcond(s, TCG_COND_EQ, a0, a1, 0, a2, TCG_TMP0);
|
||||||
|
} else {
|
||||||
|
MIPSInsn opcv2 = type == TCG_TYPE_I32 ? OPC_CLZ : OPC_DCLZ;
|
||||||
|
if (a0 == a2) {
|
||||||
|
tcg_out_opc_reg(s, opcv2, TCG_TMP0, a1, a1);
|
||||||
|
tcg_out_opc_reg(s, OPC_MOVN, a0, TCG_TMP0, a1);
|
||||||
|
} else if (a0 != a1) {
|
||||||
|
tcg_out_opc_reg(s, opcv2, a0, a1, a1);
|
||||||
|
tcg_out_opc_reg(s, OPC_MOVZ, a0, a2, a1);
|
||||||
|
} else {
|
||||||
|
tcg_out_opc_reg(s, opcv2, TCG_TMP0, a1, a1);
|
||||||
|
tcg_out_opc_reg(s, OPC_MOVZ, TCG_TMP0, a2, a1);
|
||||||
|
tcg_out_mov(s, type, a0, TCG_TMP0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void tgen_clzi(TCGContext *s, TCGType type,
|
||||||
|
TCGReg a0, TCGReg a1, tcg_target_long a2)
|
||||||
|
{
|
||||||
|
if (a2 == 0) {
|
||||||
|
tgen_clz(s, type, a0, a1, TCG_REG_ZERO);
|
||||||
|
} else if (use_mips32r6_instructions) {
|
||||||
|
MIPSInsn opcv6 = type == TCG_TYPE_I32 ? OPC_CLZ_R6 : OPC_DCLZ_R6;
|
||||||
|
tcg_out_opc_reg(s, opcv6, a0, a1, 0);
|
||||||
|
} else {
|
||||||
|
MIPSInsn opcv2 = type == TCG_TYPE_I32 ? OPC_CLZ : OPC_DCLZ;
|
||||||
|
tcg_out_opc_reg(s, opcv2, a0, a1, a1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static TCGConstraintSetIndex cset_clz(TCGType type, unsigned flags)
|
||||||
|
{
|
||||||
|
return use_mips32r2_instructions ? C_O1_I2(r, r, rzW) : C_NotImplemented;
|
||||||
|
}
|
||||||
|
|
||||||
|
static const TCGOutOpBinary outop_clz = {
|
||||||
|
.base.static_constraint = C_Dynamic,
|
||||||
|
.base.dynamic_constraint = cset_clz,
|
||||||
|
.out_rrr = tgen_clz,
|
||||||
|
.out_rri = tgen_clzi,
|
||||||
|
};
|
||||||
|
|
||||||
static void tgen_divs(TCGContext *s, TCGType type,
|
static void tgen_divs(TCGContext *s, TCGType type,
|
||||||
TCGReg a0, TCGReg a1, TCGReg a2)
|
TCGReg a0, TCGReg a1, TCGReg a2)
|
||||||
{
|
{
|
||||||
|
@ -2165,13 +2187,6 @@ static void tcg_out_op(TCGContext *s, TCGOpcode opc, TCGType type,
|
||||||
tcg_out_dsra(s, a0, a1, 32);
|
tcg_out_dsra(s, a0, a1, 32);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case INDEX_op_clz_i32:
|
|
||||||
tcg_out_clz(s, OPC_CLZ, OPC_CLZ_R6, 32, a0, a1, a2);
|
|
||||||
break;
|
|
||||||
case INDEX_op_clz_i64:
|
|
||||||
tcg_out_clz(s, OPC_DCLZ, OPC_DCLZ_R6, 64, a0, a1, a2);
|
|
||||||
break;
|
|
||||||
|
|
||||||
case INDEX_op_deposit_i32:
|
case INDEX_op_deposit_i32:
|
||||||
tcg_out_opc_bf(s, OPC_INS, a0, a2, args[3] + args[4] - 1, args[3]);
|
tcg_out_opc_bf(s, OPC_INS, a0, a2, args[3] + args[4] - 1, args[3]);
|
||||||
break;
|
break;
|
||||||
|
@ -2329,9 +2344,6 @@ tcg_target_op_def(TCGOpcode op, TCGType type, unsigned flags)
|
||||||
case INDEX_op_muls2_i64:
|
case INDEX_op_muls2_i64:
|
||||||
case INDEX_op_mulu2_i64:
|
case INDEX_op_mulu2_i64:
|
||||||
return C_O2_I2(r, r, r, r);
|
return C_O2_I2(r, r, r, r);
|
||||||
case INDEX_op_clz_i32:
|
|
||||||
case INDEX_op_clz_i64:
|
|
||||||
return C_O1_I2(r, r, rzW);
|
|
||||||
|
|
||||||
case INDEX_op_deposit_i32:
|
case INDEX_op_deposit_i32:
|
||||||
case INDEX_op_deposit_i64:
|
case INDEX_op_deposit_i64:
|
||||||
|
|
|
@ -19,7 +19,6 @@
|
||||||
/* optional instructions */
|
/* optional instructions */
|
||||||
#define TCG_TARGET_HAS_bswap16_i32 1
|
#define TCG_TARGET_HAS_bswap16_i32 1
|
||||||
#define TCG_TARGET_HAS_bswap32_i32 1
|
#define TCG_TARGET_HAS_bswap32_i32 1
|
||||||
#define TCG_TARGET_HAS_clz_i32 1
|
|
||||||
#define TCG_TARGET_HAS_ctz_i32 have_isa_3_00
|
#define TCG_TARGET_HAS_ctz_i32 have_isa_3_00
|
||||||
#define TCG_TARGET_HAS_ctpop_i32 have_isa_2_06
|
#define TCG_TARGET_HAS_ctpop_i32 have_isa_2_06
|
||||||
#define TCG_TARGET_HAS_extract2_i32 0
|
#define TCG_TARGET_HAS_extract2_i32 0
|
||||||
|
@ -35,7 +34,6 @@
|
||||||
#define TCG_TARGET_HAS_bswap16_i64 1
|
#define TCG_TARGET_HAS_bswap16_i64 1
|
||||||
#define TCG_TARGET_HAS_bswap32_i64 1
|
#define TCG_TARGET_HAS_bswap32_i64 1
|
||||||
#define TCG_TARGET_HAS_bswap64_i64 1
|
#define TCG_TARGET_HAS_bswap64_i64 1
|
||||||
#define TCG_TARGET_HAS_clz_i64 1
|
|
||||||
#define TCG_TARGET_HAS_ctz_i64 have_isa_3_00
|
#define TCG_TARGET_HAS_ctz_i64 have_isa_3_00
|
||||||
#define TCG_TARGET_HAS_ctpop_i64 have_isa_2_06
|
#define TCG_TARGET_HAS_ctpop_i64 have_isa_2_06
|
||||||
#define TCG_TARGET_HAS_extract2_i64 0
|
#define TCG_TARGET_HAS_extract2_i64 0
|
||||||
|
|
|
@ -2954,6 +2954,26 @@ static const TCGOutOpBinary outop_andc = {
|
||||||
.out_rrr = tgen_andc,
|
.out_rrr = tgen_andc,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static void tgen_clz(TCGContext *s, TCGType type,
|
||||||
|
TCGReg a0, TCGReg a1, TCGReg a2)
|
||||||
|
{
|
||||||
|
uint32_t insn = type == TCG_TYPE_I32 ? CNTLZW : CNTLZD;
|
||||||
|
tcg_out_cntxz(s, type, insn, a0, a1, a2, false);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void tgen_clzi(TCGContext *s, TCGType type,
|
||||||
|
TCGReg a0, TCGReg a1, tcg_target_long a2)
|
||||||
|
{
|
||||||
|
uint32_t insn = type == TCG_TYPE_I32 ? CNTLZW : CNTLZD;
|
||||||
|
tcg_out_cntxz(s, type, insn, a0, a1, a2, true);
|
||||||
|
}
|
||||||
|
|
||||||
|
static const TCGOutOpBinary outop_clz = {
|
||||||
|
.base.static_constraint = C_O1_I2(r, r, rZW),
|
||||||
|
.out_rrr = tgen_clz,
|
||||||
|
.out_rri = tgen_clzi,
|
||||||
|
};
|
||||||
|
|
||||||
static void tgen_eqv(TCGContext *s, TCGType type,
|
static void tgen_eqv(TCGContext *s, TCGType type,
|
||||||
TCGReg a0, TCGReg a1, TCGReg a2)
|
TCGReg a0, TCGReg a1, TCGReg a2)
|
||||||
{
|
{
|
||||||
|
@ -3350,10 +3370,6 @@ static void tcg_out_op(TCGContext *s, TCGOpcode opc, TCGType type,
|
||||||
tcg_out_mem_long(s, STD, STDX, args[0], args[1], args[2]);
|
tcg_out_mem_long(s, STD, STDX, args[0], args[1], args[2]);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case INDEX_op_clz_i32:
|
|
||||||
tcg_out_cntxz(s, TCG_TYPE_I32, CNTLZW, args[0], args[1],
|
|
||||||
args[2], const_args[2]);
|
|
||||||
break;
|
|
||||||
case INDEX_op_ctz_i32:
|
case INDEX_op_ctz_i32:
|
||||||
tcg_out_cntxz(s, TCG_TYPE_I32, CNTTZW, args[0], args[1],
|
tcg_out_cntxz(s, TCG_TYPE_I32, CNTTZW, args[0], args[1],
|
||||||
args[2], const_args[2]);
|
args[2], const_args[2]);
|
||||||
|
@ -3362,10 +3378,6 @@ static void tcg_out_op(TCGContext *s, TCGOpcode opc, TCGType type,
|
||||||
tcg_out32(s, CNTPOPW | SAB(args[1], args[0], 0));
|
tcg_out32(s, CNTPOPW | SAB(args[1], args[0], 0));
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case INDEX_op_clz_i64:
|
|
||||||
tcg_out_cntxz(s, TCG_TYPE_I64, CNTLZD, args[0], args[1],
|
|
||||||
args[2], const_args[2]);
|
|
||||||
break;
|
|
||||||
case INDEX_op_ctz_i64:
|
case INDEX_op_ctz_i64:
|
||||||
tcg_out_cntxz(s, TCG_TYPE_I64, CNTTZD, args[0], args[1],
|
tcg_out_cntxz(s, TCG_TYPE_I64, CNTTZD, args[0], args[1],
|
||||||
args[2], const_args[2]);
|
args[2], const_args[2]);
|
||||||
|
@ -4228,9 +4240,7 @@ tcg_target_op_def(TCGOpcode op, TCGType type, unsigned flags)
|
||||||
case INDEX_op_st_i64:
|
case INDEX_op_st_i64:
|
||||||
return C_O0_I2(r, r);
|
return C_O0_I2(r, r);
|
||||||
|
|
||||||
case INDEX_op_clz_i32:
|
|
||||||
case INDEX_op_ctz_i32:
|
case INDEX_op_ctz_i32:
|
||||||
case INDEX_op_clz_i64:
|
|
||||||
case INDEX_op_ctz_i64:
|
case INDEX_op_ctz_i64:
|
||||||
return C_O1_I2(r, r, rZW);
|
return C_O1_I2(r, r, rZW);
|
||||||
|
|
||||||
|
|
|
@ -18,7 +18,6 @@
|
||||||
#define TCG_TARGET_HAS_muls2_i32 0
|
#define TCG_TARGET_HAS_muls2_i32 0
|
||||||
#define TCG_TARGET_HAS_bswap16_i32 (cpuinfo & CPUINFO_ZBB)
|
#define TCG_TARGET_HAS_bswap16_i32 (cpuinfo & CPUINFO_ZBB)
|
||||||
#define TCG_TARGET_HAS_bswap32_i32 (cpuinfo & CPUINFO_ZBB)
|
#define TCG_TARGET_HAS_bswap32_i32 (cpuinfo & CPUINFO_ZBB)
|
||||||
#define TCG_TARGET_HAS_clz_i32 (cpuinfo & CPUINFO_ZBB)
|
|
||||||
#define TCG_TARGET_HAS_ctz_i32 (cpuinfo & CPUINFO_ZBB)
|
#define TCG_TARGET_HAS_ctz_i32 (cpuinfo & CPUINFO_ZBB)
|
||||||
#define TCG_TARGET_HAS_ctpop_i32 (cpuinfo & CPUINFO_ZBB)
|
#define TCG_TARGET_HAS_ctpop_i32 (cpuinfo & CPUINFO_ZBB)
|
||||||
#define TCG_TARGET_HAS_qemu_st8_i32 0
|
#define TCG_TARGET_HAS_qemu_st8_i32 0
|
||||||
|
@ -29,7 +28,6 @@
|
||||||
#define TCG_TARGET_HAS_bswap16_i64 (cpuinfo & CPUINFO_ZBB)
|
#define TCG_TARGET_HAS_bswap16_i64 (cpuinfo & CPUINFO_ZBB)
|
||||||
#define TCG_TARGET_HAS_bswap32_i64 (cpuinfo & CPUINFO_ZBB)
|
#define TCG_TARGET_HAS_bswap32_i64 (cpuinfo & CPUINFO_ZBB)
|
||||||
#define TCG_TARGET_HAS_bswap64_i64 (cpuinfo & CPUINFO_ZBB)
|
#define TCG_TARGET_HAS_bswap64_i64 (cpuinfo & CPUINFO_ZBB)
|
||||||
#define TCG_TARGET_HAS_clz_i64 (cpuinfo & CPUINFO_ZBB)
|
|
||||||
#define TCG_TARGET_HAS_ctz_i64 (cpuinfo & CPUINFO_ZBB)
|
#define TCG_TARGET_HAS_ctz_i64 (cpuinfo & CPUINFO_ZBB)
|
||||||
#define TCG_TARGET_HAS_ctpop_i64 (cpuinfo & CPUINFO_ZBB)
|
#define TCG_TARGET_HAS_ctpop_i64 (cpuinfo & CPUINFO_ZBB)
|
||||||
#define TCG_TARGET_HAS_add2_i64 1
|
#define TCG_TARGET_HAS_add2_i64 1
|
||||||
|
|
|
@ -1997,6 +1997,32 @@ static const TCGOutOpBinary outop_andc = {
|
||||||
.out_rrr = tgen_andc,
|
.out_rrr = tgen_andc,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static void tgen_clz(TCGContext *s, TCGType type,
|
||||||
|
TCGReg a0, TCGReg a1, TCGReg a2)
|
||||||
|
{
|
||||||
|
RISCVInsn insn = type == TCG_TYPE_I32 ? OPC_CLZW : OPC_CLZ;
|
||||||
|
tcg_out_cltz(s, type, insn, a0, a1, a2, false);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void tgen_clzi(TCGContext *s, TCGType type,
|
||||||
|
TCGReg a0, TCGReg a1, tcg_target_long a2)
|
||||||
|
{
|
||||||
|
RISCVInsn insn = type == TCG_TYPE_I32 ? OPC_CLZW : OPC_CLZ;
|
||||||
|
tcg_out_cltz(s, type, insn, a0, a1, a2, true);
|
||||||
|
}
|
||||||
|
|
||||||
|
static TCGConstraintSetIndex cset_clzctz(TCGType type, unsigned flags)
|
||||||
|
{
|
||||||
|
return cpuinfo & CPUINFO_ZBB ? C_N1_I2(r, r, rM) : C_NotImplemented;
|
||||||
|
}
|
||||||
|
|
||||||
|
static const TCGOutOpBinary outop_clz = {
|
||||||
|
.base.static_constraint = C_Dynamic,
|
||||||
|
.base.dynamic_constraint = cset_clzctz,
|
||||||
|
.out_rrr = tgen_clz,
|
||||||
|
.out_rri = tgen_clzi,
|
||||||
|
};
|
||||||
|
|
||||||
static void tgen_divs(TCGContext *s, TCGType type,
|
static void tgen_divs(TCGContext *s, TCGType type,
|
||||||
TCGReg a0, TCGReg a1, TCGReg a2)
|
TCGReg a0, TCGReg a1, TCGReg a2)
|
||||||
{
|
{
|
||||||
|
@ -2398,12 +2424,6 @@ static void tcg_out_op(TCGContext *s, TCGOpcode opc, TCGType type,
|
||||||
tcg_out_opc_imm(s, OPC_CPOP, a0, a1, 0);
|
tcg_out_opc_imm(s, OPC_CPOP, a0, a1, 0);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case INDEX_op_clz_i32:
|
|
||||||
tcg_out_cltz(s, TCG_TYPE_I32, OPC_CLZW, a0, a1, a2, c2);
|
|
||||||
break;
|
|
||||||
case INDEX_op_clz_i64:
|
|
||||||
tcg_out_cltz(s, TCG_TYPE_I64, OPC_CLZ, a0, a1, a2, c2);
|
|
||||||
break;
|
|
||||||
case INDEX_op_ctz_i32:
|
case INDEX_op_ctz_i32:
|
||||||
tcg_out_cltz(s, TCG_TYPE_I32, OPC_CTZW, a0, a1, a2, c2);
|
tcg_out_cltz(s, TCG_TYPE_I32, OPC_CTZW, a0, a1, a2, c2);
|
||||||
break;
|
break;
|
||||||
|
@ -2793,8 +2813,6 @@ tcg_target_op_def(TCGOpcode op, TCGType type, unsigned flags)
|
||||||
case INDEX_op_negsetcond_i64:
|
case INDEX_op_negsetcond_i64:
|
||||||
return C_O1_I2(r, r, rI);
|
return C_O1_I2(r, r, rI);
|
||||||
|
|
||||||
case INDEX_op_clz_i32:
|
|
||||||
case INDEX_op_clz_i64:
|
|
||||||
case INDEX_op_ctz_i32:
|
case INDEX_op_ctz_i32:
|
||||||
case INDEX_op_ctz_i64:
|
case INDEX_op_ctz_i64:
|
||||||
return C_N1_I2(r, r, rM);
|
return C_N1_I2(r, r, rM);
|
||||||
|
|
|
@ -31,7 +31,6 @@ extern uint64_t s390_facilities[3];
|
||||||
/* optional instructions */
|
/* optional instructions */
|
||||||
#define TCG_TARGET_HAS_bswap16_i32 1
|
#define TCG_TARGET_HAS_bswap16_i32 1
|
||||||
#define TCG_TARGET_HAS_bswap32_i32 1
|
#define TCG_TARGET_HAS_bswap32_i32 1
|
||||||
#define TCG_TARGET_HAS_clz_i32 0
|
|
||||||
#define TCG_TARGET_HAS_ctz_i32 0
|
#define TCG_TARGET_HAS_ctz_i32 0
|
||||||
#define TCG_TARGET_HAS_ctpop_i32 1
|
#define TCG_TARGET_HAS_ctpop_i32 1
|
||||||
#define TCG_TARGET_HAS_extract2_i32 0
|
#define TCG_TARGET_HAS_extract2_i32 0
|
||||||
|
@ -46,7 +45,6 @@ extern uint64_t s390_facilities[3];
|
||||||
#define TCG_TARGET_HAS_bswap16_i64 1
|
#define TCG_TARGET_HAS_bswap16_i64 1
|
||||||
#define TCG_TARGET_HAS_bswap32_i64 1
|
#define TCG_TARGET_HAS_bswap32_i64 1
|
||||||
#define TCG_TARGET_HAS_bswap64_i64 1
|
#define TCG_TARGET_HAS_bswap64_i64 1
|
||||||
#define TCG_TARGET_HAS_clz_i64 1
|
|
||||||
#define TCG_TARGET_HAS_ctz_i64 0
|
#define TCG_TARGET_HAS_ctz_i64 0
|
||||||
#define TCG_TARGET_HAS_ctpop_i64 1
|
#define TCG_TARGET_HAS_ctpop_i64 1
|
||||||
#define TCG_TARGET_HAS_extract2_i64 0
|
#define TCG_TARGET_HAS_extract2_i64 0
|
||||||
|
|
|
@ -1514,27 +1514,6 @@ static void tgen_movcond(TCGContext *s, TCGType type, TCGCond c, TCGReg dest,
|
||||||
tgen_movcond_int(s, type, dest, v3, v3const, v4, cc, inv_cc);
|
tgen_movcond_int(s, type, dest, v3, v3const, v4, cc, inv_cc);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void tgen_clz(TCGContext *s, TCGReg dest, TCGReg a1,
|
|
||||||
TCGArg a2, int a2const)
|
|
||||||
{
|
|
||||||
/* Since this sets both R and R+1, we have no choice but to store the
|
|
||||||
result into R0, allowing R1 == TCG_TMP0 to be clobbered as well. */
|
|
||||||
QEMU_BUILD_BUG_ON(TCG_TMP0 != TCG_REG_R1);
|
|
||||||
tcg_out_insn(s, RRE, FLOGR, TCG_REG_R0, a1);
|
|
||||||
|
|
||||||
if (a2const && a2 == 64) {
|
|
||||||
tcg_out_mov(s, TCG_TYPE_I64, dest, TCG_REG_R0);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Conditions from FLOGR are:
|
|
||||||
* 2 -> one bit found
|
|
||||||
* 8 -> no one bit found
|
|
||||||
*/
|
|
||||||
tgen_movcond_int(s, TCG_TYPE_I64, dest, a2, a2const, TCG_REG_R0, 8, 2);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void tgen_ctpop(TCGContext *s, TCGType type, TCGReg dest, TCGReg src)
|
static void tgen_ctpop(TCGContext *s, TCGType type, TCGReg dest, TCGReg src)
|
||||||
{
|
{
|
||||||
/* With MIE3, and bit 0 of m4 set, we get the complete result. */
|
/* With MIE3, and bit 0 of m4 set, we get the complete result. */
|
||||||
|
@ -2242,6 +2221,53 @@ static const TCGOutOpBinary outop_andc = {
|
||||||
.out_rrr = tgen_andc,
|
.out_rrr = tgen_andc,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static void tgen_clz_int(TCGContext *s, TCGReg dest, TCGReg a1,
|
||||||
|
TCGArg a2, int a2const)
|
||||||
|
{
|
||||||
|
/*
|
||||||
|
* Since this sets both R and R+1, we have no choice but to store the
|
||||||
|
* result into R0, allowing R1 == TCG_TMP0 to be clobbered as well.
|
||||||
|
*/
|
||||||
|
QEMU_BUILD_BUG_ON(TCG_TMP0 != TCG_REG_R1);
|
||||||
|
tcg_out_insn(s, RRE, FLOGR, TCG_REG_R0, a1);
|
||||||
|
|
||||||
|
if (a2const && a2 == 64) {
|
||||||
|
tcg_out_mov(s, TCG_TYPE_I64, dest, TCG_REG_R0);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Conditions from FLOGR are:
|
||||||
|
* 2 -> one bit found
|
||||||
|
* 8 -> no one bit found
|
||||||
|
*/
|
||||||
|
tgen_movcond_int(s, TCG_TYPE_I64, dest, a2, a2const, TCG_REG_R0, 8, 2);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void tgen_clz(TCGContext *s, TCGType type,
|
||||||
|
TCGReg a0, TCGReg a1, TCGReg a2)
|
||||||
|
{
|
||||||
|
tgen_clz_int(s, a0, a1, a2, false);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void tgen_clzi(TCGContext *s, TCGType type,
|
||||||
|
TCGReg a0, TCGReg a1, tcg_target_long a2)
|
||||||
|
{
|
||||||
|
tgen_clz_int(s, a0, a1, a2, true);
|
||||||
|
}
|
||||||
|
|
||||||
|
static TCGConstraintSetIndex cset_clz(TCGType type, unsigned flags)
|
||||||
|
{
|
||||||
|
return type == TCG_TYPE_I64 ? C_O1_I2(r, r, rI) : C_NotImplemented;
|
||||||
|
}
|
||||||
|
|
||||||
|
static const TCGOutOpBinary outop_clz = {
|
||||||
|
.base.static_constraint = C_Dynamic,
|
||||||
|
.base.dynamic_constraint = cset_clz,
|
||||||
|
.out_rrr = tgen_clz,
|
||||||
|
.out_rri = tgen_clzi,
|
||||||
|
};
|
||||||
|
|
||||||
static const TCGOutOpBinary outop_divs = {
|
static const TCGOutOpBinary outop_divs = {
|
||||||
.base.static_constraint = C_NotImplemented,
|
.base.static_constraint = C_NotImplemented,
|
||||||
};
|
};
|
||||||
|
@ -2884,10 +2910,6 @@ static void tcg_out_op(TCGContext *s, TCGOpcode opc, TCGType type,
|
||||||
tgen_sextract(s, args[0], args[1], args[2], args[3]);
|
tgen_sextract(s, args[0], args[1], args[2], args[3]);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case INDEX_op_clz_i64:
|
|
||||||
tgen_clz(s, args[0], args[1], args[2], const_args[2]);
|
|
||||||
break;
|
|
||||||
|
|
||||||
case INDEX_op_ctpop_i32:
|
case INDEX_op_ctpop_i32:
|
||||||
tgen_ctpop(s, TCG_TYPE_I32, args[0], args[1]);
|
tgen_ctpop(s, TCG_TYPE_I32, args[0], args[1]);
|
||||||
break;
|
break;
|
||||||
|
@ -3387,9 +3409,6 @@ tcg_target_op_def(TCGOpcode op, TCGType type, unsigned flags)
|
||||||
case INDEX_op_negsetcond_i64:
|
case INDEX_op_negsetcond_i64:
|
||||||
return C_O1_I2(r, r, rC);
|
return C_O1_I2(r, r, rC);
|
||||||
|
|
||||||
case INDEX_op_clz_i64:
|
|
||||||
return C_O1_I2(r, r, rI);
|
|
||||||
|
|
||||||
case INDEX_op_brcond_i32:
|
case INDEX_op_brcond_i32:
|
||||||
return C_O0_I2(r, ri);
|
return C_O0_I2(r, ri);
|
||||||
case INDEX_op_brcond_i64:
|
case INDEX_op_brcond_i64:
|
||||||
|
|
|
@ -16,7 +16,6 @@ extern bool use_vis3_instructions;
|
||||||
/* optional instructions */
|
/* optional instructions */
|
||||||
#define TCG_TARGET_HAS_bswap16_i32 0
|
#define TCG_TARGET_HAS_bswap16_i32 0
|
||||||
#define TCG_TARGET_HAS_bswap32_i32 0
|
#define TCG_TARGET_HAS_bswap32_i32 0
|
||||||
#define TCG_TARGET_HAS_clz_i32 0
|
|
||||||
#define TCG_TARGET_HAS_ctz_i32 0
|
#define TCG_TARGET_HAS_ctz_i32 0
|
||||||
#define TCG_TARGET_HAS_ctpop_i32 0
|
#define TCG_TARGET_HAS_ctpop_i32 0
|
||||||
#define TCG_TARGET_HAS_extract2_i32 0
|
#define TCG_TARGET_HAS_extract2_i32 0
|
||||||
|
@ -31,7 +30,6 @@ extern bool use_vis3_instructions;
|
||||||
#define TCG_TARGET_HAS_bswap16_i64 0
|
#define TCG_TARGET_HAS_bswap16_i64 0
|
||||||
#define TCG_TARGET_HAS_bswap32_i64 0
|
#define TCG_TARGET_HAS_bswap32_i64 0
|
||||||
#define TCG_TARGET_HAS_bswap64_i64 0
|
#define TCG_TARGET_HAS_bswap64_i64 0
|
||||||
#define TCG_TARGET_HAS_clz_i64 0
|
|
||||||
#define TCG_TARGET_HAS_ctz_i64 0
|
#define TCG_TARGET_HAS_ctz_i64 0
|
||||||
#define TCG_TARGET_HAS_ctpop_i64 0
|
#define TCG_TARGET_HAS_ctpop_i64 0
|
||||||
#define TCG_TARGET_HAS_extract2_i64 0
|
#define TCG_TARGET_HAS_extract2_i64 0
|
||||||
|
|
|
@ -1318,6 +1318,10 @@ static const TCGOutOpBinary outop_andc = {
|
||||||
.out_rrr = tgen_andc,
|
.out_rrr = tgen_andc,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static const TCGOutOpBinary outop_clz = {
|
||||||
|
.base.static_constraint = C_NotImplemented,
|
||||||
|
};
|
||||||
|
|
||||||
static void tgen_divs_rJ(TCGContext *s, TCGType type,
|
static void tgen_divs_rJ(TCGContext *s, TCGType type,
|
||||||
TCGReg a0, TCGReg a1, TCGArg a2, bool c2)
|
TCGReg a0, TCGReg a1, TCGArg a2, bool c2)
|
||||||
{
|
{
|
||||||
|
|
|
@ -15,7 +15,6 @@
|
||||||
#define TCG_TARGET_HAS_bswap16_i64 0
|
#define TCG_TARGET_HAS_bswap16_i64 0
|
||||||
#define TCG_TARGET_HAS_bswap32_i64 0
|
#define TCG_TARGET_HAS_bswap32_i64 0
|
||||||
#define TCG_TARGET_HAS_bswap64_i64 0
|
#define TCG_TARGET_HAS_bswap64_i64 0
|
||||||
#define TCG_TARGET_HAS_clz_i64 0
|
|
||||||
#define TCG_TARGET_HAS_ctz_i64 0
|
#define TCG_TARGET_HAS_ctz_i64 0
|
||||||
#define TCG_TARGET_HAS_ctpop_i64 0
|
#define TCG_TARGET_HAS_ctpop_i64 0
|
||||||
#define TCG_TARGET_HAS_extract2_i64 0
|
#define TCG_TARGET_HAS_extract2_i64 0
|
||||||
|
|
72
tcg/tcg-op.c
72
tcg/tcg-op.c
|
@ -723,9 +723,9 @@ void tcg_gen_orc_i32(TCGv_i32 ret, TCGv_i32 arg1, TCGv_i32 arg2)
|
||||||
|
|
||||||
void tcg_gen_clz_i32(TCGv_i32 ret, TCGv_i32 arg1, TCGv_i32 arg2)
|
void tcg_gen_clz_i32(TCGv_i32 ret, TCGv_i32 arg1, TCGv_i32 arg2)
|
||||||
{
|
{
|
||||||
if (TCG_TARGET_HAS_clz_i32) {
|
if (tcg_op_supported(INDEX_op_clz_i32, TCG_TYPE_I32, 0)) {
|
||||||
tcg_gen_op3_i32(INDEX_op_clz_i32, ret, arg1, arg2);
|
tcg_gen_op3_i32(INDEX_op_clz_i32, ret, arg1, arg2);
|
||||||
} else if (TCG_TARGET_HAS_clz_i64) {
|
} else if (tcg_op_supported(INDEX_op_clz_i64, TCG_TYPE_I64, 0)) {
|
||||||
TCGv_i64 t1 = tcg_temp_ebb_new_i64();
|
TCGv_i64 t1 = tcg_temp_ebb_new_i64();
|
||||||
TCGv_i64 t2 = tcg_temp_ebb_new_i64();
|
TCGv_i64 t2 = tcg_temp_ebb_new_i64();
|
||||||
tcg_gen_extu_i32_i64(t1, arg1);
|
tcg_gen_extu_i32_i64(t1, arg1);
|
||||||
|
@ -748,9 +748,13 @@ void tcg_gen_clzi_i32(TCGv_i32 ret, TCGv_i32 arg1, uint32_t arg2)
|
||||||
|
|
||||||
void tcg_gen_ctz_i32(TCGv_i32 ret, TCGv_i32 arg1, TCGv_i32 arg2)
|
void tcg_gen_ctz_i32(TCGv_i32 ret, TCGv_i32 arg1, TCGv_i32 arg2)
|
||||||
{
|
{
|
||||||
|
TCGv_i32 z, t;
|
||||||
|
|
||||||
if (TCG_TARGET_HAS_ctz_i32) {
|
if (TCG_TARGET_HAS_ctz_i32) {
|
||||||
tcg_gen_op3_i32(INDEX_op_ctz_i32, ret, arg1, arg2);
|
tcg_gen_op3_i32(INDEX_op_ctz_i32, ret, arg1, arg2);
|
||||||
} else if (TCG_TARGET_HAS_ctz_i64) {
|
return;
|
||||||
|
}
|
||||||
|
if (TCG_TARGET_HAS_ctz_i64) {
|
||||||
TCGv_i64 t1 = tcg_temp_ebb_new_i64();
|
TCGv_i64 t1 = tcg_temp_ebb_new_i64();
|
||||||
TCGv_i64 t2 = tcg_temp_ebb_new_i64();
|
TCGv_i64 t2 = tcg_temp_ebb_new_i64();
|
||||||
tcg_gen_extu_i32_i64(t1, arg1);
|
tcg_gen_extu_i32_i64(t1, arg1);
|
||||||
|
@ -759,29 +763,28 @@ void tcg_gen_ctz_i32(TCGv_i32 ret, TCGv_i32 arg1, TCGv_i32 arg2)
|
||||||
tcg_gen_extrl_i64_i32(ret, t1);
|
tcg_gen_extrl_i64_i32(ret, t1);
|
||||||
tcg_temp_free_i64(t1);
|
tcg_temp_free_i64(t1);
|
||||||
tcg_temp_free_i64(t2);
|
tcg_temp_free_i64(t2);
|
||||||
} else if (TCG_TARGET_HAS_ctpop_i32
|
return;
|
||||||
|| TCG_TARGET_HAS_ctpop_i64
|
}
|
||||||
|| TCG_TARGET_HAS_clz_i32
|
|
||||||
|| TCG_TARGET_HAS_clz_i64) {
|
|
||||||
TCGv_i32 z, t = tcg_temp_ebb_new_i32();
|
|
||||||
|
|
||||||
if (TCG_TARGET_HAS_ctpop_i32 || TCG_TARGET_HAS_ctpop_i64) {
|
if (TCG_TARGET_HAS_ctpop_i32 || TCG_TARGET_HAS_ctpop_i64) {
|
||||||
|
t = tcg_temp_ebb_new_i32();
|
||||||
tcg_gen_subi_i32(t, arg1, 1);
|
tcg_gen_subi_i32(t, arg1, 1);
|
||||||
tcg_gen_andc_i32(t, t, arg1);
|
tcg_gen_andc_i32(t, t, arg1);
|
||||||
tcg_gen_ctpop_i32(t, t);
|
tcg_gen_ctpop_i32(t, t);
|
||||||
} else {
|
} else if (tcg_op_supported(INDEX_op_clz_i32, TCG_TYPE_I32, 0) ||
|
||||||
/* Since all non-x86 hosts have clz(0) == 32, don't fight it. */
|
tcg_op_supported(INDEX_op_clz_i64, TCG_TYPE_I64, 0)) {
|
||||||
|
t = tcg_temp_ebb_new_i32();
|
||||||
tcg_gen_neg_i32(t, arg1);
|
tcg_gen_neg_i32(t, arg1);
|
||||||
tcg_gen_and_i32(t, t, arg1);
|
tcg_gen_and_i32(t, t, arg1);
|
||||||
tcg_gen_clzi_i32(t, t, 32);
|
tcg_gen_clzi_i32(t, t, 32);
|
||||||
tcg_gen_xori_i32(t, t, 31);
|
tcg_gen_xori_i32(t, t, 31);
|
||||||
|
} else {
|
||||||
|
gen_helper_ctz_i32(ret, arg1, arg2);
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
z = tcg_constant_i32(0);
|
z = tcg_constant_i32(0);
|
||||||
tcg_gen_movcond_i32(TCG_COND_EQ, ret, arg1, z, arg2, t);
|
tcg_gen_movcond_i32(TCG_COND_EQ, ret, arg1, z, arg2, t);
|
||||||
tcg_temp_free_i32(t);
|
tcg_temp_free_i32(t);
|
||||||
} else {
|
|
||||||
gen_helper_ctz_i32(ret, arg1, arg2);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void tcg_gen_ctzi_i32(TCGv_i32 ret, TCGv_i32 arg1, uint32_t arg2)
|
void tcg_gen_ctzi_i32(TCGv_i32 ret, TCGv_i32 arg1, uint32_t arg2)
|
||||||
|
@ -800,7 +803,8 @@ void tcg_gen_ctzi_i32(TCGv_i32 ret, TCGv_i32 arg1, uint32_t arg2)
|
||||||
|
|
||||||
void tcg_gen_clrsb_i32(TCGv_i32 ret, TCGv_i32 arg)
|
void tcg_gen_clrsb_i32(TCGv_i32 ret, TCGv_i32 arg)
|
||||||
{
|
{
|
||||||
if (TCG_TARGET_HAS_clz_i32) {
|
if (tcg_op_supported(INDEX_op_clz_i32, TCG_TYPE_I32, 0) ||
|
||||||
|
tcg_op_supported(INDEX_op_clz_i64, TCG_TYPE_I64, 0)) {
|
||||||
TCGv_i32 t = tcg_temp_ebb_new_i32();
|
TCGv_i32 t = tcg_temp_ebb_new_i32();
|
||||||
tcg_gen_sari_i32(t, arg, 31);
|
tcg_gen_sari_i32(t, arg, 31);
|
||||||
tcg_gen_xor_i32(t, t, arg);
|
tcg_gen_xor_i32(t, t, arg);
|
||||||
|
@ -2336,7 +2340,7 @@ void tcg_gen_orc_i64(TCGv_i64 ret, TCGv_i64 arg1, TCGv_i64 arg2)
|
||||||
|
|
||||||
void tcg_gen_clz_i64(TCGv_i64 ret, TCGv_i64 arg1, TCGv_i64 arg2)
|
void tcg_gen_clz_i64(TCGv_i64 ret, TCGv_i64 arg1, TCGv_i64 arg2)
|
||||||
{
|
{
|
||||||
if (TCG_TARGET_HAS_clz_i64) {
|
if (tcg_op_supported(INDEX_op_clz_i64, TCG_TYPE_I64, 0)) {
|
||||||
tcg_gen_op3_i64(INDEX_op_clz_i64, ret, arg1, arg2);
|
tcg_gen_op3_i64(INDEX_op_clz_i64, ret, arg1, arg2);
|
||||||
} else {
|
} else {
|
||||||
gen_helper_clz_i64(ret, arg1, arg2);
|
gen_helper_clz_i64(ret, arg1, arg2);
|
||||||
|
@ -2346,8 +2350,8 @@ void tcg_gen_clz_i64(TCGv_i64 ret, TCGv_i64 arg1, TCGv_i64 arg2)
|
||||||
void tcg_gen_clzi_i64(TCGv_i64 ret, TCGv_i64 arg1, uint64_t arg2)
|
void tcg_gen_clzi_i64(TCGv_i64 ret, TCGv_i64 arg1, uint64_t arg2)
|
||||||
{
|
{
|
||||||
if (TCG_TARGET_REG_BITS == 32
|
if (TCG_TARGET_REG_BITS == 32
|
||||||
&& TCG_TARGET_HAS_clz_i32
|
&& arg2 <= 0xffffffffu
|
||||||
&& arg2 <= 0xffffffffu) {
|
&& tcg_op_supported(INDEX_op_clz_i32, TCG_TYPE_I32, 0)) {
|
||||||
TCGv_i32 t = tcg_temp_ebb_new_i32();
|
TCGv_i32 t = tcg_temp_ebb_new_i32();
|
||||||
tcg_gen_clzi_i32(t, TCGV_LOW(arg1), arg2 - 32);
|
tcg_gen_clzi_i32(t, TCGV_LOW(arg1), arg2 - 32);
|
||||||
tcg_gen_addi_i32(t, t, 32);
|
tcg_gen_addi_i32(t, t, 32);
|
||||||
|
@ -2361,45 +2365,47 @@ void tcg_gen_clzi_i64(TCGv_i64 ret, TCGv_i64 arg1, uint64_t arg2)
|
||||||
|
|
||||||
void tcg_gen_ctz_i64(TCGv_i64 ret, TCGv_i64 arg1, TCGv_i64 arg2)
|
void tcg_gen_ctz_i64(TCGv_i64 ret, TCGv_i64 arg1, TCGv_i64 arg2)
|
||||||
{
|
{
|
||||||
|
TCGv_i64 z, t;
|
||||||
|
|
||||||
if (TCG_TARGET_HAS_ctz_i64) {
|
if (TCG_TARGET_HAS_ctz_i64) {
|
||||||
tcg_gen_op3_i64(INDEX_op_ctz_i64, ret, arg1, arg2);
|
tcg_gen_op3_i64(INDEX_op_ctz_i64, ret, arg1, arg2);
|
||||||
} else if (TCG_TARGET_HAS_ctpop_i64 || TCG_TARGET_HAS_clz_i64) {
|
return;
|
||||||
TCGv_i64 z, t = tcg_temp_ebb_new_i64();
|
}
|
||||||
|
|
||||||
if (TCG_TARGET_HAS_ctpop_i64) {
|
if (TCG_TARGET_HAS_ctpop_i64) {
|
||||||
|
t = tcg_temp_ebb_new_i64();
|
||||||
tcg_gen_subi_i64(t, arg1, 1);
|
tcg_gen_subi_i64(t, arg1, 1);
|
||||||
tcg_gen_andc_i64(t, t, arg1);
|
tcg_gen_andc_i64(t, t, arg1);
|
||||||
tcg_gen_ctpop_i64(t, t);
|
tcg_gen_ctpop_i64(t, t);
|
||||||
} else {
|
} else if (tcg_op_supported(INDEX_op_clz_i64, TCG_TYPE_I64, 0)) {
|
||||||
/* Since all non-x86 hosts have clz(0) == 64, don't fight it. */
|
t = tcg_temp_ebb_new_i64();
|
||||||
tcg_gen_neg_i64(t, arg1);
|
tcg_gen_neg_i64(t, arg1);
|
||||||
tcg_gen_and_i64(t, t, arg1);
|
tcg_gen_and_i64(t, t, arg1);
|
||||||
tcg_gen_clzi_i64(t, t, 64);
|
tcg_gen_clzi_i64(t, t, 64);
|
||||||
tcg_gen_xori_i64(t, t, 63);
|
tcg_gen_xori_i64(t, t, 63);
|
||||||
|
} else {
|
||||||
|
gen_helper_ctz_i64(ret, arg1, arg2);
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
z = tcg_constant_i64(0);
|
z = tcg_constant_i64(0);
|
||||||
tcg_gen_movcond_i64(TCG_COND_EQ, ret, arg1, z, arg2, t);
|
tcg_gen_movcond_i64(TCG_COND_EQ, ret, arg1, z, arg2, t);
|
||||||
tcg_temp_free_i64(t);
|
tcg_temp_free_i64(t);
|
||||||
tcg_temp_free_i64(z);
|
|
||||||
} else {
|
|
||||||
gen_helper_ctz_i64(ret, arg1, arg2);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void tcg_gen_ctzi_i64(TCGv_i64 ret, TCGv_i64 arg1, uint64_t arg2)
|
void tcg_gen_ctzi_i64(TCGv_i64 ret, TCGv_i64 arg1, uint64_t arg2)
|
||||||
{
|
{
|
||||||
if (TCG_TARGET_REG_BITS == 32
|
if (TCG_TARGET_REG_BITS == 32
|
||||||
&& TCG_TARGET_HAS_ctz_i32
|
&& arg2 <= 0xffffffffu
|
||||||
&& arg2 <= 0xffffffffu) {
|
&& tcg_op_supported(INDEX_op_ctz_i32, TCG_TYPE_I32, 0)) {
|
||||||
TCGv_i32 t32 = tcg_temp_ebb_new_i32();
|
TCGv_i32 t32 = tcg_temp_ebb_new_i32();
|
||||||
tcg_gen_ctzi_i32(t32, TCGV_HIGH(arg1), arg2 - 32);
|
tcg_gen_ctzi_i32(t32, TCGV_HIGH(arg1), arg2 - 32);
|
||||||
tcg_gen_addi_i32(t32, t32, 32);
|
tcg_gen_addi_i32(t32, t32, 32);
|
||||||
tcg_gen_ctz_i32(TCGV_LOW(ret), TCGV_LOW(arg1), t32);
|
tcg_gen_ctz_i32(TCGV_LOW(ret), TCGV_LOW(arg1), t32);
|
||||||
tcg_gen_movi_i32(TCGV_HIGH(ret), 0);
|
tcg_gen_movi_i32(TCGV_HIGH(ret), 0);
|
||||||
tcg_temp_free_i32(t32);
|
tcg_temp_free_i32(t32);
|
||||||
} else if (!TCG_TARGET_HAS_ctz_i64
|
} else if (arg2 == 64
|
||||||
&& TCG_TARGET_HAS_ctpop_i64
|
&& !tcg_op_supported(INDEX_op_ctz_i64, TCG_TYPE_I64, 0)
|
||||||
&& arg2 == 64) {
|
&& TCG_TARGET_HAS_ctpop_i64) {
|
||||||
/* This equivalence has the advantage of not requiring a fixup. */
|
/* This equivalence has the advantage of not requiring a fixup. */
|
||||||
TCGv_i64 t = tcg_temp_ebb_new_i64();
|
TCGv_i64 t = tcg_temp_ebb_new_i64();
|
||||||
tcg_gen_subi_i64(t, arg1, 1);
|
tcg_gen_subi_i64(t, arg1, 1);
|
||||||
|
@ -2413,7 +2419,7 @@ void tcg_gen_ctzi_i64(TCGv_i64 ret, TCGv_i64 arg1, uint64_t arg2)
|
||||||
|
|
||||||
void tcg_gen_clrsb_i64(TCGv_i64 ret, TCGv_i64 arg)
|
void tcg_gen_clrsb_i64(TCGv_i64 ret, TCGv_i64 arg)
|
||||||
{
|
{
|
||||||
if (TCG_TARGET_HAS_clz_i64 || TCG_TARGET_HAS_clz_i32) {
|
if (tcg_op_supported(INDEX_op_clz_i64, TCG_TYPE_I64, 0)) {
|
||||||
TCGv_i64 t = tcg_temp_ebb_new_i64();
|
TCGv_i64 t = tcg_temp_ebb_new_i64();
|
||||||
tcg_gen_sari_i64(t, arg, 63);
|
tcg_gen_sari_i64(t, arg, 63);
|
||||||
tcg_gen_xor_i64(t, t, arg);
|
tcg_gen_xor_i64(t, t, arg);
|
||||||
|
|
|
@ -1026,6 +1026,8 @@ static const TCGOutOp * const all_outop[NB_OPS] = {
|
||||||
OUTOP(INDEX_op_add, TCGOutOpBinary, outop_add),
|
OUTOP(INDEX_op_add, TCGOutOpBinary, outop_add),
|
||||||
OUTOP(INDEX_op_and, TCGOutOpBinary, outop_and),
|
OUTOP(INDEX_op_and, TCGOutOpBinary, outop_and),
|
||||||
OUTOP(INDEX_op_andc, TCGOutOpBinary, outop_andc),
|
OUTOP(INDEX_op_andc, TCGOutOpBinary, outop_andc),
|
||||||
|
OUTOP(INDEX_op_clz_i32, TCGOutOpBinary, outop_clz),
|
||||||
|
OUTOP(INDEX_op_clz_i64, TCGOutOpBinary, outop_clz),
|
||||||
OUTOP(INDEX_op_divs, TCGOutOpBinary, outop_divs),
|
OUTOP(INDEX_op_divs, TCGOutOpBinary, outop_divs),
|
||||||
OUTOP(INDEX_op_divu, TCGOutOpBinary, outop_divu),
|
OUTOP(INDEX_op_divu, TCGOutOpBinary, outop_divu),
|
||||||
OUTOP(INDEX_op_divs2, TCGOutOpDivRem, outop_divs2),
|
OUTOP(INDEX_op_divs2, TCGOutOpDivRem, outop_divs2),
|
||||||
|
@ -2288,8 +2290,6 @@ bool tcg_op_supported(TCGOpcode op, TCGType type, unsigned flags)
|
||||||
return TCG_TARGET_HAS_bswap16_i32;
|
return TCG_TARGET_HAS_bswap16_i32;
|
||||||
case INDEX_op_bswap32_i32:
|
case INDEX_op_bswap32_i32:
|
||||||
return TCG_TARGET_HAS_bswap32_i32;
|
return TCG_TARGET_HAS_bswap32_i32;
|
||||||
case INDEX_op_clz_i32:
|
|
||||||
return TCG_TARGET_HAS_clz_i32;
|
|
||||||
case INDEX_op_ctz_i32:
|
case INDEX_op_ctz_i32:
|
||||||
return TCG_TARGET_HAS_ctz_i32;
|
return TCG_TARGET_HAS_ctz_i32;
|
||||||
case INDEX_op_ctpop_i32:
|
case INDEX_op_ctpop_i32:
|
||||||
|
@ -2333,8 +2333,6 @@ bool tcg_op_supported(TCGOpcode op, TCGType type, unsigned flags)
|
||||||
return TCG_TARGET_HAS_bswap32_i64;
|
return TCG_TARGET_HAS_bswap32_i64;
|
||||||
case INDEX_op_bswap64_i64:
|
case INDEX_op_bswap64_i64:
|
||||||
return TCG_TARGET_HAS_bswap64_i64;
|
return TCG_TARGET_HAS_bswap64_i64;
|
||||||
case INDEX_op_clz_i64:
|
|
||||||
return TCG_TARGET_HAS_clz_i64;
|
|
||||||
case INDEX_op_ctz_i64:
|
case INDEX_op_ctz_i64:
|
||||||
return TCG_TARGET_HAS_ctz_i64;
|
return TCG_TARGET_HAS_ctz_i64;
|
||||||
case INDEX_op_ctpop_i64:
|
case INDEX_op_ctpop_i64:
|
||||||
|
@ -5404,6 +5402,8 @@ static void tcg_reg_alloc_op(TCGContext *s, const TCGOp *op)
|
||||||
case INDEX_op_add:
|
case INDEX_op_add:
|
||||||
case INDEX_op_and:
|
case INDEX_op_and:
|
||||||
case INDEX_op_andc:
|
case INDEX_op_andc:
|
||||||
|
case INDEX_op_clz_i32:
|
||||||
|
case INDEX_op_clz_i64:
|
||||||
case INDEX_op_divs:
|
case INDEX_op_divs:
|
||||||
case INDEX_op_divu:
|
case INDEX_op_divu:
|
||||||
case INDEX_op_eqv:
|
case INDEX_op_eqv:
|
||||||
|
|
|
@ -594,13 +594,11 @@ uintptr_t QEMU_DISABLE_CFI tcg_qemu_tb_exec(CPUArchState *env,
|
||||||
tci_args_rrr(insn, &r0, &r1, &r2);
|
tci_args_rrr(insn, &r0, &r1, &r2);
|
||||||
regs[r0] = (uint32_t)regs[r1] % (uint32_t)regs[r2];
|
regs[r0] = (uint32_t)regs[r1] % (uint32_t)regs[r2];
|
||||||
break;
|
break;
|
||||||
#if TCG_TARGET_HAS_clz_i32
|
case INDEX_op_tci_clz32:
|
||||||
case INDEX_op_clz_i32:
|
|
||||||
tci_args_rrr(insn, &r0, &r1, &r2);
|
tci_args_rrr(insn, &r0, &r1, &r2);
|
||||||
tmp32 = regs[r1];
|
tmp32 = regs[r1];
|
||||||
regs[r0] = tmp32 ? clz32(tmp32) : regs[r2];
|
regs[r0] = tmp32 ? clz32(tmp32) : regs[r2];
|
||||||
break;
|
break;
|
||||||
#endif
|
|
||||||
#if TCG_TARGET_HAS_ctz_i32
|
#if TCG_TARGET_HAS_ctz_i32
|
||||||
case INDEX_op_ctz_i32:
|
case INDEX_op_ctz_i32:
|
||||||
tci_args_rrr(insn, &r0, &r1, &r2);
|
tci_args_rrr(insn, &r0, &r1, &r2);
|
||||||
|
@ -735,12 +733,10 @@ uintptr_t QEMU_DISABLE_CFI tcg_qemu_tb_exec(CPUArchState *env,
|
||||||
tci_args_rrr(insn, &r0, &r1, &r2);
|
tci_args_rrr(insn, &r0, &r1, &r2);
|
||||||
regs[r0] = (uint64_t)regs[r1] % (uint64_t)regs[r2];
|
regs[r0] = (uint64_t)regs[r1] % (uint64_t)regs[r2];
|
||||||
break;
|
break;
|
||||||
#if TCG_TARGET_HAS_clz_i64
|
|
||||||
case INDEX_op_clz_i64:
|
case INDEX_op_clz_i64:
|
||||||
tci_args_rrr(insn, &r0, &r1, &r2);
|
tci_args_rrr(insn, &r0, &r1, &r2);
|
||||||
regs[r0] = regs[r1] ? clz64(regs[r1]) : regs[r2];
|
regs[r0] = regs[r1] ? clz64(regs[r1]) : regs[r2];
|
||||||
break;
|
break;
|
||||||
#endif
|
|
||||||
#if TCG_TARGET_HAS_ctz_i64
|
#if TCG_TARGET_HAS_ctz_i64
|
||||||
case INDEX_op_ctz_i64:
|
case INDEX_op_ctz_i64:
|
||||||
tci_args_rrr(insn, &r0, &r1, &r2);
|
tci_args_rrr(insn, &r0, &r1, &r2);
|
||||||
|
@ -1073,10 +1069,10 @@ int print_insn_tci(bfd_vma addr, disassemble_info *info)
|
||||||
case INDEX_op_shr:
|
case INDEX_op_shr:
|
||||||
case INDEX_op_sub:
|
case INDEX_op_sub:
|
||||||
case INDEX_op_xor:
|
case INDEX_op_xor:
|
||||||
case INDEX_op_clz_i32:
|
|
||||||
case INDEX_op_clz_i64:
|
case INDEX_op_clz_i64:
|
||||||
case INDEX_op_ctz_i32:
|
case INDEX_op_ctz_i32:
|
||||||
case INDEX_op_ctz_i64:
|
case INDEX_op_ctz_i64:
|
||||||
|
case INDEX_op_tci_clz32:
|
||||||
case INDEX_op_tci_divs32:
|
case INDEX_op_tci_divs32:
|
||||||
case INDEX_op_tci_divu32:
|
case INDEX_op_tci_divu32:
|
||||||
case INDEX_op_tci_rems32:
|
case INDEX_op_tci_rems32:
|
||||||
|
|
|
@ -10,7 +10,6 @@
|
||||||
#define TCG_TARGET_HAS_bswap16_i32 1
|
#define TCG_TARGET_HAS_bswap16_i32 1
|
||||||
#define TCG_TARGET_HAS_bswap32_i32 1
|
#define TCG_TARGET_HAS_bswap32_i32 1
|
||||||
#define TCG_TARGET_HAS_extract2_i32 0
|
#define TCG_TARGET_HAS_extract2_i32 0
|
||||||
#define TCG_TARGET_HAS_clz_i32 1
|
|
||||||
#define TCG_TARGET_HAS_ctz_i32 1
|
#define TCG_TARGET_HAS_ctz_i32 1
|
||||||
#define TCG_TARGET_HAS_ctpop_i32 1
|
#define TCG_TARGET_HAS_ctpop_i32 1
|
||||||
#define TCG_TARGET_HAS_negsetcond_i32 0
|
#define TCG_TARGET_HAS_negsetcond_i32 0
|
||||||
|
@ -23,7 +22,6 @@
|
||||||
#define TCG_TARGET_HAS_bswap32_i64 1
|
#define TCG_TARGET_HAS_bswap32_i64 1
|
||||||
#define TCG_TARGET_HAS_bswap64_i64 1
|
#define TCG_TARGET_HAS_bswap64_i64 1
|
||||||
#define TCG_TARGET_HAS_extract2_i64 0
|
#define TCG_TARGET_HAS_extract2_i64 0
|
||||||
#define TCG_TARGET_HAS_clz_i64 1
|
|
||||||
#define TCG_TARGET_HAS_ctz_i64 1
|
#define TCG_TARGET_HAS_ctz_i64 1
|
||||||
#define TCG_TARGET_HAS_ctpop_i64 1
|
#define TCG_TARGET_HAS_ctpop_i64 1
|
||||||
#define TCG_TARGET_HAS_negsetcond_i64 0
|
#define TCG_TARGET_HAS_negsetcond_i64 0
|
||||||
|
|
|
@ -2,6 +2,7 @@
|
||||||
/* These opcodes for use between the tci generator and interpreter. */
|
/* These opcodes for use between the tci generator and interpreter. */
|
||||||
DEF(tci_movi, 1, 0, 1, TCG_OPF_NOT_PRESENT)
|
DEF(tci_movi, 1, 0, 1, TCG_OPF_NOT_PRESENT)
|
||||||
DEF(tci_movl, 1, 0, 1, TCG_OPF_NOT_PRESENT)
|
DEF(tci_movl, 1, 0, 1, TCG_OPF_NOT_PRESENT)
|
||||||
|
DEF(tci_clz32, 1, 2, 0, TCG_OPF_NOT_PRESENT)
|
||||||
DEF(tci_divs32, 1, 2, 0, TCG_OPF_NOT_PRESENT)
|
DEF(tci_divs32, 1, 2, 0, TCG_OPF_NOT_PRESENT)
|
||||||
DEF(tci_divu32, 1, 2, 0, TCG_OPF_NOT_PRESENT)
|
DEF(tci_divu32, 1, 2, 0, TCG_OPF_NOT_PRESENT)
|
||||||
DEF(tci_rems32, 1, 2, 0, TCG_OPF_NOT_PRESENT)
|
DEF(tci_rems32, 1, 2, 0, TCG_OPF_NOT_PRESENT)
|
||||||
|
|
|
@ -83,8 +83,6 @@ tcg_target_op_def(TCGOpcode op, TCGType type, unsigned flags)
|
||||||
case INDEX_op_setcond_i64:
|
case INDEX_op_setcond_i64:
|
||||||
case INDEX_op_deposit_i32:
|
case INDEX_op_deposit_i32:
|
||||||
case INDEX_op_deposit_i64:
|
case INDEX_op_deposit_i64:
|
||||||
case INDEX_op_clz_i32:
|
|
||||||
case INDEX_op_clz_i64:
|
|
||||||
case INDEX_op_ctz_i32:
|
case INDEX_op_ctz_i32:
|
||||||
case INDEX_op_ctz_i64:
|
case INDEX_op_ctz_i64:
|
||||||
return C_O1_I2(r, r, r);
|
return C_O1_I2(r, r, r);
|
||||||
|
@ -630,6 +628,20 @@ static const TCGOutOpBinary outop_andc = {
|
||||||
.out_rrr = tgen_andc,
|
.out_rrr = tgen_andc,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static void tgen_clz(TCGContext *s, TCGType type,
|
||||||
|
TCGReg a0, TCGReg a1, TCGReg a2)
|
||||||
|
{
|
||||||
|
TCGOpcode opc = (type == TCG_TYPE_I32
|
||||||
|
? INDEX_op_tci_clz32
|
||||||
|
: INDEX_op_clz_i64);
|
||||||
|
tcg_out_op_rrr(s, opc, a0, a1, a2);
|
||||||
|
}
|
||||||
|
|
||||||
|
static const TCGOutOpBinary outop_clz = {
|
||||||
|
.base.static_constraint = C_O1_I2(r, r, r),
|
||||||
|
.out_rrr = tgen_clz,
|
||||||
|
};
|
||||||
|
|
||||||
static void tgen_divs(TCGContext *s, TCGType type,
|
static void tgen_divs(TCGContext *s, TCGType type,
|
||||||
TCGReg a0, TCGReg a1, TCGReg a2)
|
TCGReg a0, TCGReg a1, TCGReg a2)
|
||||||
{
|
{
|
||||||
|
@ -921,7 +933,6 @@ static void tcg_out_op(TCGContext *s, TCGOpcode opc, TCGType type,
|
||||||
tcg_out_ldst(s, opc, args[0], args[1], args[2]);
|
tcg_out_ldst(s, opc, args[0], args[1], args[2]);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
CASE_32_64(clz) /* Optional (TCG_TARGET_HAS_clz_*). */
|
|
||||||
CASE_32_64(ctz) /* Optional (TCG_TARGET_HAS_ctz_*). */
|
CASE_32_64(ctz) /* Optional (TCG_TARGET_HAS_ctz_*). */
|
||||||
tcg_out_op_rrr(s, opc, args[0], args[1], args[2]);
|
tcg_out_op_rrr(s, opc, args[0], args[1], args[2]);
|
||||||
break;
|
break;
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue