tcg: Add TCGOutOp structures for add/sub carry opcodes

Reviewed-by: Pierrick Bouvier <pierrick.bouvier@linaro.org>
Signed-off-by: Richard Henderson <richard.henderson@linaro.org>
This commit is contained in:
Richard Henderson 2025-01-14 23:27:53 -08:00
parent 76f4278029
commit ee60315210
11 changed files with 393 additions and 8 deletions

View file

@ -2078,6 +2078,23 @@ static const TCGOutOpBinary outop_add = {
.out_rri = tgen_addi, .out_rri = tgen_addi,
}; };
static const TCGOutOpBinary outop_addco = {
.base.static_constraint = C_NotImplemented,
};
static const TCGOutOpAddSubCarry outop_addci = {
.base.static_constraint = C_NotImplemented,
};
static const TCGOutOpBinary outop_addcio = {
.base.static_constraint = C_NotImplemented,
};
static void tcg_out_set_carry(TCGContext *s)
{
g_assert_not_reached();
}
static void tgen_and(TCGContext *s, TCGType type, static void tgen_and(TCGContext *s, TCGType type,
TCGReg a0, TCGReg a1, TCGReg a2) TCGReg a0, TCGReg a1, TCGReg a2)
{ {
@ -2421,6 +2438,23 @@ static const TCGOutOpSubtract outop_sub = {
.out_rrr = tgen_sub, .out_rrr = tgen_sub,
}; };
static const TCGOutOpAddSubCarry outop_subbo = {
.base.static_constraint = C_NotImplemented,
};
static const TCGOutOpAddSubCarry outop_subbi = {
.base.static_constraint = C_NotImplemented,
};
static const TCGOutOpAddSubCarry outop_subbio = {
.base.static_constraint = C_NotImplemented,
};
static void tcg_out_set_borrow(TCGContext *s)
{
g_assert_not_reached();
}
static void tgen_xor(TCGContext *s, TCGType type, static void tgen_xor(TCGContext *s, TCGType type,
TCGReg a0, TCGReg a1, TCGReg a2) TCGReg a0, TCGReg a1, TCGReg a2)
{ {

View file

@ -1826,6 +1826,23 @@ static const TCGOutOpBinary outop_add = {
.out_rri = tgen_addi, .out_rri = tgen_addi,
}; };
static const TCGOutOpBinary outop_addco = {
.base.static_constraint = C_NotImplemented,
};
static const TCGOutOpAddSubCarry outop_addci = {
.base.static_constraint = C_NotImplemented,
};
static const TCGOutOpBinary outop_addcio = {
.base.static_constraint = C_NotImplemented,
};
static void tcg_out_set_carry(TCGContext *s)
{
g_assert_not_reached();
}
static void tgen_and(TCGContext *s, TCGType type, static void tgen_and(TCGContext *s, TCGType type,
TCGReg a0, TCGReg a1, TCGReg a2) TCGReg a0, TCGReg a1, TCGReg a2)
{ {
@ -2135,6 +2152,23 @@ static const TCGOutOpSubtract outop_sub = {
.out_rir = tgen_subfi, .out_rir = tgen_subfi,
}; };
static const TCGOutOpAddSubCarry outop_subbo = {
.base.static_constraint = C_NotImplemented,
};
static const TCGOutOpAddSubCarry outop_subbi = {
.base.static_constraint = C_NotImplemented,
};
static const TCGOutOpAddSubCarry outop_subbio = {
.base.static_constraint = C_NotImplemented,
};
static void tcg_out_set_borrow(TCGContext *s)
{
g_assert_not_reached();
}
static void tgen_xor(TCGContext *s, TCGType type, static void tgen_xor(TCGContext *s, TCGType type,
TCGReg a0, TCGReg a1, TCGReg a2) TCGReg a0, TCGReg a1, TCGReg a2)
{ {

View file

@ -2629,6 +2629,23 @@ static const TCGOutOpBinary outop_add = {
.out_rri = tgen_addi, .out_rri = tgen_addi,
}; };
static const TCGOutOpBinary outop_addco = {
.base.static_constraint = C_NotImplemented,
};
static const TCGOutOpAddSubCarry outop_addci = {
.base.static_constraint = C_NotImplemented,
};
static const TCGOutOpBinary outop_addcio = {
.base.static_constraint = C_NotImplemented,
};
static void tcg_out_set_carry(TCGContext *s)
{
g_assert_not_reached();
}
static void tgen_and(TCGContext *s, TCGType type, static void tgen_and(TCGContext *s, TCGType type,
TCGReg a0, TCGReg a1, TCGReg a2) TCGReg a0, TCGReg a1, TCGReg a2)
{ {
@ -3054,6 +3071,23 @@ static const TCGOutOpSubtract outop_sub = {
.out_rrr = tgen_sub, .out_rrr = tgen_sub,
}; };
static const TCGOutOpAddSubCarry outop_subbo = {
.base.static_constraint = C_NotImplemented,
};
static const TCGOutOpAddSubCarry outop_subbi = {
.base.static_constraint = C_NotImplemented,
};
static const TCGOutOpAddSubCarry outop_subbio = {
.base.static_constraint = C_NotImplemented,
};
static void tcg_out_set_borrow(TCGContext *s)
{
g_assert_not_reached();
}
static void tgen_xor(TCGContext *s, TCGType type, static void tgen_xor(TCGContext *s, TCGType type,
TCGReg a0, TCGReg a1, TCGReg a2) TCGReg a0, TCGReg a1, TCGReg a2)
{ {

View file

@ -1338,6 +1338,23 @@ static const TCGOutOpBinary outop_add = {
.out_rri = tcg_out_addi, .out_rri = tcg_out_addi,
}; };
static const TCGOutOpBinary outop_addco = {
.base.static_constraint = C_NotImplemented,
};
static const TCGOutOpAddSubCarry outop_addci = {
.base.static_constraint = C_NotImplemented,
};
static const TCGOutOpBinary outop_addcio = {
.base.static_constraint = C_NotImplemented,
};
static void tcg_out_set_carry(TCGContext *s)
{
g_assert_not_reached();
}
static void tgen_and(TCGContext *s, TCGType type, static void tgen_and(TCGContext *s, TCGType type,
TCGReg a0, TCGReg a1, TCGReg a2) TCGReg a0, TCGReg a1, TCGReg a2)
{ {
@ -1727,6 +1744,23 @@ static const TCGOutOpSubtract outop_sub = {
.out_rrr = tgen_sub, .out_rrr = tgen_sub,
}; };
static const TCGOutOpAddSubCarry outop_subbo = {
.base.static_constraint = C_NotImplemented,
};
static const TCGOutOpAddSubCarry outop_subbi = {
.base.static_constraint = C_NotImplemented,
};
static const TCGOutOpAddSubCarry outop_subbio = {
.base.static_constraint = C_NotImplemented,
};
static void tcg_out_set_borrow(TCGContext *s)
{
g_assert_not_reached();
}
static void tgen_xor(TCGContext *s, TCGType type, static void tgen_xor(TCGContext *s, TCGType type,
TCGReg a0, TCGReg a1, TCGReg a2) TCGReg a0, TCGReg a1, TCGReg a2)
{ {

View file

@ -1593,6 +1593,23 @@ static const TCGOutOpBinary outop_add = {
.out_rri = tgen_addi, .out_rri = tgen_addi,
}; };
static const TCGOutOpBinary outop_addco = {
.base.static_constraint = C_NotImplemented,
};
static const TCGOutOpAddSubCarry outop_addci = {
.base.static_constraint = C_NotImplemented,
};
static const TCGOutOpBinary outop_addcio = {
.base.static_constraint = C_NotImplemented,
};
static void tcg_out_set_carry(TCGContext *s)
{
g_assert_not_reached();
}
static void tgen_and(TCGContext *s, TCGType type, static void tgen_and(TCGContext *s, TCGType type,
TCGReg a0, TCGReg a1, TCGReg a2) TCGReg a0, TCGReg a1, TCGReg a2)
{ {
@ -2044,6 +2061,23 @@ static const TCGOutOpSubtract outop_sub = {
.out_rrr = tgen_sub, .out_rrr = tgen_sub,
}; };
static const TCGOutOpAddSubCarry outop_subbo = {
.base.static_constraint = C_NotImplemented,
};
static const TCGOutOpAddSubCarry outop_subbi = {
.base.static_constraint = C_NotImplemented,
};
static const TCGOutOpAddSubCarry outop_subbio = {
.base.static_constraint = C_NotImplemented,
};
static void tcg_out_set_borrow(TCGContext *s)
{
g_assert_not_reached();
}
static void tgen_xor(TCGContext *s, TCGType type, static void tgen_xor(TCGContext *s, TCGType type,
TCGReg a0, TCGReg a1, TCGReg a2) TCGReg a0, TCGReg a1, TCGReg a2)
{ {

View file

@ -2863,6 +2863,23 @@ static const TCGOutOpBinary outop_add = {
.out_rri = tgen_addi, .out_rri = tgen_addi,
}; };
static const TCGOutOpBinary outop_addco = {
.base.static_constraint = C_NotImplemented,
};
static const TCGOutOpAddSubCarry outop_addci = {
.base.static_constraint = C_NotImplemented,
};
static const TCGOutOpBinary outop_addcio = {
.base.static_constraint = C_NotImplemented,
};
static void tcg_out_set_carry(TCGContext *s)
{
g_assert_not_reached();
}
static void tgen_and(TCGContext *s, TCGType type, static void tgen_and(TCGContext *s, TCGType type,
TCGReg a0, TCGReg a1, TCGReg a2) TCGReg a0, TCGReg a1, TCGReg a2)
{ {
@ -3267,6 +3284,23 @@ static const TCGOutOpSubtract outop_sub = {
.out_rir = tgen_subfi, .out_rir = tgen_subfi,
}; };
static const TCGOutOpAddSubCarry outop_subbo = {
.base.static_constraint = C_NotImplemented,
};
static const TCGOutOpAddSubCarry outop_subbi = {
.base.static_constraint = C_NotImplemented,
};
static const TCGOutOpAddSubCarry outop_subbio = {
.base.static_constraint = C_NotImplemented,
};
static void tcg_out_set_borrow(TCGContext *s)
{
g_assert_not_reached();
}
static void tgen_xor(TCGContext *s, TCGType type, static void tgen_xor(TCGContext *s, TCGType type,
TCGReg a0, TCGReg a1, TCGReg a2) TCGReg a0, TCGReg a1, TCGReg a2)
{ {

View file

@ -1947,6 +1947,23 @@ static const TCGOutOpBinary outop_add = {
.out_rri = tgen_addi, .out_rri = tgen_addi,
}; };
static const TCGOutOpBinary outop_addco = {
.base.static_constraint = C_NotImplemented,
};
static const TCGOutOpAddSubCarry outop_addci = {
.base.static_constraint = C_NotImplemented,
};
static const TCGOutOpBinary outop_addcio = {
.base.static_constraint = C_NotImplemented,
};
static void tcg_out_set_carry(TCGContext *s)
{
g_assert_not_reached();
}
static void tgen_and(TCGContext *s, TCGType type, static void tgen_and(TCGContext *s, TCGType type,
TCGReg a0, TCGReg a1, TCGReg a2) TCGReg a0, TCGReg a1, TCGReg a2)
{ {
@ -2333,6 +2350,23 @@ static const TCGOutOpSubtract outop_sub = {
.out_rrr = tgen_sub, .out_rrr = tgen_sub,
}; };
static const TCGOutOpAddSubCarry outop_subbo = {
.base.static_constraint = C_NotImplemented,
};
static const TCGOutOpAddSubCarry outop_subbi = {
.base.static_constraint = C_NotImplemented,
};
static const TCGOutOpAddSubCarry outop_subbio = {
.base.static_constraint = C_NotImplemented,
};
static void tcg_out_set_borrow(TCGContext *s)
{
g_assert_not_reached();
}
static void tgen_xor(TCGContext *s, TCGType type, static void tgen_xor(TCGContext *s, TCGType type,
TCGReg a0, TCGReg a1, TCGReg a2) TCGReg a0, TCGReg a1, TCGReg a2)
{ {

View file

@ -2248,6 +2248,23 @@ static const TCGOutOpBinary outop_add = {
.out_rri = tgen_addi, .out_rri = tgen_addi,
}; };
static const TCGOutOpBinary outop_addco = {
.base.static_constraint = C_NotImplemented,
};
static const TCGOutOpAddSubCarry outop_addci = {
.base.static_constraint = C_NotImplemented,
};
static const TCGOutOpBinary outop_addcio = {
.base.static_constraint = C_NotImplemented,
};
static void tcg_out_set_carry(TCGContext *s)
{
g_assert_not_reached();
}
static void tgen_and(TCGContext *s, TCGType type, static void tgen_and(TCGContext *s, TCGType type,
TCGReg a0, TCGReg a1, TCGReg a2) TCGReg a0, TCGReg a1, TCGReg a2)
{ {
@ -2766,6 +2783,23 @@ static const TCGOutOpSubtract outop_sub = {
.out_rrr = tgen_sub, .out_rrr = tgen_sub,
}; };
static const TCGOutOpAddSubCarry outop_subbo = {
.base.static_constraint = C_NotImplemented,
};
static const TCGOutOpAddSubCarry outop_subbi = {
.base.static_constraint = C_NotImplemented,
};
static const TCGOutOpAddSubCarry outop_subbio = {
.base.static_constraint = C_NotImplemented,
};
static void tcg_out_set_borrow(TCGContext *s)
{
g_assert_not_reached();
}
static void tgen_xor(TCGContext *s, TCGType type, static void tgen_xor(TCGContext *s, TCGType type,
TCGReg a0, TCGReg a1, TCGReg a2) TCGReg a0, TCGReg a1, TCGReg a2)
{ {

View file

@ -1381,6 +1381,23 @@ static const TCGOutOpBinary outop_add = {
.out_rri = tgen_addi, .out_rri = tgen_addi,
}; };
static const TCGOutOpBinary outop_addco = {
.base.static_constraint = C_NotImplemented,
};
static const TCGOutOpAddSubCarry outop_addci = {
.base.static_constraint = C_NotImplemented,
};
static const TCGOutOpBinary outop_addcio = {
.base.static_constraint = C_NotImplemented,
};
static void tcg_out_set_carry(TCGContext *s)
{
g_assert_not_reached();
}
static void tgen_and(TCGContext *s, TCGType type, static void tgen_and(TCGContext *s, TCGType type,
TCGReg a0, TCGReg a1, TCGReg a2) TCGReg a0, TCGReg a1, TCGReg a2)
{ {
@ -1717,6 +1734,23 @@ static const TCGOutOpSubtract outop_sub = {
.out_rrr = tgen_sub, .out_rrr = tgen_sub,
}; };
static const TCGOutOpAddSubCarry outop_subbo = {
.base.static_constraint = C_NotImplemented,
};
static const TCGOutOpAddSubCarry outop_subbi = {
.base.static_constraint = C_NotImplemented,
};
static const TCGOutOpAddSubCarry outop_subbio = {
.base.static_constraint = C_NotImplemented,
};
static void tcg_out_set_borrow(TCGContext *s)
{
g_assert_not_reached();
}
static void tgen_xor(TCGContext *s, TCGType type, static void tgen_xor(TCGContext *s, TCGType type,
TCGReg a0, TCGReg a1, TCGReg a2) TCGReg a0, TCGReg a1, TCGReg a2)
{ {

View file

@ -133,6 +133,8 @@ static void tcg_out_addi_ptr(TCGContext *s, TCGReg, TCGReg, tcg_target_long);
static bool tcg_out_xchg(TCGContext *s, TCGType type, TCGReg r1, TCGReg r2); static bool tcg_out_xchg(TCGContext *s, TCGType type, TCGReg r1, TCGReg r2);
static void tcg_out_exit_tb(TCGContext *s, uintptr_t arg); static void tcg_out_exit_tb(TCGContext *s, uintptr_t arg);
static void tcg_out_goto_tb(TCGContext *s, int which); static void tcg_out_goto_tb(TCGContext *s, int which);
static void tcg_out_set_carry(TCGContext *s);
static void tcg_out_set_borrow(TCGContext *s);
static void tcg_out_op(TCGContext *s, TCGOpcode opc, TCGType type, static void tcg_out_op(TCGContext *s, TCGOpcode opc, TCGType type,
const TCGArg args[TCG_MAX_OP_ARGS], const TCGArg args[TCG_MAX_OP_ARGS],
const int const_args[TCG_MAX_OP_ARGS]); const int const_args[TCG_MAX_OP_ARGS]);
@ -978,6 +980,18 @@ typedef struct TCGOutOp {
TCGConstraintSetIndex (*dynamic_constraint)(TCGType type, unsigned flags); TCGConstraintSetIndex (*dynamic_constraint)(TCGType type, unsigned flags);
} TCGOutOp; } TCGOutOp;
typedef struct TCGOutOpAddSubCarry {
TCGOutOp base;
void (*out_rrr)(TCGContext *s, TCGType type,
TCGReg a0, TCGReg a1, TCGReg a2);
void (*out_rri)(TCGContext *s, TCGType type,
TCGReg a0, TCGReg a1, tcg_target_long a2);
void (*out_rir)(TCGContext *s, TCGType type,
TCGReg a0, tcg_target_long a1, TCGReg a2);
void (*out_rii)(TCGContext *s, TCGType type,
TCGReg a0, tcg_target_long a1, tcg_target_long a2);
} TCGOutOpAddSubCarry;
typedef struct TCGOutOpBinary { typedef struct TCGOutOpBinary {
TCGOutOp base; TCGOutOp base;
void (*out_rrr)(TCGContext *s, TCGType type, void (*out_rrr)(TCGContext *s, TCGType type,
@ -1131,6 +1145,11 @@ static const TCGOutOpUnary outop_extrl_i64_i32 = {
/* Register allocation descriptions for every TCGOpcode. */ /* Register allocation descriptions for every TCGOpcode. */
static const TCGOutOp * const all_outop[NB_OPS] = { 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_addci, TCGOutOpAddSubCarry, outop_addci),
OUTOP(INDEX_op_addcio, TCGOutOpBinary, outop_addcio),
OUTOP(INDEX_op_addco, TCGOutOpBinary, outop_addco),
/* addc1o is implemented with set_carry + addcio */
OUTOP(INDEX_op_addc1o, TCGOutOpBinary, outop_addcio),
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_brcond, TCGOutOpBrcond, outop_brcond), OUTOP(INDEX_op_brcond, TCGOutOpBrcond, outop_brcond),
@ -1170,6 +1189,11 @@ static const TCGOutOp * const all_outop[NB_OPS] = {
OUTOP(INDEX_op_shl, TCGOutOpBinary, outop_shl), OUTOP(INDEX_op_shl, TCGOutOpBinary, outop_shl),
OUTOP(INDEX_op_shr, TCGOutOpBinary, outop_shr), OUTOP(INDEX_op_shr, TCGOutOpBinary, outop_shr),
OUTOP(INDEX_op_sub, TCGOutOpSubtract, outop_sub), OUTOP(INDEX_op_sub, TCGOutOpSubtract, outop_sub),
OUTOP(INDEX_op_subbi, TCGOutOpAddSubCarry, outop_subbi),
OUTOP(INDEX_op_subbio, TCGOutOpAddSubCarry, outop_subbio),
OUTOP(INDEX_op_subbo, TCGOutOpAddSubCarry, outop_subbo),
/* subb1o is implemented with set_borrow + subbio */
OUTOP(INDEX_op_subb1o, TCGOutOpAddSubCarry, outop_subbio),
OUTOP(INDEX_op_xor, TCGOutOpBinary, outop_xor), OUTOP(INDEX_op_xor, TCGOutOpBinary, outop_xor),
#if TCG_TARGET_REG_BITS == 32 #if TCG_TARGET_REG_BITS == 32
@ -5569,7 +5593,12 @@ static void tcg_reg_alloc_op(TCGContext *s, const TCGOp *op)
/* emit instruction */ /* emit instruction */
TCGType type = TCGOP_TYPE(op); TCGType type = TCGOP_TYPE(op);
switch (op->opc) { switch (op->opc) {
case INDEX_op_addc1o:
tcg_out_set_carry(s);
/* fall through */
case INDEX_op_add: case INDEX_op_add:
case INDEX_op_addcio:
case INDEX_op_addco:
case INDEX_op_and: case INDEX_op_and:
case INDEX_op_andc: case INDEX_op_andc:
case INDEX_op_clz: case INDEX_op_clz:
@ -5608,8 +5637,7 @@ static void tcg_reg_alloc_op(TCGContext *s, const TCGOp *op)
case INDEX_op_sub: case INDEX_op_sub:
{ {
const TCGOutOpSubtract *out = const TCGOutOpSubtract *out = &outop_sub;
container_of(all_outop[op->opc], TCGOutOpSubtract, base);
/* /*
* Constants should never appear in the second source operand. * Constants should never appear in the second source operand.
@ -5624,15 +5652,32 @@ static void tcg_reg_alloc_op(TCGContext *s, const TCGOp *op)
} }
break; break;
case INDEX_op_addco: case INDEX_op_subb1o:
case INDEX_op_subbo: tcg_out_set_borrow(s);
/* fall through */
case INDEX_op_addci: case INDEX_op_addci:
case INDEX_op_subbi: case INDEX_op_subbi:
case INDEX_op_addcio:
case INDEX_op_subbio: case INDEX_op_subbio:
case INDEX_op_addc1o: case INDEX_op_subbo:
case INDEX_op_subb1o: {
g_assert_not_reached(); const TCGOutOpAddSubCarry *out =
container_of(all_outop[op->opc], TCGOutOpAddSubCarry, base);
if (const_args[2]) {
if (const_args[1]) {
out->out_rii(s, type, new_args[0],
new_args[1], new_args[2]);
} else {
out->out_rri(s, type, new_args[0],
new_args[1], new_args[2]);
}
} else if (const_args[1]) {
out->out_rir(s, type, new_args[0], new_args[1], new_args[2]);
} else {
out->out_rrr(s, type, new_args[0], new_args[1], new_args[2]);
}
}
break;
case INDEX_op_bswap64: case INDEX_op_bswap64:
case INDEX_op_ext_i32_i64: case INDEX_op_ext_i32_i64:

View file

@ -573,6 +573,23 @@ static const TCGOutOpBinary outop_add = {
.out_rrr = tgen_add, .out_rrr = tgen_add,
}; };
static const TCGOutOpBinary outop_addco = {
.base.static_constraint = C_NotImplemented,
};
static const TCGOutOpAddSubCarry outop_addci = {
.base.static_constraint = C_NotImplemented,
};
static const TCGOutOpBinary outop_addcio = {
.base.static_constraint = C_NotImplemented,
};
static void tcg_out_set_carry(TCGContext *s)
{
g_assert_not_reached();
}
static void tgen_and(TCGContext *s, TCGType type, static void tgen_and(TCGContext *s, TCGType type,
TCGReg a0, TCGReg a1, TCGReg a2) TCGReg a0, TCGReg a1, TCGReg a2)
{ {
@ -893,6 +910,23 @@ static const TCGOutOpSubtract outop_sub = {
.out_rrr = tgen_sub, .out_rrr = tgen_sub,
}; };
static const TCGOutOpAddSubCarry outop_subbo = {
.base.static_constraint = C_NotImplemented,
};
static const TCGOutOpAddSubCarry outop_subbi = {
.base.static_constraint = C_NotImplemented,
};
static const TCGOutOpAddSubCarry outop_subbio = {
.base.static_constraint = C_NotImplemented,
};
static void tcg_out_set_borrow(TCGContext *s)
{
g_assert_not_reached();
}
static void tgen_xor(TCGContext *s, TCGType type, static void tgen_xor(TCGContext *s, TCGType type,
TCGReg a0, TCGReg a1, TCGReg a2) TCGReg a0, TCGReg a1, TCGReg a2)
{ {