mirror of
https://github.com/Motorhead1991/qemu.git
synced 2025-08-11 03:24:58 -06:00
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:
parent
533206f052
commit
f85b1fc4a0
4 changed files with 53 additions and 25 deletions
|
@ -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. */
|
||||||
|
|
|
@ -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
|
||||||
|
|
22
tcg/tcg-op.c
22
tcg/tcg-op.c
|
@ -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);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
30
tcg/tcg.c
30
tcg/tcg.c
|
@ -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;
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue