tcg: Link branches to the labels

This allows us to easily find all branches that use a label.
Since 'refs' is only tested vs zero, remove it and test for
an empty list instead.  Drop the use of bitfields, which had
been used to pack refs into a single 32-bit word.

Reviewed-by: Taylor Simpson <tsimpson@quicinc.com>
Signed-off-by: Richard Henderson <richard.henderson@linaro.org>
This commit is contained in:
Richard Henderson 2023-03-03 13:47:27 -08:00
parent 533206f052
commit f85b1fc4a0
4 changed files with 53 additions and 25 deletions

View file

@ -259,12 +259,7 @@ static inline void gen_set_label(TCGLabel *l)
tcg_gen_op1(INDEX_op_set_label, label_arg(l)); tcg_gen_op1(INDEX_op_set_label, label_arg(l));
} }
static inline void tcg_gen_br(TCGLabel *l) void tcg_gen_br(TCGLabel *l);
{
l->refs++;
tcg_gen_op1(INDEX_op_br, label_arg(l));
}
void tcg_gen_mb(TCGBar); void tcg_gen_mb(TCGBar);
/* Helper calls. */ /* Helper calls. */

View file

@ -238,16 +238,23 @@ struct TCGRelocation {
int type; int type;
}; };
typedef struct TCGOp TCGOp;
typedef struct TCGLabelUse TCGLabelUse;
struct TCGLabelUse {
QSIMPLEQ_ENTRY(TCGLabelUse) next;
TCGOp *op;
};
typedef struct TCGLabel TCGLabel; typedef struct TCGLabel TCGLabel;
struct TCGLabel { struct TCGLabel {
unsigned present : 1; bool present;
unsigned has_value : 1; bool has_value;
unsigned id : 14; uint16_t id;
unsigned refs : 16;
union { union {
uintptr_t value; uintptr_t value;
const tcg_insn_unit *value_ptr; const tcg_insn_unit *value_ptr;
} u; } u;
QSIMPLEQ_HEAD(, TCGLabelUse) branches;
QSIMPLEQ_HEAD(, TCGRelocation) relocs; QSIMPLEQ_HEAD(, TCGRelocation) relocs;
QSIMPLEQ_ENTRY(TCGLabel) next; QSIMPLEQ_ENTRY(TCGLabel) next;
}; };
@ -487,7 +494,7 @@ typedef struct TCGTempSet {
#define SYNC_ARG (1 << 0) #define SYNC_ARG (1 << 0)
typedef uint32_t TCGLifeData; typedef uint32_t TCGLifeData;
typedef struct TCGOp { struct TCGOp {
TCGOpcode opc : 8; TCGOpcode opc : 8;
unsigned nargs : 8; unsigned nargs : 8;
@ -506,7 +513,7 @@ typedef struct TCGOp {
/* Arguments for the opcode. */ /* Arguments for the opcode. */
TCGArg args[]; TCGArg args[];
} TCGOp; };
#define TCGOP_CALLI(X) (X)->param1 #define TCGOP_CALLI(X) (X)->param1
#define TCGOP_CALLO(X) (X)->param2 #define TCGOP_CALLO(X) (X)->param2

View file

@ -84,6 +84,22 @@ void tcg_gen_op6(TCGOpcode opc, TCGArg a1, TCGArg a2, TCGArg a3,
op->args[5] = a6; op->args[5] = a6;
} }
/* Generic ops. */
static void add_last_as_label_use(TCGLabel *l)
{
TCGLabelUse *u = tcg_malloc(sizeof(TCGLabelUse));
u->op = tcg_last_op();
QSIMPLEQ_INSERT_TAIL(&l->branches, u, next);
}
void tcg_gen_br(TCGLabel *l)
{
tcg_gen_op1(INDEX_op_br, label_arg(l));
add_last_as_label_use(l);
}
void tcg_gen_mb(TCGBar mb_type) void tcg_gen_mb(TCGBar mb_type)
{ {
if (tcg_ctx->gen_tb->cflags & CF_PARALLEL) { if (tcg_ctx->gen_tb->cflags & CF_PARALLEL) {
@ -216,8 +232,8 @@ void tcg_gen_brcond_i32(TCGCond cond, TCGv_i32 arg1, TCGv_i32 arg2, TCGLabel *l)
if (cond == TCG_COND_ALWAYS) { if (cond == TCG_COND_ALWAYS) {
tcg_gen_br(l); tcg_gen_br(l);
} else if (cond != TCG_COND_NEVER) { } else if (cond != TCG_COND_NEVER) {
l->refs++;
tcg_gen_op4ii_i32(INDEX_op_brcond_i32, arg1, arg2, cond, label_arg(l)); tcg_gen_op4ii_i32(INDEX_op_brcond_i32, arg1, arg2, cond, label_arg(l));
add_last_as_label_use(l);
} }
} }
@ -1474,7 +1490,6 @@ void tcg_gen_brcond_i64(TCGCond cond, TCGv_i64 arg1, TCGv_i64 arg2, TCGLabel *l)
if (cond == TCG_COND_ALWAYS) { if (cond == TCG_COND_ALWAYS) {
tcg_gen_br(l); tcg_gen_br(l);
} else if (cond != TCG_COND_NEVER) { } else if (cond != TCG_COND_NEVER) {
l->refs++;
if (TCG_TARGET_REG_BITS == 32) { if (TCG_TARGET_REG_BITS == 32) {
tcg_gen_op6ii_i32(INDEX_op_brcond2_i32, TCGV_LOW(arg1), tcg_gen_op6ii_i32(INDEX_op_brcond2_i32, TCGV_LOW(arg1),
TCGV_HIGH(arg1), TCGV_LOW(arg2), TCGV_HIGH(arg1), TCGV_LOW(arg2),
@ -1483,6 +1498,7 @@ void tcg_gen_brcond_i64(TCGCond cond, TCGv_i64 arg1, TCGv_i64 arg2, TCGLabel *l)
tcg_gen_op4ii_i64(INDEX_op_brcond_i64, arg1, arg2, cond, tcg_gen_op4ii_i64(INDEX_op_brcond_i64, arg1, arg2, cond,
label_arg(l)); label_arg(l));
} }
add_last_as_label_use(l);
} }
} }
@ -1493,12 +1509,12 @@ void tcg_gen_brcondi_i64(TCGCond cond, TCGv_i64 arg1, int64_t arg2, TCGLabel *l)
} else if (cond == TCG_COND_ALWAYS) { } else if (cond == TCG_COND_ALWAYS) {
tcg_gen_br(l); tcg_gen_br(l);
} else if (cond != TCG_COND_NEVER) { } else if (cond != TCG_COND_NEVER) {
l->refs++;
tcg_gen_op6ii_i32(INDEX_op_brcond2_i32, tcg_gen_op6ii_i32(INDEX_op_brcond2_i32,
TCGV_LOW(arg1), TCGV_HIGH(arg1), TCGV_LOW(arg1), TCGV_HIGH(arg1),
tcg_constant_i32(arg2), tcg_constant_i32(arg2),
tcg_constant_i32(arg2 >> 32), tcg_constant_i32(arg2 >> 32),
cond, label_arg(l)); cond, label_arg(l));
add_last_as_label_use(l);
} }
} }

