aarch64: Disallow invalid SVE addressing modes

The fix for PR22988 in 2018 added a new operand AARCH64_OPND_SVE_ADDR_R
to support implicit XZR offsets, but this fix had several flaws that
meant it accepted several invalid addressing modes:

1. The base register type wasn't properly checked when the optional
register offset was omitted.  This meant that
  ldff1b {z1.s}, p1/z,[z1.d]
was parsed as if it were
  ldff1b z1.d, p1/z, [x1.d, xzr].

2. The explicit offset parsing didn't include a shift type, so the new
operand would incorrectly parse
  ldff1h{z0.s}, p0/z, [x0, x0]
as if it were
  ldff1h{z0.s}, p0/z, [x0, x0, lsl #1].

3. Regardless of the above correctness issues, support for implicit
offsets should have been added by amending the operands in the existing
opcode table entries, instead of adding new duplicate table entires.

Issue 1 can be fixed by using an "if" instead of an "else if" in
parse_operands, while issue 2 can be fixed by failing when the first
condition is false.  This patch applies just these two fixes, leaving
issue 3 to be addressed in a subsequent more invasive patch.

The instructions removed from the test sme-5.d are architecturally
invalid. The new tests cover all of the affected ldff1 variants; the
issue also affected SME ZA ld1*/st1* instructions using the same operand
type.
This commit is contained in:
Alice Carlotti
2025-04-08 17:30:39 +01:00
parent b304edbeb6
commit 4250bea744
6 changed files with 106 additions and 9 deletions

View File

@@ -7817,7 +7817,15 @@ parse_operands (char *str, const aarch64_opcode *opcode)
info->addr.offset.is_reg = 1;
info->addr.offset.regno = 31;
}
else if (base_qualifier != AARCH64_OPND_QLF_X
else
{
/* This operand is used for different shift types, so we have to
reject explicit offsets. */
set_syntax_error (_("invalid addressing mode"));
goto failure;
}
if (base_qualifier != AARCH64_OPND_QLF_X
|| offset_qualifier != AARCH64_OPND_QLF_X)
{
set_syntax_error (_("invalid addressing mode"));

View File

@@ -87,7 +87,3 @@ Disassembly of section \.text:
134: e1dffe2f ld1q {za15v.q\[w15, 0\]}, p7/z, \[x17, xzr, lsl #4\]
138: e000ffef ld1b {za0v.b\[w15, 15\]}, p7/z, \[sp, x0\]
13c: e0010000 ld1b {za0h.b\[w12, 0\]}, p0/z, \[x0, x1\]
140: e0410000 ld1h {za0h.h\[w12, 0\]}, p0/z, \[x0, x1, lsl #1\]
144: e0819c0f ld1w {za3v.s\[w12, 3\]}, p7/z, \[x0, x1, lsl #2\]
148: e0c10000 ld1d {za0h.d\[w12, 0\]}, p0/z, \[x0, x1, lsl #3\]
14c: e1c18000 ld1q {za0v.q\[w12, 0\]}, p0/z, \[x0, x1, lsl #4\]

View File

@@ -95,7 +95,3 @@ ld1q {za15v.q[bar, #0]}, p7/z, [x17]
/* Optional LSL operator. */
ld1b {za0v.b[w15, 15]}, p7/z, [sp, x0, lsl #0]
ld1b {za0h.b[w12, 0]}, p0/z, [x0, x1]
ld1h {za0h.h[w12, 0]}, p0/z, [x0, x1]
ld1w {za3v.s[w12, 3]}, p7/z, [x0, x1]
ld1d {za0h.d[w12, 0]}, p0/z, [x0, x1]
ld1q {za0v.q[w12, 0]}, p0/z, [x0, x1]

View File

@@ -0,0 +1,4 @@
#as: -march=armv8-a+sve
#source: sve-ldff1-invalid.s
#error_output: sve-ldff1-invalid.l

View File

@@ -0,0 +1,58 @@
[^:]*: Assembler messages:
.*: Error: operand mismatch -- `ldff1b {z1\.b},p1/z,\[z1\.d\]'
.*: Info: did you mean this\?
.*: Info: ldff1b {z1\.d}, p1/z, \[z1\.d\]
.*: Error: operand mismatch -- `ldff1b {z1\.h},p1/z,\[z1\.s\]'
.*: Info: did you mean this\?
.*: Info: ldff1b {z1\.s}, p1/z, \[z1\.s\]
.*: Error: operand mismatch -- `ldff1b {z1\.s},p1/z,\[z1\.d\]'
.*: Info: did you mean this\?
.*: Info: ldff1b {z1\.s}, p1/z, \[z1\.s\]
.*: Error: operand mismatch -- `ldff1b {z1\.d},p1/z,\[z1\.s\]'
.*: Info: did you mean this\?
.*: Info: ldff1b {z1\.s}, p1/z, \[z1\.s\]
.*: Error: operand mismatch -- `ldff1d {z1\.d},p1/z,\[z1\.s\]'
.*: Info: did you mean this\?
.*: Info: ldff1d {z1\.d}, p1/z, \[z1\.d\]
.*: Error: operand mismatch -- `ldff1h {z1\.h},p1/z,\[z1\.s\]'
.*: Info: did you mean this\?
.*: Info: ldff1h {z1\.s}, p1/z, \[z1\.s\]
.*: Error: operand mismatch -- `ldff1h {z1\.s},p1/z,\[z1\.d\]'
.*: Info: did you mean this\?
.*: Info: ldff1h {z1\.s}, p1/z, \[z1\.s\]
.*: Error: operand mismatch -- `ldff1h {z1\.d},p1/z,\[z1\.s\]'
.*: Info: did you mean this\?
.*: Info: ldff1h {z1\.s}, p1/z, \[z1\.s\]
.*: Error: operand mismatch -- `ldff1sb {z1\.h},p1/z,\[z1\.d\]'
.*: Info: did you mean this\?
.*: Info: ldff1sb {z1\.d}, p1/z, \[z1\.d\]
.*: Error: operand mismatch -- `ldff1sb {z1\.s},p1/z,\[z1\.d\]'
.*: Info: did you mean this\?
.*: Info: ldff1sb {z1\.s}, p1/z, \[z1\.s\]
.*: Error: operand mismatch -- `ldff1sb {z1\.d},p1/z,\[z1\.s\]'
.*: Info: did you mean this\?
.*: Info: ldff1sb {z1\.s}, p1/z, \[z1\.s\]
.*: Error: operand mismatch -- `ldff1sh {z1\.s},p1/z,\[z1\.d\]'
.*: Info: did you mean this\?
.*: Info: ldff1sh {z1\.s}, p1/z, \[z1\.s\]
.*: Error: operand mismatch -- `ldff1sh {z1\.d},p1/z,\[z1\.s\]'
.*: Info: did you mean this\?
.*: Info: ldff1sh {z1\.s}, p1/z, \[z1\.s\]
.*: Error: operand mismatch -- `ldff1sw {z1\.d},p1/z,\[z1\.s\]'
.*: Info: did you mean this\?
.*: Info: ldff1sw {z1\.d}, p1/z, \[z1\.d\]
.*: Error: operand mismatch -- `ldff1w {z1\.s},p1/z,\[z1\.d\]'
.*: Info: did you mean this\?
.*: Info: ldff1w {z1\.s}, p1/z, \[z1\.s\]
.*: Error: operand mismatch -- `ldff1w {z1\.d},p1/z,\[z1\.s\]'
.*: Info: did you mean this\?
.*: Info: ldff1w {z1\.s}, p1/z, \[z1\.s\]
.*: Error: invalid addressing mode at operand 3 -- `ldff1d {z1\.d},p1/z,\[x1,x1\]'
.*: Error: invalid addressing mode at operand 3 -- `ldff1h {z1\.h},p1/z,\[x1,x1\]'
.*: Error: invalid addressing mode at operand 3 -- `ldff1h {z1\.s},p1/z,\[x1,x1\]'
.*: Error: invalid addressing mode at operand 3 -- `ldff1h {z1\.d},p1/z,\[x1,x1\]'
.*: Error: invalid addressing mode at operand 3 -- `ldff1sh {z1\.s},p1/z,\[x1,x1\]'
.*: Error: invalid addressing mode at operand 3 -- `ldff1sh {z1\.d},p1/z,\[x1,x1\]'
.*: Error: invalid addressing mode at operand 3 -- `ldff1sw {z1\.d},p1/z,\[x1,x1\]'
.*: Error: invalid addressing mode at operand 3 -- `ldff1w {z1\.s},p1/z,\[x1,x1\]'
.*: Error: invalid addressing mode at operand 3 -- `ldff1w {z1\.d},p1/z,\[x1,x1\]'

View File

@@ -0,0 +1,35 @@
// The fix for PR22988 accidentally allowed invalid addressing mode
// ldff1b z1.d, p1/z, [z1.d]
// which was parsed equivalently to the valid instruction:
// ldff1b z1.d, p1/z, [x1.d, xzr];
// and
// ldff1h{z1.s}, p1/z, [x1, x1]
// which was parsed as:
// ldff1h{z1.s}, p1/z, [x1, x1, lsl #1]
ldff1b {z1.b}, p1/z, [z1.d]
ldff1b {z1.h}, p1/z, [z1.s]
ldff1b {z1.s}, p1/z, [z1.d]
ldff1b {z1.d}, p1/z, [z1.s]
ldff1d {z1.d}, p1/z, [z1.s]
ldff1h {z1.h}, p1/z, [z1.s]
ldff1h {z1.s}, p1/z, [z1.d]
ldff1h {z1.d}, p1/z, [z1.s]
ldff1sb {z1.h}, p1/z, [z1.d]
ldff1sb {z1.s}, p1/z, [z1.d]
ldff1sb {z1.d}, p1/z, [z1.s]
ldff1sh {z1.s}, p1/z, [z1.d]
ldff1sh {z1.d}, p1/z, [z1.s]
ldff1sw {z1.d}, p1/z, [z1.s]
ldff1w {z1.s}, p1/z, [z1.d]
ldff1w {z1.d}, p1/z, [z1.s]
ldff1d {z1.d}, p1/z, [x1, x1]
ldff1h {z1.h}, p1/z, [x1, x1]
ldff1h {z1.s}, p1/z, [x1, x1]
ldff1h {z1.d}, p1/z, [x1, x1]
ldff1sh {z1.s}, p1/z, [x1, x1]
ldff1sh {z1.d}, p1/z, [x1, x1]
ldff1sw {z1.d}, p1/z, [x1, x1]
ldff1w {z1.s}, p1/z, [x1, x1]
ldff1w {z1.d}, p1/z, [x1, x1]