On 7/15/24 08:33, Cupertino Miranda wrote: > Both xchg and cmpxchg instructions, in the pseudo-C dialect, do not > expect their memory address operand to be surrounded by parentheses. > For example, it should be output as "w0 =cmpxchg32_32(r8+8,w0,w2)" > instead of "w0 =cmpxchg32_32((r8+8),w0,w2)".
After some brief searching, looks like these extra parens in [cmp]xchg* are an unintended consequence of: bb5f619a938 bpf: fix printing of memory operands in pseudoc asm dialect > > This patch implements an operand modifier 'M' which marks the > instruction templates that do not expect the parentheses, and adds it do > xchg and cmpxchg templates. One very minor nit below, otherwise LGTM. Please apply. Thanks for the fix! > > gcc/ChangeLog: > * config/bpf/atomic.md (atomic_compare_and_swap, > atomic_exchange): Add operand modifier %M to the first > operand. > * config/bpf/bpf.cc (no_parentheses_mem_operand): Create > variable. > (bpf_print_operand): Set no_parentheses_mem_operand variable if > %M operand is used. > (bpf_print_operand_address): Conditionally output parentheses. > > gcc/testsuite/ChangeLog: > * gcc.target/bpf/pseudoc-atomic-memaddr-op.c: Add test. > --- > gcc/config/bpf/atomic.md | 4 +-- > gcc/config/bpf/bpf.cc | 20 ++++++++--- > .../bpf/pseudoc-atomic-memaddr-op.c | 36 +++++++++++++++++++ > 3 files changed, 54 insertions(+), 6 deletions(-) > create mode 100644 gcc/testsuite/gcc.target/bpf/pseudoc-atomic-memaddr-op.c > > diff --git a/gcc/config/bpf/atomic.md b/gcc/config/bpf/atomic.md > index 65bd5f266a1..be4511bb51b 100644 > --- a/gcc/config/bpf/atomic.md > +++ b/gcc/config/bpf/atomic.md > @@ -129,7 +129,7 @@ > (set (match_dup 1) > (match_operand:AMO 2 "nonmemory_operand" "0"))] > "bpf_has_v3_atomics" > - "{axchg<msuffix>\t%1,%0|%w0 = xchg<pcaxsuffix>(%1, %w0)}") > + "{axchg<msuffix>\t%1,%0|%w0 = xchg<pcaxsuffix>(%M1, %w0)}") > > ;; The eBPF atomic-compare-and-exchange instruction has the form > ;; acmp [%dst+offset], %src > @@ -182,4 +182,4 @@ > (match_operand:AMO 3 "register_operand")] ;; desired > UNSPEC_ACMP))] > "bpf_has_v3_atomics" > - "{acmp<msuffix>\t%1,%3|%w0 = cmpxchg<pcaxsuffix>(%1, %w0, %w3)}") > + "{acmp<msuffix>\t%1,%3|%w0 = cmpxchg<pcaxsuffix>(%M1, %w0, %w3)}") > diff --git a/gcc/config/bpf/bpf.cc b/gcc/config/bpf/bpf.cc > index 687e7ba49c1..a6b6e20731c 100644 > --- a/gcc/config/bpf/bpf.cc > +++ b/gcc/config/bpf/bpf.cc > @@ -835,6 +835,11 @@ bpf_print_register (FILE *file, rtx op, int code) > } > } > > +/* Variable defined to implement 'M' operand modifier for the special cases > + where the parentheses should not be printed surrounding a memory address > + operand. */ > +static bool no_parentheses_mem_operand; > + > /* Print an instruction operand. This function is called in the macro > PRINT_OPERAND defined in bpf.h */ > > @@ -847,6 +852,7 @@ bpf_print_operand (FILE *file, rtx op, int code) > bpf_print_register (file, op, code); > break; > case MEM: > + no_parentheses_mem_operand = (code == 'M'); > output_address (GET_MODE (op), XEXP (op, 0)); > break; > case CONST_DOUBLE: > @@ -891,6 +897,9 @@ bpf_print_operand (FILE *file, rtx op, int code) > } > } > > +#define PAREN_OPEN (asm_dialect == ASM_NORMAL ? "[" : > no_parentheses_mem_operand ? "" : "(") > +#define PAREN_CLOSE (asm_dialect == ASM_NORMAL ? "]" : > no_parentheses_mem_operand ? "" : ")") > + > /* Print an operand which is an address. This function should handle > any legit address, as accepted by bpf_legitimate_address_p, and > also addresses that are valid in CALL instructions. > @@ -904,9 +913,9 @@ bpf_print_operand_address (FILE *file, rtx addr) > switch (GET_CODE (addr)) > { > case REG: > - fprintf (file, asm_dialect == ASM_NORMAL ? "[" : "("); > + fprintf (file, "%s", PAREN_OPEN); > bpf_print_register (file, addr, 0); > - fprintf (file, asm_dialect == ASM_NORMAL ? "+0]" : "+0)"); > + fprintf (file, "+0%s", PAREN_CLOSE); > break; > case PLUS: > { > @@ -923,14 +932,14 @@ bpf_print_operand_address (FILE *file, rtx addr) > || (GET_CODE (op1) == UNSPEC > && XINT (op1, 1) == UNSPEC_CORE_RELOC))) > { > - fprintf (file, asm_dialect == ASM_NORMAL ? "[" : "("); > + fprintf (file, "%s", PAREN_OPEN); > bpf_print_register (file, op0, 0); > fprintf (file, "+"); > if (GET_CODE (op1) == UNSPEC) > output_addr_const (file, XVECEXP (op1, 0, 0)); > else > output_addr_const (file, op1); > - fprintf (file, asm_dialect == ASM_NORMAL ? "]" : ")"); > + fprintf (file, "%s", PAREN_CLOSE); > } > else > fatal_insn ("invalid address in operand", addr); > @@ -948,6 +957,9 @@ bpf_print_operand_address (FILE *file, rtx addr) > } > } > > +#undef PAREN_OPEN > +#undef PAREN_CLOSE > + > /* Add a BPF builtin function with NAME, CODE and TYPE. Return > the function decl or NULL_TREE if the builtin was not added. */ > > diff --git a/gcc/testsuite/gcc.target/bpf/pseudoc-atomic-memaddr-op.c > b/gcc/testsuite/gcc.target/bpf/pseudoc-atomic-memaddr-op.c > new file mode 100644 > index 00000000000..758faf04cc2 > --- /dev/null > +++ b/gcc/testsuite/gcc.target/bpf/pseudoc-atomic-memaddr-op.c > @@ -0,0 +1,36 @@ > +/* { dg-do compile } */ > +/* { dg-options "-mv3-atomics -O2 -masm=pseudoc" } */ > + > +int > +foo (int *p, int *expected, int desired) > +{ > + return __atomic_compare_exchange (p, expected, &desired, 0, > + __ATOMIC_ACQUIRE, __ATOMIC_RELAXED); > +} > + > +int > +foo64 (long *p, long *expected, long desired) > +{ > + return __atomic_compare_exchange (p, expected, &desired, 0, > + __ATOMIC_ACQUIRE, __ATOMIC_RELAXED); > +} > + > +int bar (int *p, int *new) nit: style - return type should be on its own line > +{ > + int old; > + __atomic_exchange (p, new, &old, __ATOMIC_RELAXED); > + return old; > +} > + > +int bar64 (long *p, long *new) same here > +{ > + long old; > + __atomic_exchange (p, new, &old, __ATOMIC_SEQ_CST); > + return old; > +} > + > +/* { dg-final { scan-assembler "r. = cmpxchg_64\\(r.\\+0, r., r.\\)" } } */ > +/* { dg-final { scan-assembler "w. = cmpxchg32_32\\(r.\\+0, w., w.\\)" } } */ > +/* { dg-final { scan-assembler-times "w. = xchg32_32\\(r.\\+0, w.\\)" 1 } } > */ > +/* { dg-final { scan-assembler-times "r. = xchg_64\\(r.\\+0, r.\\)" 1 } } */ > +