mirror of
https://github.com/yuzu-emu/unicorn.git
synced 2025-01-11 09:55:30 +00:00
tcg: Introduce temp_tcgv_{i32,i64,ptr}
Backports commit 085272b35e0644fea373c33b5265c1818b7a978c from qemu
This commit is contained in:
parent
2bb5011b18
commit
960eb3f4f9
|
@ -488,7 +488,7 @@ static inline TCGTemp *tcg_global_alloc(TCGContext *s)
|
||||||
return ts;
|
return ts;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int tcg_global_reg_new_internal(TCGContext *s, TCGType type,
|
static TCGTemp *tcg_global_reg_new_internal(TCGContext *s, TCGType type,
|
||||||
TCGReg reg, const char *name)
|
TCGReg reg, const char *name)
|
||||||
{
|
{
|
||||||
TCGTemp *ts;
|
TCGTemp *ts;
|
||||||
|
@ -505,43 +505,41 @@ static int tcg_global_reg_new_internal(TCGContext *s, TCGType type,
|
||||||
ts->name = name;
|
ts->name = name;
|
||||||
tcg_regset_set_reg(s->reserved_regs, reg);
|
tcg_regset_set_reg(s->reserved_regs, reg);
|
||||||
|
|
||||||
return temp_idx(s, ts);
|
return ts;
|
||||||
}
|
}
|
||||||
|
|
||||||
void tcg_set_frame(TCGContext *s, TCGReg reg, intptr_t start, intptr_t size)
|
void tcg_set_frame(TCGContext *s, TCGReg reg, intptr_t start, intptr_t size)
|
||||||
{
|
{
|
||||||
int idx;
|
|
||||||
s->frame_start = start;
|
s->frame_start = start;
|
||||||
s->frame_end = start + size;
|
s->frame_end = start + size;
|
||||||
idx = tcg_global_reg_new_internal(s, TCG_TYPE_PTR, reg, "_frame");
|
s->frame_temp
|
||||||
s->frame_temp = &s->temps[idx];
|
= tcg_global_reg_new_internal(s, TCG_TYPE_PTR, reg, "_frame");
|
||||||
}
|
}
|
||||||
|
|
||||||
TCGv_i32 tcg_global_reg_new_i32(TCGContext *s, TCGReg reg, const char *name)
|
TCGv_i32 tcg_global_reg_new_i32(TCGContext *s, TCGReg reg, const char *name)
|
||||||
{
|
{
|
||||||
int idx;
|
TCGTemp *t;
|
||||||
|
|
||||||
if (tcg_regset_test_reg(s->reserved_regs, reg)) {
|
if (tcg_regset_test_reg(s->reserved_regs, reg)) {
|
||||||
tcg_abort();
|
tcg_abort();
|
||||||
}
|
}
|
||||||
|
t = tcg_global_reg_new_internal(s, TCG_TYPE_I32, reg, name);
|
||||||
idx = tcg_global_reg_new_internal(s, TCG_TYPE_I32, reg, name);
|
return temp_tcgv_i32(s, t);
|
||||||
return MAKE_TCGV_I32(idx);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
TCGv_i64 tcg_global_reg_new_i64(TCGContext *s, TCGReg reg, const char *name)
|
TCGv_i64 tcg_global_reg_new_i64(TCGContext *s, TCGReg reg, const char *name)
|
||||||
{
|
{
|
||||||
int idx;
|
TCGTemp *t;
|
||||||
|
|
||||||
if (tcg_regset_test_reg(s->reserved_regs, reg)) {
|
if (tcg_regset_test_reg(s->reserved_regs, reg)) {
|
||||||
tcg_abort();
|
tcg_abort();
|
||||||
}
|
}
|
||||||
|
|
||||||
idx = tcg_global_reg_new_internal(s, TCG_TYPE_I64, reg, name);
|
t = tcg_global_reg_new_internal(s, TCG_TYPE_I64, reg, name);
|
||||||
return MAKE_TCGV_I64(idx);
|
return temp_tcgv_i64(s, t);
|
||||||
}
|
}
|
||||||
|
|
||||||
int tcg_global_mem_new_internal(TCGContext *s, TCGType type, TCGv_ptr base,
|
TCGTemp *tcg_global_mem_new_internal(TCGContext *s, TCGType type, TCGv_ptr base,
|
||||||
intptr_t offset, const char *name)
|
intptr_t offset, const char *name)
|
||||||
{
|
{
|
||||||
TCGTemp *base_ts = &s->temps[GET_TCGV_PTR(base)];
|
TCGTemp *base_ts = &s->temps[GET_TCGV_PTR(base)];
|
||||||
|
@ -593,10 +591,10 @@ int tcg_global_mem_new_internal(TCGContext *s, TCGType type, TCGv_ptr base,
|
||||||
ts->mem_offset = offset;
|
ts->mem_offset = offset;
|
||||||
ts->name = name;
|
ts->name = name;
|
||||||
}
|
}
|
||||||
return temp_idx(s, ts);
|
return ts;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int tcg_temp_new_internal(TCGContext *s, TCGType type, int temp_local)
|
static TCGTemp *tcg_temp_new_internal(TCGContext *s, TCGType type, int temp_local)
|
||||||
{
|
{
|
||||||
TCGTemp *ts;
|
TCGTemp *ts;
|
||||||
int idx, k;
|
int idx, k;
|
||||||
|
@ -632,35 +630,29 @@ static int tcg_temp_new_internal(TCGContext *s, TCGType type, int temp_local)
|
||||||
ts->temp_allocated = 1;
|
ts->temp_allocated = 1;
|
||||||
ts->temp_local = temp_local;
|
ts->temp_local = temp_local;
|
||||||
}
|
}
|
||||||
idx = temp_idx(s, ts);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#if defined(CONFIG_DEBUG_TCG)
|
#if defined(CONFIG_DEBUG_TCG)
|
||||||
s->temps_in_use++;
|
s->temps_in_use++;
|
||||||
#endif
|
#endif
|
||||||
return idx;
|
return ts;
|
||||||
}
|
}
|
||||||
|
|
||||||
TCGv_i32 tcg_temp_new_internal_i32(TCGContext *s, int temp_local)
|
TCGv_i32 tcg_temp_new_internal_i32(TCGContext *s, int temp_local)
|
||||||
{
|
{
|
||||||
int idx;
|
TCGTemp *t = tcg_temp_new_internal(s, TCG_TYPE_I32, temp_local);
|
||||||
|
return temp_tcgv_i32(s, t);
|
||||||
idx = tcg_temp_new_internal(s, TCG_TYPE_I32, temp_local);
|
|
||||||
return MAKE_TCGV_I32(idx);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
TCGv_i64 tcg_temp_new_internal_i64(TCGContext *s, int temp_local)
|
TCGv_i64 tcg_temp_new_internal_i64(TCGContext *s, int temp_local)
|
||||||
{
|
{
|
||||||
int idx;
|
TCGTemp *t = tcg_temp_new_internal(s, TCG_TYPE_I64, temp_local);
|
||||||
|
return temp_tcgv_i64(s, t);
|
||||||
idx = tcg_temp_new_internal(s, TCG_TYPE_I64, temp_local);
|
|
||||||
return MAKE_TCGV_I64(idx);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void tcg_temp_free_internal(TCGContext *s, int idx)
|
static void tcg_temp_free_internal(TCGContext *s, TCGTemp *ts)
|
||||||
{
|
{
|
||||||
TCGTemp *ts;
|
int k, idx;
|
||||||
int k;
|
|
||||||
|
|
||||||
#if defined(CONFIG_DEBUG_TCG)
|
#if defined(CONFIG_DEBUG_TCG)
|
||||||
s->temps_in_use--;
|
s->temps_in_use--;
|
||||||
|
@ -669,23 +661,23 @@ static void tcg_temp_free_internal(TCGContext *s, int idx)
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
tcg_debug_assert(idx >= s->nb_globals && idx < s->nb_temps);
|
tcg_debug_assert(ts->temp_global == 0);
|
||||||
ts = &s->temps[idx];
|
|
||||||
tcg_debug_assert(ts->temp_allocated != 0);
|
tcg_debug_assert(ts->temp_allocated != 0);
|
||||||
ts->temp_allocated = 0;
|
ts->temp_allocated = 0;
|
||||||
|
|
||||||
|
idx = temp_idx(s, ts);
|
||||||
k = ts->base_type + (ts->temp_local ? TCG_TYPE_COUNT : 0);
|
k = ts->base_type + (ts->temp_local ? TCG_TYPE_COUNT : 0);
|
||||||
set_bit(idx, s->free_temps[k].l);
|
set_bit(idx, s->free_temps[k].l);
|
||||||
}
|
}
|
||||||
|
|
||||||
void tcg_temp_free_i32(TCGContext *s, TCGv_i32 arg)
|
void tcg_temp_free_i32(TCGContext *s, TCGv_i32 arg)
|
||||||
{
|
{
|
||||||
tcg_temp_free_internal(s, GET_TCGV_I32(arg));
|
tcg_temp_free_internal(s, tcgv_i32_temp(s, arg));
|
||||||
}
|
}
|
||||||
|
|
||||||
void tcg_temp_free_i64(TCGContext *s, TCGv_i64 arg)
|
void tcg_temp_free_i64(TCGContext *s, TCGv_i64 arg)
|
||||||
{
|
{
|
||||||
tcg_temp_free_internal(s, GET_TCGV_I64(arg));
|
tcg_temp_free_internal(s, tcgv_i64_temp(s, arg));
|
||||||
}
|
}
|
||||||
|
|
||||||
TCGv_i32 tcg_const_i32(TCGContext *s, int32_t val)
|
TCGv_i32 tcg_const_i32(TCGContext *s, int32_t val)
|
||||||
|
@ -991,7 +983,7 @@ void tcg_gen_callN(TCGContext *s, void *func, TCGTemp *ret, int nargs, TCGTemp *
|
||||||
for (i = real_args = 0; i < nargs; ++i) {
|
for (i = real_args = 0; i < nargs; ++i) {
|
||||||
int is_64bit = sizemask & (1 << (i+1)*2);
|
int is_64bit = sizemask & (1 << (i+1)*2);
|
||||||
if (is_64bit) {
|
if (is_64bit) {
|
||||||
TCGv_i64 orig = MAKE_TCGV_I64(temp_idx(s, args[i]));
|
TCGv_i64 orig = temp_tcgv_i64(s, args[i]);
|
||||||
TCGv_i32 h = tcg_temp_new_i32(s);
|
TCGv_i32 h = tcg_temp_new_i32(s);
|
||||||
TCGv_i32 l = tcg_temp_new_i32(s);
|
TCGv_i32 l = tcg_temp_new_i32(s);
|
||||||
tcg_gen_extr_i64_i32(l, h, orig);
|
tcg_gen_extr_i64_i32(l, h, orig);
|
||||||
|
@ -1011,7 +1003,7 @@ void tcg_gen_callN(TCGContext *s, void *func, TCGTemp *ret, int nargs, TCGTemp *
|
||||||
int is_signed = sizemask & (2 << (i+1)*2);
|
int is_signed = sizemask & (2 << (i+1)*2);
|
||||||
if (!is_64bit) {
|
if (!is_64bit) {
|
||||||
TCGv_i64 temp = tcg_temp_new_i64(s);
|
TCGv_i64 temp = tcg_temp_new_i64(s);
|
||||||
TCGv_i64 orig = MAKE_TCGV_I64(temp_idx(s, args[i]));
|
TCGv_i64 orig = temp_tcgv_i64(s, args[i]);
|
||||||
if (is_signed) {
|
if (is_signed) {
|
||||||
tcg_gen_ext32s_i64(s, temp, orig);
|
tcg_gen_ext32s_i64(s, temp, orig);
|
||||||
} else {
|
} else {
|
||||||
|
@ -1119,10 +1111,8 @@ void tcg_gen_callN(TCGContext *s, void *func, TCGTemp *ret, int nargs, TCGTemp *
|
||||||
for (i = real_args = 0; i < orig_nargs; ++i) {
|
for (i = real_args = 0; i < orig_nargs; ++i) {
|
||||||
int is_64bit = orig_sizemask & (1 << (i+1)*2);
|
int is_64bit = orig_sizemask & (1 << (i+1)*2);
|
||||||
if (is_64bit) {
|
if (is_64bit) {
|
||||||
TCGv_i32 h = MAKE_TCGV_I32(temp_idx(s, args[real_args++]));
|
tcg_temp_free_internal(s, args[real_args++]);
|
||||||
TCGv_i32 l = MAKE_TCGV_I32(temp_idx(s, args[real_args++]));
|
tcg_temp_free_internal(s, args[real_args++]);
|
||||||
tcg_temp_free_i32(s, h);
|
|
||||||
tcg_temp_free_i32(s, l);
|
|
||||||
} else {
|
} else {
|
||||||
real_args++;
|
real_args++;
|
||||||
}
|
}
|
||||||
|
@ -1131,7 +1121,7 @@ void tcg_gen_callN(TCGContext *s, void *func, TCGTemp *ret, int nargs, TCGTemp *
|
||||||
/* The 32-bit ABI returned two 32-bit pieces. Re-assemble them.
|
/* The 32-bit ABI returned two 32-bit pieces. Re-assemble them.
|
||||||
Note that describing these as TCGv_i64 eliminates an unnecessary
|
Note that describing these as TCGv_i64 eliminates an unnecessary
|
||||||
zero-extension that tcg_gen_concat_i32_i64 would create. */
|
zero-extension that tcg_gen_concat_i32_i64 would create. */
|
||||||
tcg_gen_concat32_i64(s, MAKE_TCGV_I64(temp_idx(s, ret)), retl, reth);
|
tcg_gen_concat32_i64(s, temp_tcgv_i64(s, ret), retl, reth);
|
||||||
tcg_temp_free_i64(s, retl);
|
tcg_temp_free_i64(s, retl);
|
||||||
tcg_temp_free_i64(s, reth);
|
tcg_temp_free_i64(s, reth);
|
||||||
}
|
}
|
||||||
|
@ -1139,8 +1129,7 @@ void tcg_gen_callN(TCGContext *s, void *func, TCGTemp *ret, int nargs, TCGTemp *
|
||||||
for (i = 0; i < nargs; ++i) {
|
for (i = 0; i < nargs; ++i) {
|
||||||
int is_64bit = sizemask & (1 << (i+1)*2);
|
int is_64bit = sizemask & (1 << (i+1)*2);
|
||||||
if (!is_64bit) {
|
if (!is_64bit) {
|
||||||
TCGv_i64 temp = MAKE_TCGV_I64(temp_idx(s, args[i]));
|
tcg_temp_free_internal(s, args[i]);
|
||||||
tcg_temp_free_i64(s, temp);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
#endif /* TCG_TARGET_EXTEND_ARGS */
|
#endif /* TCG_TARGET_EXTEND_ARGS */
|
||||||
|
|
106
qemu/tcg/tcg.h
106
qemu/tcg/tcg.h
|
@ -660,51 +660,6 @@ int tcg_gen_code(TCGContext *s, TranslationBlock *tb);
|
||||||
|
|
||||||
void tcg_set_frame(TCGContext *s, TCGReg reg, intptr_t start, intptr_t size);
|
void tcg_set_frame(TCGContext *s, TCGReg reg, intptr_t start, intptr_t size);
|
||||||
|
|
||||||
int tcg_global_mem_new_internal(TCGContext *s, TCGType type, TCGv_ptr base, intptr_t offset, const char *name);
|
|
||||||
|
|
||||||
TCGv_i32 tcg_global_reg_new_i32(TCGContext *s, TCGReg reg, const char *name);
|
|
||||||
TCGv_i64 tcg_global_reg_new_i64(TCGContext *s, TCGReg reg, const char *name);
|
|
||||||
|
|
||||||
TCGv_i32 tcg_temp_new_internal_i32(TCGContext *s, int temp_local);
|
|
||||||
TCGv_i64 tcg_temp_new_internal_i64(TCGContext *s, int temp_local);
|
|
||||||
|
|
||||||
void tcg_temp_free_i32(TCGContext *s, TCGv_i32 arg);
|
|
||||||
void tcg_temp_free_i64(TCGContext *s, TCGv_i64 arg);
|
|
||||||
|
|
||||||
static inline TCGv_i32 tcg_global_mem_new_i32(TCGContext *s, TCGv_ptr reg,
|
|
||||||
intptr_t offset, const char *name)
|
|
||||||
{
|
|
||||||
int idx = tcg_global_mem_new_internal(s, TCG_TYPE_I32, reg, offset, name);
|
|
||||||
return MAKE_TCGV_I32(idx);
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline TCGv_i32 tcg_temp_new_i32(TCGContext *s)
|
|
||||||
{
|
|
||||||
return tcg_temp_new_internal_i32(s, 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline TCGv_i32 tcg_temp_local_new_i32(TCGContext *s)
|
|
||||||
{
|
|
||||||
return tcg_temp_new_internal_i32(s, 1);
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline TCGv_i64 tcg_global_mem_new_i64(TCGContext *s, TCGv_ptr reg,
|
|
||||||
intptr_t offset, const char *name)
|
|
||||||
{
|
|
||||||
int idx = tcg_global_mem_new_internal(s, TCG_TYPE_I64, reg, offset, name);
|
|
||||||
return MAKE_TCGV_I64(idx);
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline TCGv_i64 tcg_temp_new_i64(TCGContext *s)
|
|
||||||
{
|
|
||||||
return tcg_temp_new_internal_i64(s, 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline TCGv_i64 tcg_temp_local_new_i64(TCGContext *s)
|
|
||||||
{
|
|
||||||
return tcg_temp_new_internal_i64(s, 1);
|
|
||||||
}
|
|
||||||
|
|
||||||
#if defined(CONFIG_DEBUG_TCG)
|
#if defined(CONFIG_DEBUG_TCG)
|
||||||
/* If you call tcg_clear_temp_count() at the start of a section of
|
/* If you call tcg_clear_temp_count() at the start of a section of
|
||||||
* code which is not supposed to leak any TCG temporaries, then
|
* code which is not supposed to leak any TCG temporaries, then
|
||||||
|
@ -1044,6 +999,21 @@ static inline TCGTemp *tcgv_ptr_temp(TCGContext *s, TCGv_ptr t)
|
||||||
return arg_temp(s, tcgv_ptr_arg(t));
|
return arg_temp(s, tcgv_ptr_arg(t));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static inline TCGv_i32 temp_tcgv_i32(TCGContext *s, TCGTemp *t)
|
||||||
|
{
|
||||||
|
return (TCGv_i32)temp_idx(s, t);
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline TCGv_i64 temp_tcgv_i64(TCGContext *s, TCGTemp *t)
|
||||||
|
{
|
||||||
|
return (TCGv_i64)temp_idx(s, t);
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline TCGv_ptr temp_tcgv_ptr(TCGContext *s, TCGTemp *t)
|
||||||
|
{
|
||||||
|
return (TCGv_ptr)temp_idx(s, t);
|
||||||
|
}
|
||||||
|
|
||||||
static inline void tcg_set_insn_param(TCGContext *tcg_ctx, int op_idx, int arg, TCGArg v)
|
static inline void tcg_set_insn_param(TCGContext *tcg_ctx, int op_idx, int arg, TCGArg v)
|
||||||
{
|
{
|
||||||
tcg_ctx->gen_op_buf[op_idx].args[arg] = v;
|
tcg_ctx->gen_op_buf[op_idx].args[arg] = v;
|
||||||
|
@ -1061,6 +1031,52 @@ static inline bool tcg_op_buf_full(TCGContext *tcg_ctx)
|
||||||
return tcg_op_buf_count(tcg_ctx) >= OPC_MAX_SIZE;
|
return tcg_op_buf_count(tcg_ctx) >= OPC_MAX_SIZE;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
TCGTemp *tcg_global_mem_new_internal(TCGContext *s, TCGType type, TCGv_ptr base,
|
||||||
|
intptr_t offset, const char *name);
|
||||||
|
|
||||||
|
TCGv_i32 tcg_global_reg_new_i32(TCGContext *s, TCGReg reg, const char *name);
|
||||||
|
TCGv_i64 tcg_global_reg_new_i64(TCGContext *s, TCGReg reg, const char *name);
|
||||||
|
|
||||||
|
TCGv_i32 tcg_temp_new_internal_i32(TCGContext *s, int temp_local);
|
||||||
|
TCGv_i64 tcg_temp_new_internal_i64(TCGContext *s, int temp_local);
|
||||||
|
|
||||||
|
void tcg_temp_free_i32(TCGContext *s, TCGv_i32 arg);
|
||||||
|
void tcg_temp_free_i64(TCGContext *s, TCGv_i64 arg);
|
||||||
|
|
||||||
|
static inline TCGv_i32 tcg_global_mem_new_i32(TCGContext *s, TCGv_ptr reg,
|
||||||
|
intptr_t offset, const char *name)
|
||||||
|
{
|
||||||
|
TCGTemp *t = tcg_global_mem_new_internal(s, TCG_TYPE_I32, reg, offset, name);
|
||||||
|
return temp_tcgv_i32(s, t);
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline TCGv_i32 tcg_temp_new_i32(TCGContext *s)
|
||||||
|
{
|
||||||
|
return tcg_temp_new_internal_i32(s, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline TCGv_i32 tcg_temp_local_new_i32(TCGContext *s)
|
||||||
|
{
|
||||||
|
return tcg_temp_new_internal_i32(s, 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline TCGv_i64 tcg_global_mem_new_i64(TCGContext *s, TCGv_ptr reg,
|
||||||
|
intptr_t offset, const char *name)
|
||||||
|
{
|
||||||
|
TCGTemp *t = tcg_global_mem_new_internal(s, TCG_TYPE_I64, reg, offset, name);
|
||||||
|
return temp_tcgv_i64(s, t);
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline TCGv_i64 tcg_temp_new_i64(TCGContext *s)
|
||||||
|
{
|
||||||
|
return tcg_temp_new_internal_i64(s, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline TCGv_i64 tcg_temp_local_new_i64(TCGContext *s)
|
||||||
|
{
|
||||||
|
return tcg_temp_new_internal_i64(s, 1);
|
||||||
|
}
|
||||||
|
|
||||||
// UNICORN: Added
|
// UNICORN: Added
|
||||||
#define TCG_OP_DEFS_TABLE_SIZE 136
|
#define TCG_OP_DEFS_TABLE_SIZE 136
|
||||||
extern const TCGOpDef tcg_op_defs_org[TCG_OP_DEFS_TABLE_SIZE];
|
extern const TCGOpDef tcg_op_defs_org[TCG_OP_DEFS_TABLE_SIZE];
|
||||||
|
|
Loading…
Reference in a new issue