mirror of
https://github.com/Motorhead1991/qemu.git
synced 2025-08-09 10:34:58 -06:00
tcg: Convert mul to TCGOutOpBinary
Reviewed-by: Philippe Mathieu-Daudé <philmd@linaro.org> Signed-off-by: Richard Henderson <richard.henderson@linaro.org>
This commit is contained in:
parent
5c62d3779b
commit
1e4ac0ea5d
11 changed files with 210 additions and 148 deletions
|
@ -2168,6 +2168,17 @@ static const TCGOutOpBinary outop_eqv = {
|
|||
.out_rrr = tgen_eqv,
|
||||
};
|
||||
|
||||
static void tgen_mul(TCGContext *s, TCGType type,
|
||||
TCGReg a0, TCGReg a1, TCGReg a2)
|
||||
{
|
||||
tcg_out_insn(s, 3509, MADD, type, a0, a1, a2, TCG_REG_XZR);
|
||||
}
|
||||
|
||||
static const TCGOutOpBinary outop_mul = {
|
||||
.base.static_constraint = C_O1_I2(r, r, r),
|
||||
.out_rrr = tgen_mul,
|
||||
};
|
||||
|
||||
static const TCGOutOpBinary outop_nand = {
|
||||
.base.static_constraint = C_NotImplemented,
|
||||
};
|
||||
|
@ -2322,11 +2333,6 @@ static void tcg_out_op(TCGContext *s, TCGOpcode opc, TCGType ext,
|
|||
tcg_out_ldst(s, I3312_STRX, a0, a1, a2, 3);
|
||||
break;
|
||||
|
||||
case INDEX_op_mul_i64:
|
||||
case INDEX_op_mul_i32:
|
||||
tcg_out_insn(s, 3509, MADD, ext, a0, a1, a2, TCG_REG_XZR);
|
||||
break;
|
||||
|
||||
case INDEX_op_div_i64:
|
||||
case INDEX_op_div_i32:
|
||||
tcg_out_insn(s, 3508, SDIV, ext, a0, a1, a2);
|
||||
|
@ -3029,8 +3035,6 @@ tcg_target_op_def(TCGOpcode op, TCGType type, unsigned flags)
|
|||
case INDEX_op_negsetcond_i64:
|
||||
return C_O1_I2(r, r, rC);
|
||||
|
||||
case INDEX_op_mul_i32:
|
||||
case INDEX_op_mul_i64:
|
||||
case INDEX_op_div_i32:
|
||||
case INDEX_op_div_i64:
|
||||
case INDEX_op_divu_i32:
|
||||
|
|
|
@ -921,13 +921,6 @@ static void tcg_out_dat_rIN(TCGContext *s, ARMCond cond, ARMInsn opc,
|
|||
}
|
||||
}
|
||||
|
||||
static void tcg_out_mul32(TCGContext *s, ARMCond cond, TCGReg rd,
|
||||
TCGReg rn, TCGReg rm)
|
||||
{
|
||||
/* mul */
|
||||
tcg_out32(s, (cond << 28) | 0x90 | (rd << 16) | (rm << 8) | rn);
|
||||
}
|
||||
|
||||
static void tcg_out_umull32(TCGContext *s, ARMCond cond, TCGReg rd0,
|
||||
TCGReg rd1, TCGReg rn, TCGReg rm)
|
||||
{
|
||||
|
@ -1885,6 +1878,18 @@ static const TCGOutOpBinary outop_eqv = {
|
|||
.base.static_constraint = C_NotImplemented,
|
||||
};
|
||||
|
||||
static void tgen_mul(TCGContext *s, TCGType type,
|
||||
TCGReg a0, TCGReg a1, TCGReg a2)
|
||||
{
|
||||
/* mul */
|
||||
tcg_out32(s, (COND_AL << 28) | 0x90 | (a0 << 16) | (a1 << 8) | a2);
|
||||
}
|
||||
|
||||
static const TCGOutOpBinary outop_mul = {
|
||||
.base.static_constraint = C_O1_I2(r, r, r),
|
||||
.out_rrr = tgen_mul,
|
||||
};
|
||||
|
||||
static const TCGOutOpBinary outop_nand = {
|
||||
.base.static_constraint = C_NotImplemented,
|
||||
};
|
||||
|
@ -2060,9 +2065,6 @@ static void tcg_out_op(TCGContext *s, TCGOpcode opc, TCGType type,
|
|||
}
|
||||
tcg_out_mov_reg(s, COND_AL, args[0], a0);
|
||||
break;
|
||||
case INDEX_op_mul_i32:
|
||||
tcg_out_mul32(s, COND_AL, args[0], args[1], args[2]);
|
||||
break;
|
||||
case INDEX_op_mulu2_i32:
|
||||
tcg_out_umull32(s, COND_AL, args[0], args[1], args[2], args[3]);
|
||||
break;
|
||||
|
@ -2258,7 +2260,6 @@ tcg_target_op_def(TCGOpcode op, TCGType type, unsigned flags)
|
|||
case INDEX_op_ctz_i32:
|
||||
return C_O1_I2(r, r, rIK);
|
||||
|
||||
case INDEX_op_mul_i32:
|
||||
case INDEX_op_div_i32:
|
||||
case INDEX_op_divu_i32:
|
||||
return C_O1_I2(r, r, r);
|
||||
|
|
|
@ -2637,6 +2637,33 @@ static const TCGOutOpBinary outop_eqv = {
|
|||
.base.static_constraint = C_NotImplemented,
|
||||
};
|
||||
|
||||
static void tgen_mul(TCGContext *s, TCGType type,
|
||||
TCGReg a0, TCGReg a1, TCGReg a2)
|
||||
{
|
||||
int rexw = type == TCG_TYPE_I32 ? 0 : P_REXW;
|
||||
tcg_out_modrm(s, OPC_IMUL_GvEv + rexw, a0, a2);
|
||||
}
|
||||
|
||||
static void tgen_muli(TCGContext *s, TCGType type,
|
||||
TCGReg a0, TCGReg a1, tcg_target_long a2)
|
||||
{
|
||||
int rexw = type == TCG_TYPE_I32 ? 0 : P_REXW;
|
||||
|
||||
if (a2 == (int8_t)a2) {
|
||||
tcg_out_modrm(s, OPC_IMUL_GvEvIb + rexw, a0, a0);
|
||||
tcg_out8(s, a2);
|
||||
} else {
|
||||
tcg_out_modrm(s, OPC_IMUL_GvEvIz + rexw, a0, a0);
|
||||
tcg_out32(s, a2);
|
||||
}
|
||||
}
|
||||
|
||||
static const TCGOutOpBinary outop_mul = {
|
||||
.base.static_constraint = C_O1_I2(r, 0, re),
|
||||
.out_rrr = tgen_mul,
|
||||
.out_rri = tgen_muli,
|
||||
};
|
||||
|
||||
static const TCGOutOpBinary outop_nand = {
|
||||
.base.static_constraint = C_NotImplemented,
|
||||
};
|
||||
|
@ -2804,22 +2831,6 @@ static void tcg_out_op(TCGContext *s, TCGOpcode opc, TCGType type,
|
|||
}
|
||||
break;
|
||||
|
||||
OP_32_64(mul):
|
||||
if (const_a2) {
|
||||
int32_t val;
|
||||
val = a2;
|
||||
if (val == (int8_t)val) {
|
||||
tcg_out_modrm(s, OPC_IMUL_GvEvIb + rexw, a0, a0);
|
||||
tcg_out8(s, val);
|
||||
} else {
|
||||
tcg_out_modrm(s, OPC_IMUL_GvEvIz + rexw, a0, a0);
|
||||
tcg_out32(s, val);
|
||||
}
|
||||
} else {
|
||||
tcg_out_modrm(s, OPC_IMUL_GvEv + rexw, a0, a2);
|
||||
}
|
||||
break;
|
||||
|
||||
OP_32_64(div2):
|
||||
tcg_out_modrm(s, OPC_GRP3_Ev + rexw, EXT3_IDIV, args[4]);
|
||||
break;
|
||||
|
@ -3707,10 +3718,6 @@ tcg_target_op_def(TCGOpcode op, TCGType type, unsigned flags)
|
|||
case INDEX_op_st_i64:
|
||||
return C_O0_I2(re, r);
|
||||
|
||||
case INDEX_op_mul_i32:
|
||||
case INDEX_op_mul_i64:
|
||||
return C_O1_I2(r, 0, re);
|
||||
|
||||
case INDEX_op_shl_i32:
|
||||
case INDEX_op_shl_i64:
|
||||
case INDEX_op_shr_i32:
|
||||
|
|
|
@ -1332,6 +1332,21 @@ static const TCGOutOpBinary outop_eqv = {
|
|||
.base.static_constraint = C_NotImplemented,
|
||||
};
|
||||
|
||||
static void tgen_mul(TCGContext *s, TCGType type,
|
||||
TCGReg a0, TCGReg a1, TCGReg a2)
|
||||
{
|
||||
if (type == TCG_TYPE_I32) {
|
||||
tcg_out_opc_mul_w(s, a0, a1, a2);
|
||||
} else {
|
||||
tcg_out_opc_mul_d(s, a0, a1, a2);
|
||||
}
|
||||
}
|
||||
|
||||
static const TCGOutOpBinary outop_mul = {
|
||||
.base.static_constraint = C_O1_I2(r, r, r),
|
||||
.out_rrr = tgen_mul,
|
||||
};
|
||||
|
||||
static const TCGOutOpBinary outop_nand = {
|
||||
.base.static_constraint = C_NotImplemented,
|
||||
};
|
||||
|
@ -1626,13 +1641,6 @@ static void tcg_out_op(TCGContext *s, TCGOpcode opc, TCGType type,
|
|||
}
|
||||
break;
|
||||
|
||||
case INDEX_op_mul_i32:
|
||||
tcg_out_opc_mul_w(s, a0, a1, a2);
|
||||
break;
|
||||
case INDEX_op_mul_i64:
|
||||
tcg_out_opc_mul_d(s, a0, a1, a2);
|
||||
break;
|
||||
|
||||
case INDEX_op_mulsh_i32:
|
||||
tcg_out_opc_mulh_w(s, a0, a1, a2);
|
||||
break;
|
||||
|
@ -2333,8 +2341,6 @@ tcg_target_op_def(TCGOpcode op, TCGType type, unsigned flags)
|
|||
case INDEX_op_setcond_i64:
|
||||
return C_O1_I2(r, rz, rJ);
|
||||
|
||||
case INDEX_op_mul_i32:
|
||||
case INDEX_op_mul_i64:
|
||||
case INDEX_op_mulsh_i32:
|
||||
case INDEX_op_mulsh_i64:
|
||||
case INDEX_op_muluh_i32:
|
||||
|
|
|
@ -1716,6 +1716,33 @@ static const TCGOutOpBinary outop_eqv = {
|
|||
.base.static_constraint = C_NotImplemented,
|
||||
};
|
||||
|
||||
static void tgen_mul(TCGContext *s, TCGType type,
|
||||
TCGReg a0, TCGReg a1, TCGReg a2)
|
||||
{
|
||||
MIPSInsn insn;
|
||||
|
||||
if (type == TCG_TYPE_I32) {
|
||||
if (use_mips32_instructions) {
|
||||
tcg_out_opc_reg(s, OPC_MUL, a0, a1, a2);
|
||||
return;
|
||||
}
|
||||
insn = OPC_MULT;
|
||||
} else {
|
||||
if (use_mips32r6_instructions) {
|
||||
tcg_out_opc_reg(s, OPC_DMUL, a0, a1, a2);
|
||||
return;
|
||||
}
|
||||
insn = OPC_DMULT;
|
||||
}
|
||||
tcg_out_opc_reg(s, insn, 0, a1, a2);
|
||||
tcg_out_opc_reg(s, OPC_MFLO, a0, 0, 0);
|
||||
}
|
||||
|
||||
static const TCGOutOpBinary outop_mul = {
|
||||
.base.static_constraint = C_O1_I2(r, r, r),
|
||||
.out_rrr = tgen_mul,
|
||||
};
|
||||
|
||||
static const TCGOutOpBinary outop_nand = {
|
||||
.base.static_constraint = C_NotImplemented,
|
||||
};
|
||||
|
@ -1876,13 +1903,6 @@ static void tcg_out_op(TCGContext *s, TCGOpcode opc, TCGType type,
|
|||
tcg_out_ldst(s, i1, a0, a1, a2);
|
||||
break;
|
||||
|
||||
case INDEX_op_mul_i32:
|
||||
if (use_mips32_instructions) {
|
||||
tcg_out_opc_reg(s, OPC_MUL, a0, a1, a2);
|
||||
break;
|
||||
}
|
||||
i1 = OPC_MULT, i2 = OPC_MFLO;
|
||||
goto do_hilo1;
|
||||
case INDEX_op_mulsh_i32:
|
||||
if (use_mips32r6_instructions) {
|
||||
tcg_out_opc_reg(s, OPC_MUH, a0, a1, a2);
|
||||
|
@ -1925,13 +1945,6 @@ static void tcg_out_op(TCGContext *s, TCGOpcode opc, TCGType type,
|
|||
}
|
||||
i1 = OPC_DIVU, i2 = OPC_MFHI;
|
||||
goto do_hilo1;
|
||||
case INDEX_op_mul_i64:
|
||||
if (use_mips32r6_instructions) {
|
||||
tcg_out_opc_reg(s, OPC_DMUL, a0, a1, a2);
|
||||
break;
|
||||
}
|
||||
i1 = OPC_DMULT, i2 = OPC_MFLO;
|
||||
goto do_hilo1;
|
||||
case INDEX_op_mulsh_i64:
|
||||
if (use_mips32r6_instructions) {
|
||||
tcg_out_opc_reg(s, OPC_DMUH, a0, a1, a2);
|
||||
|
@ -2232,7 +2245,6 @@ tcg_target_op_def(TCGOpcode op, TCGType type, unsigned flags)
|
|||
case INDEX_op_st_i64:
|
||||
return C_O0_I2(rz, r);
|
||||
|
||||
case INDEX_op_mul_i32:
|
||||
case INDEX_op_mulsh_i32:
|
||||
case INDEX_op_muluh_i32:
|
||||
case INDEX_op_div_i32:
|
||||
|
@ -2240,7 +2252,6 @@ tcg_target_op_def(TCGOpcode op, TCGType type, unsigned flags)
|
|||
case INDEX_op_rem_i32:
|
||||
case INDEX_op_remu_i32:
|
||||
case INDEX_op_setcond_i32:
|
||||
case INDEX_op_mul_i64:
|
||||
case INDEX_op_mulsh_i64:
|
||||
case INDEX_op_muluh_i64:
|
||||
case INDEX_op_div_i64:
|
||||
|
|
|
@ -2965,6 +2965,25 @@ static const TCGOutOpBinary outop_eqv = {
|
|||
.out_rrr = tgen_eqv,
|
||||
};
|
||||
|
||||
static void tgen_mul(TCGContext *s, TCGType type,
|
||||
TCGReg a0, TCGReg a1, TCGReg a2)
|
||||
{
|
||||
uint32_t insn = type == TCG_TYPE_I32 ? MULLW : MULLD;
|
||||
tcg_out32(s, insn | TAB(a0, a1, a2));
|
||||
}
|
||||
|
||||
static void tgen_muli(TCGContext *s, TCGType type,
|
||||
TCGReg a0, TCGReg a1, tcg_target_long a2)
|
||||
{
|
||||
tcg_out32(s, MULLI | TAI(a0, a1, a2));
|
||||
}
|
||||
|
||||
static const TCGOutOpBinary outop_mul = {
|
||||
.base.static_constraint = C_O1_I2(r, r, rI),
|
||||
.out_rrr = tgen_mul,
|
||||
.out_rri = tgen_muli,
|
||||
};
|
||||
|
||||
static void tgen_nand(TCGContext *s, TCGType type,
|
||||
TCGReg a0, TCGReg a1, TCGReg a2)
|
||||
{
|
||||
|
@ -3077,7 +3096,7 @@ static void tcg_out_op(TCGContext *s, TCGOpcode opc, TCGType type,
|
|||
const TCGArg args[TCG_MAX_OP_ARGS],
|
||||
const int const_args[TCG_MAX_OP_ARGS])
|
||||
{
|
||||
TCGArg a0, a1, a2;
|
||||
TCGArg a0, a1;
|
||||
|
||||
switch (opc) {
|
||||
case INDEX_op_goto_ptr:
|
||||
|
@ -3166,15 +3185,6 @@ static void tcg_out_op(TCGContext *s, TCGOpcode opc, TCGType type,
|
|||
tcg_out32(s, CNTPOPD | SAB(args[1], args[0], 0));
|
||||
break;
|
||||
|
||||
case INDEX_op_mul_i32:
|
||||
a0 = args[0], a1 = args[1], a2 = args[2];
|
||||
if (const_args[2]) {
|
||||
tcg_out32(s, MULLI | TAI(a0, a1, a2));
|
||||
} else {
|
||||
tcg_out32(s, MULLW | TAB(a0, a1, a2));
|
||||
}
|
||||
break;
|
||||
|
||||
case INDEX_op_div_i32:
|
||||
tcg_out32(s, DIVW | TAB(args[0], args[1], args[2]));
|
||||
break;
|
||||
|
@ -3283,14 +3293,6 @@ static void tcg_out_op(TCGContext *s, TCGOpcode opc, TCGType type,
|
|||
}
|
||||
break;
|
||||
|
||||
case INDEX_op_mul_i64:
|
||||
a0 = args[0], a1 = args[1], a2 = args[2];
|
||||
if (const_args[2]) {
|
||||
tcg_out32(s, MULLI | TAI(a0, a1, a2));
|
||||
} else {
|
||||
tcg_out32(s, MULLD | TAB(a0, a1, a2));
|
||||
}
|
||||
break;
|
||||
case INDEX_op_div_i64:
|
||||
tcg_out32(s, DIVD | TAB(args[0], args[1], args[2]));
|
||||
break;
|
||||
|
@ -4171,10 +4173,6 @@ tcg_target_op_def(TCGOpcode op, TCGType type, unsigned flags)
|
|||
case INDEX_op_rotr_i64:
|
||||
return C_O1_I2(r, r, ri);
|
||||
|
||||
case INDEX_op_mul_i32:
|
||||
case INDEX_op_mul_i64:
|
||||
return C_O1_I2(r, r, rI);
|
||||
|
||||
case INDEX_op_div_i32:
|
||||
case INDEX_op_divu_i32:
|
||||
case INDEX_op_rem_i32:
|
||||
|
|
|
@ -2009,6 +2009,18 @@ static const TCGOutOpBinary outop_eqv = {
|
|||
.out_rrr = tgen_eqv,
|
||||
};
|
||||
|
||||
static void tgen_mul(TCGContext *s, TCGType type,
|
||||
TCGReg a0, TCGReg a1, TCGReg a2)
|
||||
{
|
||||
RISCVInsn insn = type == TCG_TYPE_I32 ? OPC_MULW : OPC_MUL;
|
||||
tcg_out_opc_reg(s, insn, a0, a1, a2);
|
||||
}
|
||||
|
||||
static const TCGOutOpBinary outop_mul = {
|
||||
.base.static_constraint = C_O1_I2(r, r, r),
|
||||
.out_rrr = tgen_mul,
|
||||
};
|
||||
|
||||
static const TCGOutOpBinary outop_nand = {
|
||||
.base.static_constraint = C_NotImplemented,
|
||||
};
|
||||
|
@ -2160,13 +2172,6 @@ static void tcg_out_op(TCGContext *s, TCGOpcode opc, TCGType type,
|
|||
tcg_out_ldst(s, OPC_SD, a0, a1, a2);
|
||||
break;
|
||||
|
||||
case INDEX_op_mul_i32:
|
||||
tcg_out_opc_reg(s, OPC_MULW, a0, a1, a2);
|
||||
break;
|
||||
case INDEX_op_mul_i64:
|
||||
tcg_out_opc_reg(s, OPC_MUL, a0, a1, a2);
|
||||
break;
|
||||
|
||||
case INDEX_op_div_i32:
|
||||
tcg_out_opc_reg(s, OPC_DIVW, a0, a1, a2);
|
||||
break;
|
||||
|
@ -2706,14 +2711,12 @@ tcg_target_op_def(TCGOpcode op, TCGType type, unsigned flags)
|
|||
case INDEX_op_negsetcond_i64:
|
||||
return C_O1_I2(r, r, rI);
|
||||
|
||||
case INDEX_op_mul_i32:
|
||||
case INDEX_op_mulsh_i32:
|
||||
case INDEX_op_muluh_i32:
|
||||
case INDEX_op_div_i32:
|
||||
case INDEX_op_divu_i32:
|
||||
case INDEX_op_rem_i32:
|
||||
case INDEX_op_remu_i32:
|
||||
case INDEX_op_mul_i64:
|
||||
case INDEX_op_mulsh_i64:
|
||||
case INDEX_op_muluh_i64:
|
||||
case INDEX_op_div_i64:
|
||||
|
|
|
@ -2258,6 +2258,57 @@ static const TCGOutOpBinary outop_eqv = {
|
|||
.out_rrr = tgen_eqv,
|
||||
};
|
||||
|
||||
static void tgen_mul(TCGContext *s, TCGType type,
|
||||
TCGReg a0, TCGReg a1, TCGReg a2)
|
||||
{
|
||||
if (type == TCG_TYPE_I32) {
|
||||
if (a0 == a1) {
|
||||
tcg_out_insn(s, RRE, MSR, a0, a2);
|
||||
} else {
|
||||
tcg_out_insn(s, RRFa, MSRKC, a0, a1, a2);
|
||||
}
|
||||
} else {
|
||||
if (a0 == a1) {
|
||||
tcg_out_insn(s, RRE, MSGR, a0, a2);
|
||||
} else {
|
||||
tcg_out_insn(s, RRFa, MSGRKC, a0, a1, a2);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
static void tgen_muli(TCGContext *s, TCGType type,
|
||||
TCGReg a0, TCGReg a1, tcg_target_long a2)
|
||||
{
|
||||
tcg_out_mov(s, type, a0, a1);
|
||||
if (type == TCG_TYPE_I32) {
|
||||
if (a2 == (int16_t)a2) {
|
||||
tcg_out_insn(s, RI, MHI, a0, a2);
|
||||
} else {
|
||||
tcg_out_insn(s, RIL, MSFI, a0, a2);
|
||||
}
|
||||
} else {
|
||||
if (a2 == (int16_t)a2) {
|
||||
tcg_out_insn(s, RI, MGHI, a0, a2);
|
||||
} else {
|
||||
tcg_out_insn(s, RIL, MSGFI, a0, a2);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
static TCGConstraintSetIndex cset_mul(TCGType type, unsigned flags)
|
||||
{
|
||||
return (HAVE_FACILITY(MISC_INSN_EXT2)
|
||||
? C_O1_I2(r, r, rJ)
|
||||
: C_O1_I2(r, 0, rJ));
|
||||
}
|
||||
|
||||
static const TCGOutOpBinary outop_mul = {
|
||||
.base.static_constraint = C_Dynamic,
|
||||
.base.dynamic_constraint = cset_mul,
|
||||
.out_rrr = tgen_mul,
|
||||
.out_rri = tgen_muli,
|
||||
};
|
||||
|
||||
static void tgen_nand(TCGContext *s, TCGType type,
|
||||
TCGReg a0, TCGReg a1, TCGReg a2)
|
||||
{
|
||||
|
@ -2460,22 +2511,6 @@ static void tcg_out_op(TCGContext *s, TCGOpcode opc, TCGType type,
|
|||
tcg_out_st(s, TCG_TYPE_I32, args[0], args[1], args[2]);
|
||||
break;
|
||||
|
||||
case INDEX_op_mul_i32:
|
||||
a0 = args[0], a1 = args[1], a2 = (int32_t)args[2];
|
||||
if (const_args[2]) {
|
||||
tcg_out_mov(s, TCG_TYPE_I32, a0, a1);
|
||||
if (a2 == (int16_t)a2) {
|
||||
tcg_out_insn(s, RI, MHI, a0, a2);
|
||||
} else {
|
||||
tcg_out_insn(s, RIL, MSFI, a0, a2);
|
||||
}
|
||||
} else if (a0 == a1) {
|
||||
tcg_out_insn(s, RRE, MSR, a0, a2);
|
||||
} else {
|
||||
tcg_out_insn(s, RRFa, MSRKC, a0, a1, a2);
|
||||
}
|
||||
break;
|
||||
|
||||
case INDEX_op_div2_i32:
|
||||
tcg_debug_assert(args[0] == args[2]);
|
||||
tcg_debug_assert(args[1] == args[3]);
|
||||
|
@ -2651,22 +2686,6 @@ static void tcg_out_op(TCGContext *s, TCGOpcode opc, TCGType type,
|
|||
tcg_out_insn(s, RRE, LRVGR, args[0], args[1]);
|
||||
break;
|
||||
|
||||
case INDEX_op_mul_i64:
|
||||
a0 = args[0], a1 = args[1], a2 = args[2];
|
||||
if (const_args[2]) {
|
||||
tcg_out_mov(s, TCG_TYPE_I64, a0, a1);
|
||||
if (a2 == (int16_t)a2) {
|
||||
tcg_out_insn(s, RI, MGHI, a0, a2);
|
||||
} else {
|
||||
tcg_out_insn(s, RIL, MSGFI, a0, a2);
|
||||
}
|
||||
} else if (a0 == a1) {
|
||||
tcg_out_insn(s, RRE, MSGR, a0, a2);
|
||||
} else {
|
||||
tcg_out_insn(s, RRFa, MSGRKC, a0, a1, a2);
|
||||
}
|
||||
break;
|
||||
|
||||
case INDEX_op_div2_i64:
|
||||
/*
|
||||
* ??? We get an unnecessary sign-extension of the dividend
|
||||
|
@ -3316,15 +3335,6 @@ tcg_target_op_def(TCGOpcode op, TCGType type, unsigned flags)
|
|||
case INDEX_op_clz_i64:
|
||||
return C_O1_I2(r, r, rI);
|
||||
|
||||
case INDEX_op_mul_i32:
|
||||
return (HAVE_FACILITY(MISC_INSN_EXT2)
|
||||
? C_O1_I2(r, r, ri)
|
||||
: C_O1_I2(r, 0, ri));
|
||||
case INDEX_op_mul_i64:
|
||||
return (HAVE_FACILITY(MISC_INSN_EXT2)
|
||||
? C_O1_I2(r, r, rJ)
|
||||
: C_O1_I2(r, 0, rJ));
|
||||
|
||||
case INDEX_op_shl_i32:
|
||||
case INDEX_op_shr_i32:
|
||||
case INDEX_op_sar_i32:
|
||||
|
|
|
@ -1337,6 +1337,26 @@ static const TCGOutOpBinary outop_eqv = {
|
|||
.base.static_constraint = C_NotImplemented,
|
||||
};
|
||||
|
||||
static void tgen_mul(TCGContext *s, TCGType type,
|
||||
TCGReg a0, TCGReg a1, TCGReg a2)
|
||||
{
|
||||
uint32_t insn = type == TCG_TYPE_I32 ? ARITH_UMUL : ARITH_MULX;
|
||||
tcg_out_arith(s, a0, a1, a2, insn);
|
||||
}
|
||||
|
||||
static void tgen_muli(TCGContext *s, TCGType type,
|
||||
TCGReg a0, TCGReg a1, tcg_target_long a2)
|
||||
{
|
||||
uint32_t insn = type == TCG_TYPE_I32 ? ARITH_UMUL : ARITH_MULX;
|
||||
tcg_out_arithi(s, a0, a1, a2, insn);
|
||||
}
|
||||
|
||||
static const TCGOutOpBinary outop_mul = {
|
||||
.base.static_constraint = C_O1_I2(r, r, rJ),
|
||||
.out_rrr = tgen_mul,
|
||||
.out_rri = tgen_muli,
|
||||
};
|
||||
|
||||
static const TCGOutOpBinary outop_nand = {
|
||||
.base.static_constraint = C_NotImplemented,
|
||||
};
|
||||
|
@ -1489,9 +1509,6 @@ static void tcg_out_op(TCGContext *s, TCGOpcode opc, TCGType type,
|
|||
case INDEX_op_sar_i32:
|
||||
c = SHIFT_SRA;
|
||||
goto do_shift32;
|
||||
case INDEX_op_mul_i32:
|
||||
c = ARITH_UMUL;
|
||||
goto gen_arith;
|
||||
|
||||
case INDEX_op_div_i32:
|
||||
tcg_out_div32(s, a0, a1, a2, c2, 0);
|
||||
|
@ -1568,9 +1585,6 @@ static void tcg_out_op(TCGContext *s, TCGOpcode opc, TCGType type,
|
|||
case INDEX_op_sar_i64:
|
||||
c = SHIFT_SRAX;
|
||||
goto do_shift64;
|
||||
case INDEX_op_mul_i64:
|
||||
c = ARITH_MULX;
|
||||
goto gen_arith;
|
||||
case INDEX_op_div_i64:
|
||||
c = ARITH_SDIVX;
|
||||
goto gen_arith;
|
||||
|
@ -1667,8 +1681,6 @@ tcg_target_op_def(TCGOpcode op, TCGType type, unsigned flags)
|
|||
case INDEX_op_qemu_st_i64:
|
||||
return C_O0_I2(rz, r);
|
||||
|
||||
case INDEX_op_mul_i32:
|
||||
case INDEX_op_mul_i64:
|
||||
case INDEX_op_div_i32:
|
||||
case INDEX_op_div_i64:
|
||||
case INDEX_op_divu_i32:
|
||||
|
|
|
@ -1021,6 +1021,8 @@ static const TCGOutOp * const all_outop[NB_OPS] = {
|
|||
OUTOP(INDEX_op_and, TCGOutOpBinary, outop_and),
|
||||
OUTOP(INDEX_op_andc, TCGOutOpBinary, outop_andc),
|
||||
OUTOP(INDEX_op_eqv, TCGOutOpBinary, outop_eqv),
|
||||
OUTOP(INDEX_op_mul_i32, TCGOutOpBinary, outop_mul),
|
||||
OUTOP(INDEX_op_mul_i64, TCGOutOpBinary, outop_mul),
|
||||
OUTOP(INDEX_op_nand, TCGOutOpBinary, outop_nand),
|
||||
OUTOP(INDEX_op_neg, TCGOutOpUnary, outop_neg),
|
||||
OUTOP(INDEX_op_nor, TCGOutOpBinary, outop_nor),
|
||||
|
@ -2247,7 +2249,6 @@ bool tcg_op_supported(TCGOpcode op, TCGType type, unsigned flags)
|
|||
case INDEX_op_st8_i32:
|
||||
case INDEX_op_st16_i32:
|
||||
case INDEX_op_st_i32:
|
||||
case INDEX_op_mul_i32:
|
||||
case INDEX_op_shl_i32:
|
||||
case INDEX_op_shr_i32:
|
||||
case INDEX_op_sar_i32:
|
||||
|
@ -2313,7 +2314,6 @@ bool tcg_op_supported(TCGOpcode op, TCGType type, unsigned flags)
|
|||
case INDEX_op_st16_i64:
|
||||
case INDEX_op_st32_i64:
|
||||
case INDEX_op_st_i64:
|
||||
case INDEX_op_mul_i64:
|
||||
case INDEX_op_shl_i64:
|
||||
case INDEX_op_shr_i64:
|
||||
case INDEX_op_sar_i64:
|
||||
|
@ -5436,6 +5436,8 @@ static void tcg_reg_alloc_op(TCGContext *s, const TCGOp *op)
|
|||
case INDEX_op_and:
|
||||
case INDEX_op_andc:
|
||||
case INDEX_op_eqv:
|
||||
case INDEX_op_mul_i32:
|
||||
case INDEX_op_mul_i64:
|
||||
case INDEX_op_nand:
|
||||
case INDEX_op_nor:
|
||||
case INDEX_op_or:
|
||||
|
|
|
@ -87,8 +87,6 @@ tcg_target_op_def(TCGOpcode op, TCGType type, unsigned flags)
|
|||
case INDEX_op_rem_i64:
|
||||
case INDEX_op_remu_i32:
|
||||
case INDEX_op_remu_i64:
|
||||
case INDEX_op_mul_i32:
|
||||
case INDEX_op_mul_i64:
|
||||
case INDEX_op_shl_i32:
|
||||
case INDEX_op_shl_i64:
|
||||
case INDEX_op_shr_i32:
|
||||
|
@ -661,6 +659,17 @@ static const TCGOutOpBinary outop_eqv = {
|
|||
.out_rrr = tgen_eqv,
|
||||
};
|
||||
|
||||
static void tgen_mul(TCGContext *s, TCGType type,
|
||||
TCGReg a0, TCGReg a1, TCGReg a2)
|
||||
{
|
||||
tcg_out_op_rrr(s, glue(INDEX_op_mul_i,TCG_TARGET_REG_BITS), a0, a1, a2);
|
||||
}
|
||||
|
||||
static const TCGOutOpBinary outop_mul = {
|
||||
.base.static_constraint = C_O1_I2(r, r, r),
|
||||
.out_rrr = tgen_mul,
|
||||
};
|
||||
|
||||
static void tgen_nand(TCGContext *s, TCGType type,
|
||||
TCGReg a0, TCGReg a1, TCGReg a2)
|
||||
{
|
||||
|
@ -789,7 +798,6 @@ static void tcg_out_op(TCGContext *s, TCGOpcode opc, TCGType type,
|
|||
tcg_out_ldst(s, opc, args[0], args[1], args[2]);
|
||||
break;
|
||||
|
||||
CASE_32_64(mul)
|
||||
CASE_32_64(shl)
|
||||
CASE_32_64(shr)
|
||||
CASE_32_64(sar)
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue