mirror of https://github.com/xemu-project/xemu.git
target/arm: Convert SRHADD, URHADD to gvec
Reviewed-by: Peter Maydell <peter.maydell@linaro.org> Signed-off-by: Richard Henderson <richard.henderson@linaro.org> Message-id: 20240528203044.612851-25-richard.henderson@linaro.org Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
This commit is contained in:
parent
fdaf45d852
commit
8989b95e71
|
@ -268,13 +268,6 @@ DEF_HELPER_FLAGS_2(fjcvtzs, TCG_CALL_NO_RWG, i64, f64, ptr)
|
|||
DEF_HELPER_FLAGS_3(check_hcr_el2_trap, TCG_CALL_NO_WG, void, env, i32, i32)
|
||||
|
||||
/* neon_helper.c */
|
||||
DEF_HELPER_2(neon_rhadd_s8, i32, i32, i32)
|
||||
DEF_HELPER_2(neon_rhadd_u8, i32, i32, i32)
|
||||
DEF_HELPER_2(neon_rhadd_s16, i32, i32, i32)
|
||||
DEF_HELPER_2(neon_rhadd_u16, i32, i32, i32)
|
||||
DEF_HELPER_2(neon_rhadd_s32, s32, s32, s32)
|
||||
DEF_HELPER_2(neon_rhadd_u32, i32, i32, i32)
|
||||
|
||||
DEF_HELPER_2(neon_pmin_u8, i32, i32, i32)
|
||||
DEF_HELPER_2(neon_pmin_s8, i32, i32, i32)
|
||||
DEF_HELPER_2(neon_pmin_u16, i32, i32, i32)
|
||||
|
|
|
@ -2149,3 +2149,147 @@ void gen_gvec_uhsub(unsigned vece, uint32_t rd_ofs, uint32_t rn_ofs,
|
|||
assert(vece <= MO_32);
|
||||
tcg_gen_gvec_3(rd_ofs, rn_ofs, rm_ofs, opr_sz, max_sz, &g[vece]);
|
||||
}
|
||||
|
||||
static void gen_srhadd8_i64(TCGv_i64 d, TCGv_i64 a, TCGv_i64 b)
|
||||
{
|
||||
TCGv_i64 t = tcg_temp_new_i64();
|
||||
|
||||
tcg_gen_or_i64(t, a, b);
|
||||
tcg_gen_vec_sar8i_i64(a, a, 1);
|
||||
tcg_gen_vec_sar8i_i64(b, b, 1);
|
||||
tcg_gen_andi_i64(t, t, dup_const(MO_8, 1));
|
||||
tcg_gen_vec_add8_i64(d, a, b);
|
||||
tcg_gen_vec_add8_i64(d, d, t);
|
||||
}
|
||||
|
||||
static void gen_srhadd16_i64(TCGv_i64 d, TCGv_i64 a, TCGv_i64 b)
|
||||
{
|
||||
TCGv_i64 t = tcg_temp_new_i64();
|
||||
|
||||
tcg_gen_or_i64(t, a, b);
|
||||
tcg_gen_vec_sar16i_i64(a, a, 1);
|
||||
tcg_gen_vec_sar16i_i64(b, b, 1);
|
||||
tcg_gen_andi_i64(t, t, dup_const(MO_16, 1));
|
||||
tcg_gen_vec_add16_i64(d, a, b);
|
||||
tcg_gen_vec_add16_i64(d, d, t);
|
||||
}
|
||||
|
||||
static void gen_srhadd_i32(TCGv_i32 d, TCGv_i32 a, TCGv_i32 b)
|
||||
{
|
||||
TCGv_i32 t = tcg_temp_new_i32();
|
||||
|
||||
tcg_gen_or_i32(t, a, b);
|
||||
tcg_gen_sari_i32(a, a, 1);
|
||||
tcg_gen_sari_i32(b, b, 1);
|
||||
tcg_gen_andi_i32(t, t, 1);
|
||||
tcg_gen_add_i32(d, a, b);
|
||||
tcg_gen_add_i32(d, d, t);
|
||||
}
|
||||
|
||||
static void gen_srhadd_vec(unsigned vece, TCGv_vec d, TCGv_vec a, TCGv_vec b)
|
||||
{
|
||||
TCGv_vec t = tcg_temp_new_vec_matching(d);
|
||||
|
||||
tcg_gen_or_vec(vece, t, a, b);
|
||||
tcg_gen_sari_vec(vece, a, a, 1);
|
||||
tcg_gen_sari_vec(vece, b, b, 1);
|
||||
tcg_gen_and_vec(vece, t, t, tcg_constant_vec_matching(d, vece, 1));
|
||||
tcg_gen_add_vec(vece, d, a, b);
|
||||
tcg_gen_add_vec(vece, d, d, t);
|
||||
}
|
||||
|
||||
void gen_gvec_srhadd(unsigned vece, uint32_t rd_ofs, uint32_t rn_ofs,
|
||||
uint32_t rm_ofs, uint32_t opr_sz, uint32_t max_sz)
|
||||
{
|
||||
static const TCGOpcode vecop_list[] = {
|
||||
INDEX_op_sari_vec, INDEX_op_add_vec, 0
|
||||
};
|
||||
static const GVecGen3 g[] = {
|
||||
{ .fni8 = gen_srhadd8_i64,
|
||||
.fniv = gen_srhadd_vec,
|
||||
.opt_opc = vecop_list,
|
||||
.vece = MO_8 },
|
||||
{ .fni8 = gen_srhadd16_i64,
|
||||
.fniv = gen_srhadd_vec,
|
||||
.opt_opc = vecop_list,
|
||||
.vece = MO_16 },
|
||||
{ .fni4 = gen_srhadd_i32,
|
||||
.fniv = gen_srhadd_vec,
|
||||
.opt_opc = vecop_list,
|
||||
.vece = MO_32 },
|
||||
};
|
||||
assert(vece <= MO_32);
|
||||
tcg_gen_gvec_3(rd_ofs, rn_ofs, rm_ofs, opr_sz, max_sz, &g[vece]);
|
||||
}
|
||||
|
||||
static void gen_urhadd8_i64(TCGv_i64 d, TCGv_i64 a, TCGv_i64 b)
|
||||
{
|
||||
TCGv_i64 t = tcg_temp_new_i64();
|
||||
|
||||
tcg_gen_or_i64(t, a, b);
|
||||
tcg_gen_vec_shr8i_i64(a, a, 1);
|
||||
tcg_gen_vec_shr8i_i64(b, b, 1);
|
||||
tcg_gen_andi_i64(t, t, dup_const(MO_8, 1));
|
||||
tcg_gen_vec_add8_i64(d, a, b);
|
||||
tcg_gen_vec_add8_i64(d, d, t);
|
||||
}
|
||||
|
||||
static void gen_urhadd16_i64(TCGv_i64 d, TCGv_i64 a, TCGv_i64 b)
|
||||
{
|
||||
TCGv_i64 t = tcg_temp_new_i64();
|
||||
|
||||
tcg_gen_or_i64(t, a, b);
|
||||
tcg_gen_vec_shr16i_i64(a, a, 1);
|
||||
tcg_gen_vec_shr16i_i64(b, b, 1);
|
||||
tcg_gen_andi_i64(t, t, dup_const(MO_16, 1));
|
||||
tcg_gen_vec_add16_i64(d, a, b);
|
||||
tcg_gen_vec_add16_i64(d, d, t);
|
||||
}
|
||||
|
||||
static void gen_urhadd_i32(TCGv_i32 d, TCGv_i32 a, TCGv_i32 b)
|
||||
{
|
||||
TCGv_i32 t = tcg_temp_new_i32();
|
||||
|
||||
tcg_gen_or_i32(t, a, b);
|
||||
tcg_gen_shri_i32(a, a, 1);
|
||||
tcg_gen_shri_i32(b, b, 1);
|
||||
tcg_gen_andi_i32(t, t, 1);
|
||||
tcg_gen_add_i32(d, a, b);
|
||||
tcg_gen_add_i32(d, d, t);
|
||||
}
|
||||
|
||||
static void gen_urhadd_vec(unsigned vece, TCGv_vec d, TCGv_vec a, TCGv_vec b)
|
||||
{
|
||||
TCGv_vec t = tcg_temp_new_vec_matching(d);
|
||||
|
||||
tcg_gen_or_vec(vece, t, a, b);
|
||||
tcg_gen_shri_vec(vece, a, a, 1);
|
||||
tcg_gen_shri_vec(vece, b, b, 1);
|
||||
tcg_gen_and_vec(vece, t, t, tcg_constant_vec_matching(d, vece, 1));
|
||||
tcg_gen_add_vec(vece, d, a, b);
|
||||
tcg_gen_add_vec(vece, d, d, t);
|
||||
}
|
||||
|
||||
void gen_gvec_urhadd(unsigned vece, uint32_t rd_ofs, uint32_t rn_ofs,
|
||||
uint32_t rm_ofs, uint32_t opr_sz, uint32_t max_sz)
|
||||
{
|
||||
static const TCGOpcode vecop_list[] = {
|
||||
INDEX_op_shri_vec, INDEX_op_add_vec, 0
|
||||
};
|
||||
static const GVecGen3 g[] = {
|
||||
{ .fni8 = gen_urhadd8_i64,
|
||||
.fniv = gen_urhadd_vec,
|
||||
.opt_opc = vecop_list,
|
||||
.vece = MO_8 },
|
||||
{ .fni8 = gen_urhadd16_i64,
|
||||
.fniv = gen_urhadd_vec,
|
||||
.opt_opc = vecop_list,
|
||||
.vece = MO_16 },
|
||||
{ .fni4 = gen_urhadd_i32,
|
||||
.fniv = gen_urhadd_vec,
|
||||
.opt_opc = vecop_list,
|
||||
.vece = MO_32 },
|
||||
};
|
||||
assert(vece <= MO_32);
|
||||
tcg_gen_gvec_3(rd_ofs, rn_ofs, rm_ofs, opr_sz, max_sz, &g[vece]);
|
||||
}
|
||||
|
|
|
@ -179,33 +179,6 @@ uint32_t HELPER(glue(neon_,name))(uint32_t arg) \
|
|||
return arg; \
|
||||
}
|
||||
|
||||
#define NEON_FN(dest, src1, src2) dest = (src1 + src2 + 1) >> 1
|
||||
NEON_VOP(rhadd_s8, neon_s8, 4)
|
||||
NEON_VOP(rhadd_u8, neon_u8, 4)
|
||||
NEON_VOP(rhadd_s16, neon_s16, 2)
|
||||
NEON_VOP(rhadd_u16, neon_u16, 2)
|
||||
#undef NEON_FN
|
||||
|
||||
int32_t HELPER(neon_rhadd_s32)(int32_t src1, int32_t src2)
|
||||
{
|
||||
int32_t dest;
|
||||
|
||||
dest = (src1 >> 1) + (src2 >> 1);
|
||||
if ((src1 | src2) & 1)
|
||||
dest++;
|
||||
return dest;
|
||||
}
|
||||
|
||||
uint32_t HELPER(neon_rhadd_u32)(uint32_t src1, uint32_t src2)
|
||||
{
|
||||
uint32_t dest;
|
||||
|
||||
dest = (src1 >> 1) + (src2 >> 1);
|
||||
if ((src1 | src2) & 1)
|
||||
dest++;
|
||||
return dest;
|
||||
}
|
||||
|
||||
#define NEON_FN(dest, src1, src2) dest = (src1 < src2) ? src1 : src2
|
||||
NEON_POP(pmin_s8, neon_s8, 4)
|
||||
NEON_POP(pmin_u8, neon_u8, 4)
|
||||
|
|
|
@ -10915,7 +10915,6 @@ static void disas_simd_3same_int(DisasContext *s, uint32_t insn)
|
|||
int rm = extract32(insn, 16, 5);
|
||||
int rn = extract32(insn, 5, 5);
|
||||
int rd = extract32(insn, 0, 5);
|
||||
int pass;
|
||||
|
||||
switch (opcode) {
|
||||
case 0x13: /* MUL, PMUL */
|
||||
|
@ -10969,6 +10968,13 @@ static void disas_simd_3same_int(DisasContext *s, uint32_t insn)
|
|||
}
|
||||
|
||||
switch (opcode) {
|
||||
case 0x02: /* SRHADD, URHADD */
|
||||
if (u) {
|
||||
gen_gvec_fn3(s, is_q, rd, rn, rm, gen_gvec_urhadd, size);
|
||||
} else {
|
||||
gen_gvec_fn3(s, is_q, rd, rn, rm, gen_gvec_srhadd, size);
|
||||
}
|
||||
return;
|
||||
case 0x0c: /* SMAX, UMAX */
|
||||
if (u) {
|
||||
gen_gvec_fn3(s, is_q, rd, rn, rm, tcg_gen_gvec_umax, size);
|
||||
|
@ -11021,45 +11027,7 @@ static void disas_simd_3same_int(DisasContext *s, uint32_t insn)
|
|||
}
|
||||
return;
|
||||
}
|
||||
|
||||
if (size == 3) {
|
||||
g_assert_not_reached();
|
||||
} else {
|
||||
for (pass = 0; pass < (is_q ? 4 : 2); pass++) {
|
||||
TCGv_i32 tcg_op1 = tcg_temp_new_i32();
|
||||
TCGv_i32 tcg_op2 = tcg_temp_new_i32();
|
||||
TCGv_i32 tcg_res = tcg_temp_new_i32();
|
||||
NeonGenTwoOpFn *genfn = NULL;
|
||||
NeonGenTwoOpEnvFn *genenvfn = NULL;
|
||||
|
||||
read_vec_element_i32(s, tcg_op1, rn, pass, MO_32);
|
||||
read_vec_element_i32(s, tcg_op2, rm, pass, MO_32);
|
||||
|
||||
switch (opcode) {
|
||||
case 0x2: /* SRHADD, URHADD */
|
||||
{
|
||||
static NeonGenTwoOpFn * const fns[3][2] = {
|
||||
{ gen_helper_neon_rhadd_s8, gen_helper_neon_rhadd_u8 },
|
||||
{ gen_helper_neon_rhadd_s16, gen_helper_neon_rhadd_u16 },
|
||||
{ gen_helper_neon_rhadd_s32, gen_helper_neon_rhadd_u32 },
|
||||
};
|
||||
genfn = fns[size][u];
|
||||
break;
|
||||
}
|
||||
default:
|
||||
g_assert_not_reached();
|
||||
}
|
||||
|
||||
if (genenvfn) {
|
||||
genenvfn(tcg_res, tcg_env, tcg_op1, tcg_op2);
|
||||
} else {
|
||||
genfn(tcg_res, tcg_op1, tcg_op2);
|
||||
}
|
||||
|
||||
write_vec_element_i32(s, tcg_res, rd, pass, MO_32);
|
||||
}
|
||||
}
|
||||
clear_vec_high(s, is_q, rd);
|
||||
g_assert_not_reached();
|
||||
}
|
||||
|
||||
/* AdvSIMD three same
|
||||
|
|
|
@ -845,6 +845,8 @@ DO_3SAME_NO_SZ_3(VHADD_S, gen_gvec_shadd)
|
|||
DO_3SAME_NO_SZ_3(VHADD_U, gen_gvec_uhadd)
|
||||
DO_3SAME_NO_SZ_3(VHSUB_S, gen_gvec_shsub)
|
||||
DO_3SAME_NO_SZ_3(VHSUB_U, gen_gvec_uhsub)
|
||||
DO_3SAME_NO_SZ_3(VRHADD_S, gen_gvec_srhadd)
|
||||
DO_3SAME_NO_SZ_3(VRHADD_U, gen_gvec_urhadd)
|
||||
|
||||
#define DO_3SAME_CMP(INSN, COND) \
|
||||
static void gen_##INSN##_3s(unsigned vece, uint32_t rd_ofs, \
|
||||
|
@ -922,27 +924,6 @@ DO_SHA2(SHA256H, gen_helper_crypto_sha256h)
|
|||
DO_SHA2(SHA256H2, gen_helper_crypto_sha256h2)
|
||||
DO_SHA2(SHA256SU1, gen_helper_crypto_sha256su1)
|
||||
|
||||
#define DO_3SAME_32(INSN, FUNC) \
|
||||
static void gen_##INSN##_3s(unsigned vece, uint32_t rd_ofs, \
|
||||
uint32_t rn_ofs, uint32_t rm_ofs, \
|
||||
uint32_t oprsz, uint32_t maxsz) \
|
||||
{ \
|
||||
static const GVecGen3 ops[4] = { \
|
||||
{ .fni4 = gen_helper_neon_##FUNC##8 }, \
|
||||
{ .fni4 = gen_helper_neon_##FUNC##16 }, \
|
||||
{ .fni4 = gen_helper_neon_##FUNC##32 }, \
|
||||
{ 0 }, \
|
||||
}; \
|
||||
tcg_gen_gvec_3(rd_ofs, rn_ofs, rm_ofs, oprsz, maxsz, &ops[vece]); \
|
||||
} \
|
||||
static bool trans_##INSN##_3s(DisasContext *s, arg_3same *a) \
|
||||
{ \
|
||||
if (a->size > 2) { \
|
||||
return false; \
|
||||
} \
|
||||
return do_3same(s, a, gen_##INSN##_3s); \
|
||||
}
|
||||
|
||||
/*
|
||||
* Some helper functions need to be passed the tcg_env. In order
|
||||
* to use those with the gvec APIs like tcg_gen_gvec_3() we need
|
||||
|
@ -955,9 +936,6 @@ DO_SHA2(SHA256SU1, gen_helper_crypto_sha256su1)
|
|||
FUNC(d, tcg_env, n, m); \
|
||||
}
|
||||
|
||||
DO_3SAME_32(VRHADD_S, rhadd_s)
|
||||
DO_3SAME_32(VRHADD_U, rhadd_u)
|
||||
|
||||
#define DO_3SAME_VQDMULH(INSN, FUNC) \
|
||||
WRAP_ENV_FN(gen_##INSN##_tramp16, gen_helper_neon_##FUNC##_s16); \
|
||||
WRAP_ENV_FN(gen_##INSN##_tramp32, gen_helper_neon_##FUNC##_s32); \
|
||||
|
|
|
@ -480,6 +480,10 @@ void gen_gvec_shsub(unsigned vece, uint32_t rd_ofs, uint32_t rn_ofs,
|
|||
uint32_t rm_ofs, uint32_t opr_sz, uint32_t max_sz);
|
||||
void gen_gvec_uhsub(unsigned vece, uint32_t rd_ofs, uint32_t rn_ofs,
|
||||
uint32_t rm_ofs, uint32_t opr_sz, uint32_t max_sz);
|
||||
void gen_gvec_srhadd(unsigned vece, uint32_t rd_ofs, uint32_t rn_ofs,
|
||||
uint32_t rm_ofs, uint32_t opr_sz, uint32_t max_sz);
|
||||
void gen_gvec_urhadd(unsigned vece, uint32_t rd_ofs, uint32_t rn_ofs,
|
||||
uint32_t rm_ofs, uint32_t opr_sz, uint32_t max_sz);
|
||||
|
||||
void gen_cmtst_i64(TCGv_i64 d, TCGv_i64 a, TCGv_i64 b);
|
||||
void gen_ushl_i32(TCGv_i32 d, TCGv_i32 a, TCGv_i32 b);
|
||||
|
|
Loading…
Reference in New Issue