target/arm: Introduce gen_gvec_cnt, gen_gvec_rbit
Add gvec interfaces for CNT and RBIT operations. Use ctpop8 for CNT and revbit+bswap for RBIT. Reviewed-by: Peter Maydell <peter.maydell@linaro.org> Signed-off-by: Richard Henderson <richard.henderson@linaro.org> Message-id: 20241211163036.2297116-40-richard.henderson@linaro.org Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
This commit is contained in:
parent
4e728364c9
commit
4694d57458
@ -363,8 +363,8 @@ DEF_HELPER_1(neon_clz_u16, i32, i32)
|
|||||||
DEF_HELPER_1(neon_cls_s8, i32, i32)
|
DEF_HELPER_1(neon_cls_s8, i32, i32)
|
||||||
DEF_HELPER_1(neon_cls_s16, i32, i32)
|
DEF_HELPER_1(neon_cls_s16, i32, i32)
|
||||||
DEF_HELPER_1(neon_cls_s32, i32, i32)
|
DEF_HELPER_1(neon_cls_s32, i32, i32)
|
||||||
DEF_HELPER_1(neon_cnt_u8, i32, i32)
|
DEF_HELPER_FLAGS_3(gvec_cnt_b, TCG_CALL_NO_RWG, void, ptr, ptr, i32)
|
||||||
DEF_HELPER_FLAGS_1(neon_rbit_u8, TCG_CALL_NO_RWG_SE, i32, i32)
|
DEF_HELPER_FLAGS_3(gvec_rbit_b, TCG_CALL_NO_RWG, void, ptr, ptr, i32)
|
||||||
|
|
||||||
DEF_HELPER_3(neon_qdmulh_s16, i32, env, i32, i32)
|
DEF_HELPER_3(neon_qdmulh_s16, i32, env, i32, i32)
|
||||||
DEF_HELPER_3(neon_qrdmulh_s16, i32, env, i32, i32)
|
DEF_HELPER_3(neon_qrdmulh_s16, i32, env, i32, i32)
|
||||||
|
@ -2393,3 +2393,19 @@ void gen_gvec_clz(unsigned vece, uint32_t rd_ofs, uint32_t rn_ofs,
|
|||||||
assert(vece <= MO_32);
|
assert(vece <= MO_32);
|
||||||
tcg_gen_gvec_2(rd_ofs, rn_ofs, opr_sz, max_sz, &g[vece]);
|
tcg_gen_gvec_2(rd_ofs, rn_ofs, opr_sz, max_sz, &g[vece]);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void gen_gvec_cnt(unsigned vece, uint32_t rd_ofs, uint32_t rn_ofs,
|
||||||
|
uint32_t opr_sz, uint32_t max_sz)
|
||||||
|
{
|
||||||
|
assert(vece == MO_8);
|
||||||
|
tcg_gen_gvec_2_ool(rd_ofs, rn_ofs, opr_sz, max_sz, 0,
|
||||||
|
gen_helper_gvec_cnt_b);
|
||||||
|
}
|
||||||
|
|
||||||
|
void gen_gvec_rbit(unsigned vece, uint32_t rd_ofs, uint32_t rn_ofs,
|
||||||
|
uint32_t opr_sz, uint32_t max_sz)
|
||||||
|
{
|
||||||
|
assert(vece == MO_8);
|
||||||
|
tcg_gen_gvec_2_ool(rd_ofs, rn_ofs, opr_sz, max_sz, 0,
|
||||||
|
gen_helper_gvec_rbit_b);
|
||||||
|
}
|
||||||
|
@ -525,27 +525,6 @@ uint32_t HELPER(neon_cls_s32)(uint32_t x)
|
|||||||
return count - 1;
|
return count - 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Bit count. */
|
|
||||||
uint32_t HELPER(neon_cnt_u8)(uint32_t x)
|
|
||||||
{
|
|
||||||
x = (x & 0x55555555) + ((x >> 1) & 0x55555555);
|
|
||||||
x = (x & 0x33333333) + ((x >> 2) & 0x33333333);
|
|
||||||
x = (x & 0x0f0f0f0f) + ((x >> 4) & 0x0f0f0f0f);
|
|
||||||
return x;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Reverse bits in each 8 bit word */
|
|
||||||
uint32_t HELPER(neon_rbit_u8)(uint32_t x)
|
|
||||||
{
|
|
||||||
x = ((x & 0xf0f0f0f0) >> 4)
|
|
||||||
| ((x & 0x0f0f0f0f) << 4);
|
|
||||||
x = ((x & 0x88888888) >> 3)
|
|
||||||
| ((x & 0x44444444) >> 1)
|
|
||||||
| ((x & 0x22222222) << 1)
|
|
||||||
| ((x & 0x11111111) << 3);
|
|
||||||
return x;
|
|
||||||
}
|
|
||||||
|
|
||||||
#define NEON_QDMULH16(dest, src1, src2, round) do { \
|
#define NEON_QDMULH16(dest, src1, src2, round) do { \
|
||||||
uint32_t tmp = (int32_t)(int16_t) src1 * (int16_t) src2; \
|
uint32_t tmp = (int32_t)(int16_t) src1 * (int16_t) src2; \
|
||||||
if ((tmp ^ (tmp << 1)) & SIGNBIT) { \
|
if ((tmp ^ (tmp << 1)) & SIGNBIT) { \
|
||||||
|
@ -10324,12 +10324,15 @@ static void disas_simd_two_reg_misc(DisasContext *s, uint32_t insn)
|
|||||||
}
|
}
|
||||||
|
|
||||||
switch (opcode) {
|
switch (opcode) {
|
||||||
case 0x5:
|
case 0x5: /* CNT, NOT, RBIT */
|
||||||
if (u && size == 0) { /* NOT */
|
if (!u) {
|
||||||
|
gen_gvec_fn2(s, is_q, rd, rn, gen_gvec_cnt, 0);
|
||||||
|
} else if (size) {
|
||||||
|
gen_gvec_fn2(s, is_q, rd, rn, gen_gvec_rbit, 0);
|
||||||
|
} else {
|
||||||
gen_gvec_fn2(s, is_q, rd, rn, tcg_gen_gvec_not, 0);
|
gen_gvec_fn2(s, is_q, rd, rn, tcg_gen_gvec_not, 0);
|
||||||
return;
|
|
||||||
}
|
}
|
||||||
break;
|
return;
|
||||||
case 0x8: /* CMGT, CMGE */
|
case 0x8: /* CMGT, CMGE */
|
||||||
if (u) {
|
if (u) {
|
||||||
gen_gvec_fn2(s, is_q, rd, rn, gen_gvec_cge0, size);
|
gen_gvec_fn2(s, is_q, rd, rn, gen_gvec_cge0, size);
|
||||||
@ -10374,13 +10377,14 @@ static void disas_simd_two_reg_misc(DisasContext *s, uint32_t insn)
|
|||||||
} else {
|
} else {
|
||||||
int pass;
|
int pass;
|
||||||
|
|
||||||
|
assert(size == 2);
|
||||||
for (pass = 0; pass < (is_q ? 4 : 2); pass++) {
|
for (pass = 0; pass < (is_q ? 4 : 2); pass++) {
|
||||||
TCGv_i32 tcg_op = tcg_temp_new_i32();
|
TCGv_i32 tcg_op = tcg_temp_new_i32();
|
||||||
TCGv_i32 tcg_res = tcg_temp_new_i32();
|
TCGv_i32 tcg_res = tcg_temp_new_i32();
|
||||||
|
|
||||||
read_vec_element_i32(s, tcg_op, rn, pass, MO_32);
|
read_vec_element_i32(s, tcg_op, rn, pass, MO_32);
|
||||||
|
|
||||||
if (size == 2) {
|
{
|
||||||
/* Special cases for 32 bit elements */
|
/* Special cases for 32 bit elements */
|
||||||
switch (opcode) {
|
switch (opcode) {
|
||||||
case 0x2f: /* FABS */
|
case 0x2f: /* FABS */
|
||||||
@ -10434,25 +10438,7 @@ static void disas_simd_two_reg_misc(DisasContext *s, uint32_t insn)
|
|||||||
case 0x7: /* SQABS, SQNEG */
|
case 0x7: /* SQABS, SQNEG */
|
||||||
g_assert_not_reached();
|
g_assert_not_reached();
|
||||||
}
|
}
|
||||||
} else {
|
|
||||||
/* Use helpers for 8 and 16 bit elements */
|
|
||||||
switch (opcode) {
|
|
||||||
case 0x5: /* CNT, RBIT */
|
|
||||||
/* For these two insns size is part of the opcode specifier
|
|
||||||
* (handled earlier); they always operate on byte elements.
|
|
||||||
*/
|
|
||||||
if (u) {
|
|
||||||
gen_helper_neon_rbit_u8(tcg_res, tcg_op);
|
|
||||||
} else {
|
|
||||||
gen_helper_neon_cnt_u8(tcg_res, tcg_op);
|
|
||||||
}
|
}
|
||||||
break;
|
|
||||||
default:
|
|
||||||
case 0x7: /* SQABS, SQNEG */
|
|
||||||
g_assert_not_reached();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
write_vec_element_i32(s, tcg_res, rd, pass, MO_32);
|
write_vec_element_i32(s, tcg_res, rd, pass, MO_32);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -3131,6 +3131,14 @@ static bool trans_VMVN(DisasContext *s, arg_2misc *a)
|
|||||||
return do_2misc_vec(s, a, tcg_gen_gvec_not);
|
return do_2misc_vec(s, a, tcg_gen_gvec_not);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static bool trans_VCNT(DisasContext *s, arg_2misc *a)
|
||||||
|
{
|
||||||
|
if (a->size != 0) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return do_2misc_vec(s, a, gen_gvec_cnt);
|
||||||
|
}
|
||||||
|
|
||||||
#define WRAP_2M_3_OOL_FN(WRAPNAME, FUNC, DATA) \
|
#define WRAP_2M_3_OOL_FN(WRAPNAME, FUNC, DATA) \
|
||||||
static void WRAPNAME(unsigned vece, uint32_t rd_ofs, \
|
static void WRAPNAME(unsigned vece, uint32_t rd_ofs, \
|
||||||
uint32_t rm_ofs, uint32_t oprsz, \
|
uint32_t rm_ofs, uint32_t oprsz, \
|
||||||
@ -3229,14 +3237,6 @@ static bool trans_VREV16(DisasContext *s, arg_2misc *a)
|
|||||||
return do_2misc(s, a, gen_rev16);
|
return do_2misc(s, a, gen_rev16);
|
||||||
}
|
}
|
||||||
|
|
||||||
static bool trans_VCNT(DisasContext *s, arg_2misc *a)
|
|
||||||
{
|
|
||||||
if (a->size != 0) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
return do_2misc(s, a, gen_helper_neon_cnt_u8);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void gen_VABS_F(unsigned vece, uint32_t rd_ofs, uint32_t rm_ofs,
|
static void gen_VABS_F(unsigned vece, uint32_t rd_ofs, uint32_t rm_ofs,
|
||||||
uint32_t oprsz, uint32_t maxsz)
|
uint32_t oprsz, uint32_t maxsz)
|
||||||
{
|
{
|
||||||
|
@ -582,6 +582,10 @@ void gen_gvec_cls(unsigned vece, uint32_t rd_ofs, uint32_t rn_ofs,
|
|||||||
uint32_t opr_sz, uint32_t max_sz);
|
uint32_t opr_sz, uint32_t max_sz);
|
||||||
void gen_gvec_clz(unsigned vece, uint32_t rd_ofs, uint32_t rn_ofs,
|
void gen_gvec_clz(unsigned vece, uint32_t rd_ofs, uint32_t rn_ofs,
|
||||||
uint32_t opr_sz, uint32_t max_sz);
|
uint32_t opr_sz, uint32_t max_sz);
|
||||||
|
void gen_gvec_cnt(unsigned vece, uint32_t rd_ofs, uint32_t rn_ofs,
|
||||||
|
uint32_t opr_sz, uint32_t max_sz);
|
||||||
|
void gen_gvec_rbit(unsigned vece, uint32_t rd_ofs, uint32_t rn_ofs,
|
||||||
|
uint32_t opr_sz, uint32_t max_sz);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Forward to the isar_feature_* tests given a DisasContext pointer.
|
* Forward to the isar_feature_* tests given a DisasContext pointer.
|
||||||
|
@ -3066,3 +3066,27 @@ DO_CLAMP(gvec_uclamp_b, uint8_t)
|
|||||||
DO_CLAMP(gvec_uclamp_h, uint16_t)
|
DO_CLAMP(gvec_uclamp_h, uint16_t)
|
||||||
DO_CLAMP(gvec_uclamp_s, uint32_t)
|
DO_CLAMP(gvec_uclamp_s, uint32_t)
|
||||||
DO_CLAMP(gvec_uclamp_d, uint64_t)
|
DO_CLAMP(gvec_uclamp_d, uint64_t)
|
||||||
|
|
||||||
|
/* Bit count in each 8-bit word. */
|
||||||
|
void HELPER(gvec_cnt_b)(void *vd, void *vn, uint32_t desc)
|
||||||
|
{
|
||||||
|
intptr_t i, opr_sz = simd_oprsz(desc);
|
||||||
|
uint8_t *d = vd, *n = vn;
|
||||||
|
|
||||||
|
for (i = 0; i < opr_sz; ++i) {
|
||||||
|
d[i] = ctpop8(n[i]);
|
||||||
|
}
|
||||||
|
clear_tail(d, opr_sz, simd_maxsz(desc));
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Reverse bits in each 8 bit word */
|
||||||
|
void HELPER(gvec_rbit_b)(void *vd, void *vn, uint32_t desc)
|
||||||
|
{
|
||||||
|
intptr_t i, opr_sz = simd_oprsz(desc);
|
||||||
|
uint64_t *d = vd, *n = vn;
|
||||||
|
|
||||||
|
for (i = 0; i < opr_sz / 8; ++i) {
|
||||||
|
d[i] = revbit64(bswap64(n[i]));
|
||||||
|
}
|
||||||
|
clear_tail(d, opr_sz, simd_maxsz(desc));
|
||||||
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user