target/riscv: rvv: Apply vext_check_input_eew to vrgather instructions to check mismatched input EEWs encoding constraint
According to the v spec, a vector register cannot be used to provide source
operands with more than one EEW for a single instruction.
The vs1 EEW of vrgatherei16.vv is 16.
Co-authored-by: Anton Blanchard <antonb@tenstorrent.com>
Reviewed-by: Daniel Henrique Barboza <dbarboza@ventanamicro.com>
Signed-off-by: Max Chou <max.chou@sifive.com>
Message-ID: <20250408103938.3623486-4-max.chou@sifive.com>
Signed-off-by: Alistair Francis <alistair.francis@wdc.com>
Cc: qemu-stable@nongnu.org
(cherry picked from commit 629c2a8dd7)
Signed-off-by: Michael Tokarev <mjt@tls.msk.ru>
This commit is contained in:
parent
027ea4a2f5
commit
bc502d4de9
1 changed files with 32 additions and 0 deletions
|
|
@ -379,6 +379,35 @@ static bool vext_check_ld_index(DisasContext *s, int vd, int vs2,
|
|||
return ret;
|
||||
}
|
||||
|
||||
/*
|
||||
* Check whether a vector register is used to provide source operands with
|
||||
* more than one EEW for the vector instruction.
|
||||
* Returns true if the instruction has valid encoding
|
||||
* Returns false if encoding violates the mismatched input EEWs constraint
|
||||
*/
|
||||
static bool vext_check_input_eew(DisasContext *s, int vs1, uint8_t eew_vs1,
|
||||
int vs2, uint8_t eew_vs2, int vm)
|
||||
{
|
||||
bool is_valid = true;
|
||||
int8_t emul_vs1 = eew_vs1 - s->sew + s->lmul;
|
||||
int8_t emul_vs2 = eew_vs2 - s->sew + s->lmul;
|
||||
|
||||
/* When vm is 0, vs1 & vs2(EEW!=1) group can't overlap v0 (EEW=1) */
|
||||
if ((vs1 != -1 && !require_vm(vm, vs1)) ||
|
||||
(vs2 != -1 && !require_vm(vm, vs2))) {
|
||||
is_valid = false;
|
||||
}
|
||||
|
||||
/* When eew_vs1 != eew_vs2, check whether vs1 and vs2 are overlapped */
|
||||
if ((vs1 != -1 && vs2 != -1) && (eew_vs1 != eew_vs2) &&
|
||||
is_overlapped(vs1, 1 << MAX(emul_vs1, 0),
|
||||
vs2, 1 << MAX(emul_vs2, 0))) {
|
||||
is_valid = false;
|
||||
}
|
||||
|
||||
return is_valid;
|
||||
}
|
||||
|
||||
static bool vext_check_ss(DisasContext *s, int vd, int vs, int vm)
|
||||
{
|
||||
return require_vm(vm, vd) &&
|
||||
|
|
@ -3449,6 +3478,7 @@ static bool vrgather_vv_check(DisasContext *s, arg_rmrr *a)
|
|||
{
|
||||
return require_rvv(s) &&
|
||||
vext_check_isa_ill(s) &&
|
||||
vext_check_input_eew(s, a->rs1, s->sew, a->rs2, s->sew, a->vm) &&
|
||||
require_align(a->rd, s->lmul) &&
|
||||
require_align(a->rs1, s->lmul) &&
|
||||
require_align(a->rs2, s->lmul) &&
|
||||
|
|
@ -3461,6 +3491,7 @@ static bool vrgatherei16_vv_check(DisasContext *s, arg_rmrr *a)
|
|||
int8_t emul = MO_16 - s->sew + s->lmul;
|
||||
return require_rvv(s) &&
|
||||
vext_check_isa_ill(s) &&
|
||||
vext_check_input_eew(s, a->rs1, MO_16, a->rs2, s->sew, a->vm) &&
|
||||
(emul >= -3 && emul <= 3) &&
|
||||
require_align(a->rd, s->lmul) &&
|
||||
require_align(a->rs1, emul) &&
|
||||
|
|
@ -3480,6 +3511,7 @@ static bool vrgather_vx_check(DisasContext *s, arg_rmrr *a)
|
|||
{
|
||||
return require_rvv(s) &&
|
||||
vext_check_isa_ill(s) &&
|
||||
vext_check_input_eew(s, -1, MO_64, a->rs2, s->sew, a->vm) &&
|
||||
require_align(a->rd, s->lmul) &&
|
||||
require_align(a->rs2, s->lmul) &&
|
||||
(a->rd != a->rs2) &&
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue