mirror of
https://github.com/yuzu-emu/unicorn.git
synced 2025-01-22 10:41:08 +00:00
target/arm: check CF_PARALLEL instead of parallel_cpus
Thereby decoupling the resulting translated code from the current state of the system. Backports commit 2399d4e7cec22ecf1c51062d2ebfd45220dbaace from qemu
This commit is contained in:
parent
4a858100f4
commit
1715f382b4
|
@ -3430,7 +3430,9 @@
|
||||||
#define helper_pacia helper_pacia_aarch64
|
#define helper_pacia helper_pacia_aarch64
|
||||||
#define helper_pacib helper_pacib_aarch64
|
#define helper_pacib helper_pacib_aarch64
|
||||||
#define helper_paired_cmpxchg64_be helper_paired_cmpxchg64_be_aarch64
|
#define helper_paired_cmpxchg64_be helper_paired_cmpxchg64_be_aarch64
|
||||||
|
#define helper_paired_cmpxchg64_be_parallel helper_paired_cmpxchg64_be_parallel_aarch64
|
||||||
#define helper_paired_cmpxchg64_le helper_paired_cmpxchg64_le_aarch64
|
#define helper_paired_cmpxchg64_le helper_paired_cmpxchg64_le_aarch64
|
||||||
|
#define helper_paired_cmpxchg64_le_parallel helper_paired_cmpxchg64_le_parallel_aarch64
|
||||||
#define helper_rbit64 helper_rbit64_aarch64
|
#define helper_rbit64 helper_rbit64_aarch64
|
||||||
#define helper_recpsf_f16 helper_recpsf_f16_aarch64
|
#define helper_recpsf_f16 helper_recpsf_f16_aarch64
|
||||||
#define helper_recpsf_f32 helper_recpsf_f32_aarch64
|
#define helper_recpsf_f32 helper_recpsf_f32_aarch64
|
||||||
|
|
|
@ -3430,7 +3430,9 @@
|
||||||
#define helper_pacia helper_pacia_aarch64eb
|
#define helper_pacia helper_pacia_aarch64eb
|
||||||
#define helper_pacib helper_pacib_aarch64eb
|
#define helper_pacib helper_pacib_aarch64eb
|
||||||
#define helper_paired_cmpxchg64_be helper_paired_cmpxchg64_be_aarch64eb
|
#define helper_paired_cmpxchg64_be helper_paired_cmpxchg64_be_aarch64eb
|
||||||
|
#define helper_paired_cmpxchg64_be_parallel helper_paired_cmpxchg64_be_parallel_aarch64eb
|
||||||
#define helper_paired_cmpxchg64_le helper_paired_cmpxchg64_le_aarch64eb
|
#define helper_paired_cmpxchg64_le helper_paired_cmpxchg64_le_aarch64eb
|
||||||
|
#define helper_paired_cmpxchg64_le_parallel helper_paired_cmpxchg64_le_parallel_aarch64eb
|
||||||
#define helper_rbit64 helper_rbit64_aarch64eb
|
#define helper_rbit64 helper_rbit64_aarch64eb
|
||||||
#define helper_recpsf_f16 helper_recpsf_f16_aarch64eb
|
#define helper_recpsf_f16 helper_recpsf_f16_aarch64eb
|
||||||
#define helper_recpsf_f32 helper_recpsf_f32_aarch64eb
|
#define helper_recpsf_f32 helper_recpsf_f32_aarch64eb
|
||||||
|
|
|
@ -3485,7 +3485,9 @@ aarch64_symbols = (
|
||||||
'helper_pacia',
|
'helper_pacia',
|
||||||
'helper_pacib',
|
'helper_pacib',
|
||||||
'helper_paired_cmpxchg64_be',
|
'helper_paired_cmpxchg64_be',
|
||||||
|
'helper_paired_cmpxchg64_be_parallel',
|
||||||
'helper_paired_cmpxchg64_le',
|
'helper_paired_cmpxchg64_le',
|
||||||
|
'helper_paired_cmpxchg64_le_parallel',
|
||||||
'helper_rbit64',
|
'helper_rbit64',
|
||||||
'helper_recpsf_f16',
|
'helper_recpsf_f16',
|
||||||
'helper_recpsf_f32',
|
'helper_recpsf_f32',
|
||||||
|
|
|
@ -538,16 +538,29 @@ uint64_t HELPER(crc32c_64)(uint64_t acc, uint64_t val, uint32_t bytes)
|
||||||
return crc32c(acc, buf, bytes) ^ 0xffffffff;
|
return crc32c(acc, buf, bytes) ^ 0xffffffff;
|
||||||
}
|
}
|
||||||
|
|
||||||
uint64_t HELPER(paired_cmpxchg64_le)(CPUARMState *env, uint64_t addr,
|
static uint64_t do_paired_cmpxchg64_le(CPUARMState *env, uint64_t addr,
|
||||||
uint64_t new_lo, uint64_t new_hi)
|
uint64_t new_lo, uint64_t new_hi,
|
||||||
|
bool parallel)
|
||||||
{
|
{
|
||||||
Int128 cmpv = int128_make128(env->exclusive_val, env->exclusive_high);
|
|
||||||
Int128 newv = int128_make128(new_lo, new_hi);
|
|
||||||
Int128 oldv;
|
|
||||||
uintptr_t ra = GETPC();
|
uintptr_t ra = GETPC();
|
||||||
uint64_t o0, o1;
|
Int128 oldv, cmpv, newv;
|
||||||
bool success;
|
bool success;
|
||||||
|
|
||||||
|
cmpv = int128_make128(env->exclusive_val, env->exclusive_high);
|
||||||
|
newv = int128_make128(new_lo, new_hi);
|
||||||
|
|
||||||
|
if (parallel) {
|
||||||
|
#ifndef CONFIG_ATOMIC128
|
||||||
|
cpu_loop_exit_atomic(ENV_GET_CPU(env), ra);
|
||||||
|
#else
|
||||||
|
int mem_idx = cpu_mmu_index(env, false);
|
||||||
|
TCGMemOpIdx oi = make_memop_idx(MO_LEQ | MO_ALIGN_16, mem_idx);
|
||||||
|
oldv = helper_atomic_cmpxchgo_le_mmu(env, addr, cmpv, newv, oi, ra);
|
||||||
|
success = int128_eq(oldv, cmpv);
|
||||||
|
#endif
|
||||||
|
} else {
|
||||||
|
uint64_t o0, o1;
|
||||||
|
|
||||||
#ifdef CONFIG_USER_ONLY
|
#ifdef CONFIG_USER_ONLY
|
||||||
/* ??? Enforce alignment. */
|
/* ??? Enforce alignment. */
|
||||||
uint64_t *haddr = g2h(addr);
|
uint64_t *haddr = g2h(addr);
|
||||||
|
@ -578,24 +591,46 @@ uint64_t HELPER(paired_cmpxchg64_le)(CPUARMState *env, uint64_t addr,
|
||||||
helper_le_stq_mmu(env, addr + 8, int128_gethi(newv), oi1, ra);
|
helper_le_stq_mmu(env, addr + 8, int128_gethi(newv), oi1, ra);
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
return !success;
|
return !success;
|
||||||
}
|
}
|
||||||
|
|
||||||
uint64_t HELPER(paired_cmpxchg64_be)(CPUARMState *env, uint64_t addr,
|
uint64_t HELPER(paired_cmpxchg64_le)(CPUARMState *env, uint64_t addr,
|
||||||
uint64_t new_lo, uint64_t new_hi)
|
uint64_t new_lo, uint64_t new_hi)
|
||||||
{
|
{
|
||||||
/*
|
return do_paired_cmpxchg64_le(env, addr, new_lo, new_hi, false);
|
||||||
* High and low need to be switched here because this is not actually a
|
}
|
||||||
* 128bit store but two doublewords stored consecutively
|
|
||||||
*/
|
uint64_t HELPER(paired_cmpxchg64_le_parallel)(CPUARMState *env, uint64_t addr,
|
||||||
Int128 cmpv = int128_make128(env->exclusive_high, env->exclusive_val);
|
uint64_t new_lo, uint64_t new_hi)
|
||||||
Int128 newv = int128_make128(new_hi, new_lo);
|
{
|
||||||
Int128 oldv;
|
return do_paired_cmpxchg64_le(env, addr, new_lo, new_hi, true);
|
||||||
|
}
|
||||||
|
|
||||||
|
static uint64_t do_paired_cmpxchg64_be(CPUARMState *env, uint64_t addr,
|
||||||
|
uint64_t new_lo, uint64_t new_hi,
|
||||||
|
bool parallel)
|
||||||
|
{
|
||||||
uintptr_t ra = GETPC();
|
uintptr_t ra = GETPC();
|
||||||
uint64_t o0, o1;
|
Int128 oldv, cmpv, newv;
|
||||||
bool success;
|
bool success;
|
||||||
|
|
||||||
|
cmpv = int128_make128(env->exclusive_val, env->exclusive_high);
|
||||||
|
newv = int128_make128(new_lo, new_hi);
|
||||||
|
|
||||||
|
if (parallel) {
|
||||||
|
#ifndef CONFIG_ATOMIC128
|
||||||
|
cpu_loop_exit_atomic(ENV_GET_CPU(env), ra);
|
||||||
|
#else
|
||||||
|
int mem_idx = cpu_mmu_index(env, false);
|
||||||
|
TCGMemOpIdx oi = make_memop_idx(MO_BEQ | MO_ALIGN_16, mem_idx);
|
||||||
|
oldv = helper_atomic_cmpxchgo_be_mmu(env, addr, cmpv, newv, oi, ra);
|
||||||
|
success = int128_eq(oldv, cmpv);
|
||||||
|
#endif
|
||||||
|
} else {
|
||||||
|
uint64_t o0, o1;
|
||||||
|
|
||||||
#ifdef CONFIG_USER_ONLY
|
#ifdef CONFIG_USER_ONLY
|
||||||
/* ??? Enforce alignment. */
|
/* ??? Enforce alignment. */
|
||||||
uint64_t *haddr = g2h(addr);
|
uint64_t *haddr = g2h(addr);
|
||||||
|
@ -626,10 +661,23 @@ uint64_t HELPER(paired_cmpxchg64_be)(CPUARMState *env, uint64_t addr,
|
||||||
helper_be_stq_mmu(env, addr + 8, int128_getlo(newv), oi1, ra);
|
helper_be_stq_mmu(env, addr + 8, int128_getlo(newv), oi1, ra);
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
return !success;
|
return !success;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
uint64_t HELPER(paired_cmpxchg64_be)(CPUARMState *env, uint64_t addr,
|
||||||
|
uint64_t new_lo, uint64_t new_hi)
|
||||||
|
{
|
||||||
|
return do_paired_cmpxchg64_be(env, addr, new_lo, new_hi, false);
|
||||||
|
}
|
||||||
|
|
||||||
|
uint64_t HELPER(paired_cmpxchg64_be_parallel)(CPUARMState *env, uint64_t addr,
|
||||||
|
uint64_t new_lo, uint64_t new_hi)
|
||||||
|
{
|
||||||
|
return do_paired_cmpxchg64_be(env, addr, new_lo, new_hi, true);
|
||||||
|
}
|
||||||
|
|
||||||
/* Writes back the old data into Rs. */
|
/* Writes back the old data into Rs. */
|
||||||
void HELPER(casp_le_parallel)(CPUARMState *env, uint32_t rs, uint64_t addr,
|
void HELPER(casp_le_parallel)(CPUARMState *env, uint32_t rs, uint64_t addr,
|
||||||
uint64_t new_lo, uint64_t new_hi)
|
uint64_t new_lo, uint64_t new_hi)
|
||||||
|
|
|
@ -51,7 +51,11 @@ DEF_HELPER_FLAGS_2(fcvtx_f64_to_f32, TCG_CALL_NO_RWG, f32, f64, env)
|
||||||
DEF_HELPER_FLAGS_3(crc32_64, TCG_CALL_NO_RWG_SE, i64, i64, i64, i32)
|
DEF_HELPER_FLAGS_3(crc32_64, TCG_CALL_NO_RWG_SE, i64, i64, i64, i32)
|
||||||
DEF_HELPER_FLAGS_3(crc32c_64, TCG_CALL_NO_RWG_SE, i64, i64, i64, i32)
|
DEF_HELPER_FLAGS_3(crc32c_64, TCG_CALL_NO_RWG_SE, i64, i64, i64, i32)
|
||||||
DEF_HELPER_FLAGS_4(paired_cmpxchg64_le, TCG_CALL_NO_WG, i64, env, i64, i64, i64)
|
DEF_HELPER_FLAGS_4(paired_cmpxchg64_le, TCG_CALL_NO_WG, i64, env, i64, i64, i64)
|
||||||
|
DEF_HELPER_FLAGS_4(paired_cmpxchg64_le_parallel, TCG_CALL_NO_WG,
|
||||||
|
i64, env, i64, i64, i64)
|
||||||
DEF_HELPER_FLAGS_4(paired_cmpxchg64_be, TCG_CALL_NO_WG, i64, env, i64, i64, i64)
|
DEF_HELPER_FLAGS_4(paired_cmpxchg64_be, TCG_CALL_NO_WG, i64, env, i64, i64, i64)
|
||||||
|
DEF_HELPER_FLAGS_4(paired_cmpxchg64_be_parallel, TCG_CALL_NO_WG,
|
||||||
|
i64, env, i64, i64, i64)
|
||||||
DEF_HELPER_5(casp_le_parallel, void, env, i32, i64, i64, i64)
|
DEF_HELPER_5(casp_le_parallel, void, env, i32, i64, i64, i64)
|
||||||
DEF_HELPER_5(casp_be_parallel, void, env, i32, i64, i64, i64)
|
DEF_HELPER_5(casp_be_parallel, void, env, i32, i64, i64, i64)
|
||||||
DEF_HELPER_FLAGS_3(advsimd_maxh, TCG_CALL_NO_RWG, f16, f16, f16, ptr)
|
DEF_HELPER_FLAGS_3(advsimd_maxh, TCG_CALL_NO_RWG, f16, f16, f16, ptr)
|
||||||
|
|
|
@ -485,13 +485,6 @@ void HELPER(yield)(CPUARMState *env)
|
||||||
ARMCPU *cpu = arm_env_get_cpu(env);
|
ARMCPU *cpu = arm_env_get_cpu(env);
|
||||||
CPUState *cs = CPU(cpu);
|
CPUState *cs = CPU(cpu);
|
||||||
|
|
||||||
/* When running in MTTCG we don't generate jumps to the yield and
|
|
||||||
* WFE helpers as it won't affect the scheduling of other vCPUs.
|
|
||||||
* If we wanted to more completely model WFE/SEV so we don't busy
|
|
||||||
* spin unnecessarily we would need to do something more involved.
|
|
||||||
*/
|
|
||||||
g_assert(!cs->uc->parallel_cpus);
|
|
||||||
|
|
||||||
/* This is a non-trappable hint instruction that generally indicates
|
/* This is a non-trappable hint instruction that generally indicates
|
||||||
* that the guest is currently busy-looping. Yield control back to the
|
* that the guest is currently busy-looping. Yield control back to the
|
||||||
* top level loop so that a more deserving VCPU has a chance to run.
|
* top level loop so that a more deserving VCPU has a chance to run.
|
||||||
|
|
|
@ -1556,12 +1556,12 @@ static void handle_hint(DisasContext *s, uint32_t insn,
|
||||||
s->base.is_jmp = DISAS_WFI;
|
s->base.is_jmp = DISAS_WFI;
|
||||||
break;
|
break;
|
||||||
case 0b00001: /* YIELD */
|
case 0b00001: /* YIELD */
|
||||||
if (!s->uc->parallel_cpus) {
|
if (!(tb_cflags(s->base.tb) & CF_PARALLEL)) {
|
||||||
s->base.is_jmp = DISAS_YIELD;
|
s->base.is_jmp = DISAS_YIELD;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case 0b00010: /* WFE */
|
case 0b00010: /* WFE */
|
||||||
if (!s->uc->parallel_cpus) {
|
if (!(tb_cflags(s->base.tb) & CF_PARALLEL)) {
|
||||||
s->base.is_jmp = DISAS_WFE;
|
s->base.is_jmp = DISAS_WFE;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
@ -2465,12 +2465,26 @@ static void gen_store_exclusive(DisasContext *s, int rd, int rt, int rt2,
|
||||||
MO_64 | MO_ALIGN | s->be_data);
|
MO_64 | MO_ALIGN | s->be_data);
|
||||||
tcg_gen_setcond_i64(tcg_ctx, TCG_COND_NE, tmp, tmp, tcg_ctx->cpu_exclusive_val);
|
tcg_gen_setcond_i64(tcg_ctx, TCG_COND_NE, tmp, tmp, tcg_ctx->cpu_exclusive_val);
|
||||||
} else if (s->be_data == MO_LE) {
|
} else if (s->be_data == MO_LE) {
|
||||||
|
if (tb_cflags(s->base.tb) & CF_PARALLEL) {
|
||||||
|
gen_helper_paired_cmpxchg64_le_parallel(tcg_ctx, tmp, tcg_ctx->cpu_env,
|
||||||
|
tcg_ctx->cpu_exclusive_addr,
|
||||||
|
cpu_reg(s, rt),
|
||||||
|
cpu_reg(s, rt2));
|
||||||
|
} else {
|
||||||
gen_helper_paired_cmpxchg64_le(tcg_ctx, tmp, tcg_ctx->cpu_env, tcg_ctx->cpu_exclusive_addr,
|
gen_helper_paired_cmpxchg64_le(tcg_ctx, tmp, tcg_ctx->cpu_env, tcg_ctx->cpu_exclusive_addr,
|
||||||
cpu_reg(s, rt), cpu_reg(s, rt2));
|
cpu_reg(s, rt), cpu_reg(s, rt2));
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (tb_cflags(s->base.tb) & CF_PARALLEL) {
|
||||||
|
gen_helper_paired_cmpxchg64_be_parallel(tcg_ctx, tmp, tcg_ctx->cpu_env,
|
||||||
|
tcg_ctx->cpu_exclusive_addr,
|
||||||
|
cpu_reg(s, rt),
|
||||||
|
cpu_reg(s, rt2));
|
||||||
} else {
|
} else {
|
||||||
gen_helper_paired_cmpxchg64_be(tcg_ctx, tmp, tcg_ctx->cpu_env, tcg_ctx->cpu_exclusive_addr,
|
gen_helper_paired_cmpxchg64_be(tcg_ctx, tmp, tcg_ctx->cpu_env, tcg_ctx->cpu_exclusive_addr,
|
||||||
cpu_reg(s, rt), cpu_reg(s, rt2));
|
cpu_reg(s, rt), cpu_reg(s, rt2));
|
||||||
}
|
}
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
tcg_gen_atomic_cmpxchg_i64(tcg_ctx, tmp, tcg_ctx->cpu_exclusive_addr, tcg_ctx->cpu_exclusive_val,
|
tcg_gen_atomic_cmpxchg_i64(tcg_ctx, tmp, tcg_ctx->cpu_exclusive_addr, tcg_ctx->cpu_exclusive_val,
|
||||||
cpu_reg(s, rt), get_mem_index(s),
|
cpu_reg(s, rt), get_mem_index(s),
|
||||||
|
|
|
@ -4941,7 +4941,7 @@ static void gen_nop_hint(DisasContext *s, int val)
|
||||||
* spin unnecessarily we would need to do something more involved.
|
* spin unnecessarily we would need to do something more involved.
|
||||||
*/
|
*/
|
||||||
case 1: /* yield */
|
case 1: /* yield */
|
||||||
if (!s->uc->parallel_cpus) {
|
if (!(tb_cflags(s->base.tb) & CF_PARALLEL)) {
|
||||||
gen_set_pc_im(s, s->pc);
|
gen_set_pc_im(s, s->pc);
|
||||||
s->base.is_jmp = DISAS_YIELD;
|
s->base.is_jmp = DISAS_YIELD;
|
||||||
}
|
}
|
||||||
|
@ -4951,7 +4951,7 @@ static void gen_nop_hint(DisasContext *s, int val)
|
||||||
s->base.is_jmp = DISAS_WFI;
|
s->base.is_jmp = DISAS_WFI;
|
||||||
break;
|
break;
|
||||||
case 2: /* wfe */
|
case 2: /* wfe */
|
||||||
if (!s->uc->parallel_cpus) {
|
if (!(tb_cflags(s->base.tb) & CF_PARALLEL)) {
|
||||||
gen_set_pc_im(s, s->pc);
|
gen_set_pc_im(s, s->pc);
|
||||||
s->base.is_jmp = DISAS_WFE;
|
s->base.is_jmp = DISAS_WFE;
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue