mirror of
https://github.com/Motorhead1991/qemu.git
synced 2025-08-04 08:13:54 -06:00
target/arm: Implement SVE2 LD1RO
Reviewed-by: Peter Maydell <peter.maydell@linaro.org> Signed-off-by: Richard Henderson <richard.henderson@linaro.org> Message-id: 20210525010358.152808-79-richard.henderson@linaro.org Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
This commit is contained in:
parent
7924d239f4
commit
12c563f683
2 changed files with 97 additions and 0 deletions
|
@ -1126,11 +1126,15 @@ LD_zpri 1010010 .. nreg:2 0.... 111 ... ..... ..... @rpri_load_msz
|
||||||
# SVE load and broadcast quadword (scalar plus scalar)
|
# SVE load and broadcast quadword (scalar plus scalar)
|
||||||
LD1RQ_zprr 1010010 .. 00 ..... 000 ... ..... ..... \
|
LD1RQ_zprr 1010010 .. 00 ..... 000 ... ..... ..... \
|
||||||
@rprr_load_msz nreg=0
|
@rprr_load_msz nreg=0
|
||||||
|
LD1RO_zprr 1010010 .. 01 ..... 000 ... ..... ..... \
|
||||||
|
@rprr_load_msz nreg=0
|
||||||
|
|
||||||
# SVE load and broadcast quadword (scalar plus immediate)
|
# SVE load and broadcast quadword (scalar plus immediate)
|
||||||
# LD1RQB, LD1RQH, LD1RQS, LD1RQD
|
# LD1RQB, LD1RQH, LD1RQS, LD1RQD
|
||||||
LD1RQ_zpri 1010010 .. 00 0.... 001 ... ..... ..... \
|
LD1RQ_zpri 1010010 .. 00 0.... 001 ... ..... ..... \
|
||||||
@rpri_load_msz nreg=0
|
@rpri_load_msz nreg=0
|
||||||
|
LD1RO_zpri 1010010 .. 01 0.... 001 ... ..... ..... \
|
||||||
|
@rpri_load_msz nreg=0
|
||||||
|
|
||||||
# SVE 32-bit gather prefetch (scalar plus 32-bit scaled offsets)
|
# SVE 32-bit gather prefetch (scalar plus 32-bit scaled offsets)
|
||||||
PRF 1000010 00 -1 ----- 0-- --- ----- 0 ----
|
PRF 1000010 00 -1 ----- 0-- --- ----- 0 ----
|
||||||
|
|
|
@ -5643,6 +5643,99 @@ static bool trans_LD1RQ_zpri(DisasContext *s, arg_rpri_load *a)
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void do_ldro(DisasContext *s, int zt, int pg, TCGv_i64 addr, int dtype)
|
||||||
|
{
|
||||||
|
unsigned vsz = vec_full_reg_size(s);
|
||||||
|
unsigned vsz_r32;
|
||||||
|
TCGv_ptr t_pg;
|
||||||
|
int poff, doff;
|
||||||
|
|
||||||
|
if (vsz < 32) {
|
||||||
|
/*
|
||||||
|
* Note that this UNDEFINED check comes after CheckSVEEnabled()
|
||||||
|
* in the ARM pseudocode, which is the sve_access_check() done
|
||||||
|
* in our caller. We should not now return false from the caller.
|
||||||
|
*/
|
||||||
|
unallocated_encoding(s);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Load the first octaword using the normal predicated load helpers. */
|
||||||
|
|
||||||
|
poff = pred_full_reg_offset(s, pg);
|
||||||
|
if (vsz > 32) {
|
||||||
|
/*
|
||||||
|
* Zero-extend the first 32 bits of the predicate into a temporary.
|
||||||
|
* This avoids triggering an assert making sure we don't have bits
|
||||||
|
* set within a predicate beyond VQ, but we have lowered VQ to 2
|
||||||
|
* for this load operation.
|
||||||
|
*/
|
||||||
|
TCGv_i64 tmp = tcg_temp_new_i64();
|
||||||
|
#ifdef HOST_WORDS_BIGENDIAN
|
||||||
|
poff += 4;
|
||||||
|
#endif
|
||||||
|
tcg_gen_ld32u_i64(tmp, cpu_env, poff);
|
||||||
|
|
||||||
|
poff = offsetof(CPUARMState, vfp.preg_tmp);
|
||||||
|
tcg_gen_st_i64(tmp, cpu_env, poff);
|
||||||
|
tcg_temp_free_i64(tmp);
|
||||||
|
}
|
||||||
|
|
||||||
|
t_pg = tcg_temp_new_ptr();
|
||||||
|
tcg_gen_addi_ptr(t_pg, cpu_env, poff);
|
||||||
|
|
||||||
|
gen_helper_gvec_mem *fn
|
||||||
|
= ldr_fns[s->mte_active[0]][s->be_data == MO_BE][dtype][0];
|
||||||
|
fn(cpu_env, t_pg, addr, tcg_constant_i32(simd_desc(32, 32, zt)));
|
||||||
|
|
||||||
|
tcg_temp_free_ptr(t_pg);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Replicate that first octaword.
|
||||||
|
* The replication happens in units of 32; if the full vector size
|
||||||
|
* is not a multiple of 32, the final bits are zeroed.
|
||||||
|
*/
|
||||||
|
doff = vec_full_reg_offset(s, zt);
|
||||||
|
vsz_r32 = QEMU_ALIGN_DOWN(vsz, 32);
|
||||||
|
if (vsz >= 64) {
|
||||||
|
tcg_gen_gvec_dup_mem(5, doff + 32, doff, vsz_r32 - 32, vsz_r32 - 32);
|
||||||
|
}
|
||||||
|
vsz -= vsz_r32;
|
||||||
|
if (vsz) {
|
||||||
|
tcg_gen_gvec_dup_imm(MO_64, doff + vsz_r32, vsz, vsz, 0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static bool trans_LD1RO_zprr(DisasContext *s, arg_rprr_load *a)
|
||||||
|
{
|
||||||
|
if (!dc_isar_feature(aa64_sve_f64mm, s)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
if (a->rm == 31) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
if (sve_access_check(s)) {
|
||||||
|
TCGv_i64 addr = new_tmp_a64(s);
|
||||||
|
tcg_gen_shli_i64(addr, cpu_reg(s, a->rm), dtype_msz(a->dtype));
|
||||||
|
tcg_gen_add_i64(addr, addr, cpu_reg_sp(s, a->rn));
|
||||||
|
do_ldro(s, a->rd, a->pg, addr, a->dtype);
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
static bool trans_LD1RO_zpri(DisasContext *s, arg_rpri_load *a)
|
||||||
|
{
|
||||||
|
if (!dc_isar_feature(aa64_sve_f64mm, s)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
if (sve_access_check(s)) {
|
||||||
|
TCGv_i64 addr = new_tmp_a64(s);
|
||||||
|
tcg_gen_addi_i64(addr, cpu_reg_sp(s, a->rn), a->imm * 32);
|
||||||
|
do_ldro(s, a->rd, a->pg, addr, a->dtype);
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
/* Load and broadcast element. */
|
/* Load and broadcast element. */
|
||||||
static bool trans_LD1R_zpri(DisasContext *s, arg_rpri_load *a)
|
static bool trans_LD1R_zpri(DisasContext *s, arg_rpri_load *a)
|
||||||
{
|
{
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue