mirror of
https://github.com/yuzu-emu/unicorn.git
synced 2025-01-25 20:11:01 +00:00
target/arm: Convert Neon VTBL, VTBX to decodetree
Convert the Neon VTBL, VTBX instructions to decodetree. The actual implementation of the insn is copied across to the new trans function unchanged except for renaming 'tmp5' to 'tmp4'. Backports commit 54e96c744b70a5d19f14b212a579dd3be8fcaad9 from qemu
This commit is contained in:
parent
4731a69d66
commit
8de8a4500a
|
@ -419,6 +419,9 @@ Vimm_1r 1111 001 . 1 . 000 ... .... cmode:4 0 . op:1 1 .... @1reg_imm
|
||||||
##################################################################
|
##################################################################
|
||||||
VEXT 1111 001 0 1 . 11 .... .... imm:4 . q:1 . 0 .... \
|
VEXT 1111 001 0 1 . 11 .... .... imm:4 . q:1 . 0 .... \
|
||||||
vm=%vm_dp vn=%vn_dp vd=%vd_dp
|
vm=%vm_dp vn=%vn_dp vd=%vd_dp
|
||||||
|
|
||||||
|
VTBL 1111 001 1 1 . 11 .... .... 10 len:2 . op:1 . 0 .... \
|
||||||
|
vm=%vm_dp vn=%vn_dp vd=%vd_dp
|
||||||
]
|
]
|
||||||
|
|
||||||
# Subgroup for size != 0b11
|
# Subgroup for size != 0b11
|
||||||
|
|
|
@ -2923,3 +2923,60 @@ static bool trans_VEXT(DisasContext *s, arg_VEXT *a)
|
||||||
}
|
}
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static bool trans_VTBL(DisasContext *s, arg_VTBL *a)
|
||||||
|
{
|
||||||
|
TCGContext *tcg_ctx = s->uc->tcg_ctx;
|
||||||
|
int n;
|
||||||
|
TCGv_i32 tmp, tmp2, tmp3, tmp4;
|
||||||
|
TCGv_ptr ptr1;
|
||||||
|
|
||||||
|
if (!arm_dc_feature(s, ARM_FEATURE_NEON)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* UNDEF accesses to D16-D31 if they don't exist. */
|
||||||
|
if (!dc_isar_feature(aa32_simd_r32, s) &&
|
||||||
|
((a->vd | a->vn | a->vm) & 0x10)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!vfp_access_check(s)) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
n = a->len + 1;
|
||||||
|
if ((a->vn + n) > 32) {
|
||||||
|
/*
|
||||||
|
* This is UNPREDICTABLE; we choose to UNDEF to avoid the
|
||||||
|
* helper function running off the end of the register file.
|
||||||
|
*/
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
n <<= 3;
|
||||||
|
if (a->op) {
|
||||||
|
tmp = neon_load_reg(s, a->vd, 0);
|
||||||
|
} else {
|
||||||
|
tmp = tcg_temp_new_i32(tcg_ctx);
|
||||||
|
tcg_gen_movi_i32(tcg_ctx, tmp, 0);
|
||||||
|
}
|
||||||
|
tmp2 = neon_load_reg(s, a->vm, 0);
|
||||||
|
ptr1 = vfp_reg_ptr(s, true, a->vn);
|
||||||
|
tmp4 = tcg_const_i32(tcg_ctx, n);
|
||||||
|
gen_helper_neon_tbl(tcg_ctx, tmp2, tmp2, tmp, ptr1, tmp4);
|
||||||
|
tcg_temp_free_i32(tcg_ctx, tmp);
|
||||||
|
if (a->op) {
|
||||||
|
tmp = neon_load_reg(s, a->vd, 1);
|
||||||
|
} else {
|
||||||
|
tmp = tcg_temp_new_i32(tcg_ctx);
|
||||||
|
tcg_gen_movi_i32(tcg_ctx, tmp, 0);
|
||||||
|
}
|
||||||
|
tmp3 = neon_load_reg(s, a->vm, 1);
|
||||||
|
gen_helper_neon_tbl(tcg_ctx, tmp3, tmp3, tmp, ptr1, tmp4);
|
||||||
|
tcg_temp_free_i32(tcg_ctx, tmp4);
|
||||||
|
tcg_temp_free_ptr(tcg_ctx, ptr1);
|
||||||
|
neon_store_reg(s, a->vd, 0, tmp2);
|
||||||
|
neon_store_reg(s, a->vd, 1, tmp3);
|
||||||
|
tcg_temp_free_i32(tcg_ctx, tmp);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
|
@ -5138,13 +5138,12 @@ static int disas_neon_data_insn(DisasContext *s, uint32_t insn)
|
||||||
TCGContext *tcg_ctx = s->uc->tcg_ctx;
|
TCGContext *tcg_ctx = s->uc->tcg_ctx;
|
||||||
int op;
|
int op;
|
||||||
int q;
|
int q;
|
||||||
int rd, rn, rm, rd_ofs, rm_ofs;
|
int rd, rm, rd_ofs, rm_ofs;
|
||||||
int size;
|
int size;
|
||||||
int pass;
|
int pass;
|
||||||
int u;
|
int u;
|
||||||
int vec_size;
|
int vec_size;
|
||||||
TCGv_i32 tmp, tmp2, tmp3, tmp5;
|
TCGv_i32 tmp, tmp2, tmp3;
|
||||||
TCGv_ptr ptr1;
|
|
||||||
|
|
||||||
if (!arm_dc_feature(s, ARM_FEATURE_NEON)) {
|
if (!arm_dc_feature(s, ARM_FEATURE_NEON)) {
|
||||||
return 1;
|
return 1;
|
||||||
|
@ -5165,7 +5164,6 @@ static int disas_neon_data_insn(DisasContext *s, uint32_t insn)
|
||||||
q = (insn & (1 << 6)) != 0;
|
q = (insn & (1 << 6)) != 0;
|
||||||
u = (insn >> 24) & 1;
|
u = (insn >> 24) & 1;
|
||||||
VFP_DREG_D(rd, insn);
|
VFP_DREG_D(rd, insn);
|
||||||
VFP_DREG_N(rn, insn);
|
|
||||||
VFP_DREG_M(rm, insn);
|
VFP_DREG_M(rm, insn);
|
||||||
size = (insn >> 20) & 3;
|
size = (insn >> 20) & 3;
|
||||||
vec_size = q ? 16 : 8;
|
vec_size = q ? 16 : 8;
|
||||||
|
@ -5691,39 +5689,8 @@ static int disas_neon_data_insn(DisasContext *s, uint32_t insn)
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
} else if ((insn & (1 << 10)) == 0) {
|
} else if ((insn & (1 << 10)) == 0) {
|
||||||
/* VTBL, VTBX. */
|
/* VTBL, VTBX: handled by decodetree */
|
||||||
int n = ((insn >> 8) & 3) + 1;
|
return 1;
|
||||||
if ((rn + n) > 32) {
|
|
||||||
/* This is UNPREDICTABLE; we choose to UNDEF to avoid the
|
|
||||||
* helper function running off the end of the register file.
|
|
||||||
*/
|
|
||||||
return 1;
|
|
||||||
}
|
|
||||||
n <<= 3;
|
|
||||||
if (insn & (1 << 6)) {
|
|
||||||
tmp = neon_load_reg(s, rd, 0);
|
|
||||||
} else {
|
|
||||||
tmp = tcg_temp_new_i32(tcg_ctx);
|
|
||||||
tcg_gen_movi_i32(tcg_ctx, tmp, 0);
|
|
||||||
}
|
|
||||||
tmp2 = neon_load_reg(s, rm, 0);
|
|
||||||
ptr1 = vfp_reg_ptr(s, true, rn);
|
|
||||||
tmp5 = tcg_const_i32(tcg_ctx, n);
|
|
||||||
gen_helper_neon_tbl(tcg_ctx, tmp2, tmp2, tmp, ptr1, tmp5);
|
|
||||||
tcg_temp_free_i32(tcg_ctx, tmp);
|
|
||||||
if (insn & (1 << 6)) {
|
|
||||||
tmp = neon_load_reg(s, rd, 1);
|
|
||||||
} else {
|
|
||||||
tmp = tcg_temp_new_i32(tcg_ctx);
|
|
||||||
tcg_gen_movi_i32(tcg_ctx, tmp, 0);
|
|
||||||
}
|
|
||||||
tmp3 = neon_load_reg(s, rm, 1);
|
|
||||||
gen_helper_neon_tbl(tcg_ctx, tmp3, tmp3, tmp, ptr1, tmp5);
|
|
||||||
tcg_temp_free_i32(tcg_ctx, tmp5);
|
|
||||||
tcg_temp_free_ptr(tcg_ctx, ptr1);
|
|
||||||
neon_store_reg(s, rd, 0, tmp2);
|
|
||||||
neon_store_reg(s, rd, 1, tmp3);
|
|
||||||
tcg_temp_free_i32(tcg_ctx, tmp);
|
|
||||||
} else if ((insn & 0x380) == 0) {
|
} else if ((insn & 0x380) == 0) {
|
||||||
/* VDUP */
|
/* VDUP */
|
||||||
int element;
|
int element;
|
||||||
|
|
Loading…
Reference in a new issue