View file

@ -283,6 +283,7 @@ TCGLabel *gen_new_label(void)
memset(l, 0, sizeof(TCGLabel)); memset(l, 0, sizeof(TCGLabel));
l->id = s->nb_labels++; l->id = s->nb_labels++;
QSIMPLEQ_INIT(&l->branches);
QSIMPLEQ_INIT(&l->relocs); QSIMPLEQ_INIT(&l->relocs);
QSIMPLEQ_INSERT_TAIL(&s->labels, l, next); QSIMPLEQ_INSERT_TAIL(&s->labels, l, next);
@ -2520,23 +2521,32 @@ static void process_op_defs(TCGContext *s)
} }
} }
static void remove_label_use(TCGOp *op, int idx)
{
TCGLabel *label = arg_label(op->args[idx]);
TCGLabelUse *use;
QSIMPLEQ_FOREACH(use, &label->branches, next) {
if (use->op == op) {
QSIMPLEQ_REMOVE(&label->branches, use, TCGLabelUse, next);
return;
}
}
g_assert_not_reached();
}
void tcg_op_remove(TCGContext *s, TCGOp *op) void tcg_op_remove(TCGContext *s, TCGOp *op)
{ {
TCGLabel *label;
switch (op->opc) { switch (op->opc) {
case INDEX_op_br: case INDEX_op_br:
label = arg_label(op->args[0]); remove_label_use(op, 0);
label->refs--;
break; break;
case INDEX_op_brcond_i32: case INDEX_op_brcond_i32:
case INDEX_op_brcond_i64: case INDEX_op_brcond_i64:
label = arg_label(op->args[3]); remove_label_use(op, 3);
label->refs--;
break; break;
case INDEX_op_brcond2_i32: case INDEX_op_brcond2_i32:
label = arg_label(op->args[5]); remove_label_use(op, 5);
label->refs--;
break; break;
default: default:
break; break;
@ -2648,7 +2658,7 @@ reachable_code_pass(TCGContext *s)
dead = false; dead = false;
} }
if (label->refs == 0) { if (QSIMPLEQ_EMPTY(&label->branches)) {
/* /*
* While there is an occasional backward branch, virtually * While there is an occasional backward branch, virtually
* all branches generated by the translators are forward. * all branches generated by the translators are forward.
@ -4892,7 +4902,7 @@ int tcg_gen_code(TCGContext *s, TranslationBlock *tb, target_ulong pc_start)
bool error = false; bool error = false;
QSIMPLEQ_FOREACH(l, &s->labels, next) { QSIMPLEQ_FOREACH(l, &s->labels, next) {
if (unlikely(!l->present) && l->refs) { if (unlikely(!l->present) && !QSIMPLEQ_EMPTY(&l->branches)) {
qemu_log_mask(CPU_LOG_TB_OP, qemu_log_mask(CPU_LOG_TB_OP,
"$L%d referenced but not present.\n", l->id); "$L%d referenced but not present.\n", l->id);
error = true; error = true;