]> gcc.gnu.org Git - gcc.git/commitdiff
aarch64: Fix up aarch64_compare_and_swaphi pattern [PR94368]
authorJakub Jelinek <jakub@redhat.com>
Tue, 31 Mar 2020 09:08:22 +0000 (11:08 +0200)
committerJakub Jelinek <jakub@redhat.com>
Tue, 31 Mar 2020 09:08:22 +0000 (11:08 +0200)
The following testcase ICEs in final_scan_insn_1.  The problem is in the
@aarch64_compare_and_swaphi define_insn_and_split, since 9 it uses
aarch64_plushi_operand predicate for the "expected value" operand, which
allows either 0..0xfff constants or 0x1000..0xf000 constants (i.e. HImode
values which when zero extended are either 0..0xfff or (0..0xfff) << 12).
The problem is that RA doesn't care about predicates, it honors just
constraints and the used constraint on the operand is n, which means any
HImode CONST_SCALAR_INT.  In the testcase LRA thus propagates the -1
value into the insn.
This is a define_insn_and_split which requires mandatory split.
But during split2 pass, we check the predicate (and don't check
constraints), which fails and thus we don't split it and during final ICE
because the mandatory splitting didn't happen.

The following patch fixes it by adding a matching constraint to the
predicate and using it.

2020-03-31  Jakub Jelinek  <jakub@redhat.com>

PR target/94368
* config/aarch64/constraints.md (Uph): New constraint.
* config/aarch64/atomics.md (cas_short_expected_imm): New mode attr.
(@aarch64_compare_and_swap<mode>): Use it instead of n in operand 2's
constraint.

* gcc.dg/pr94368.c: New test.

gcc/ChangeLog
gcc/config/aarch64/atomics.md
gcc/config/aarch64/constraints.md
gcc/testsuite/ChangeLog
gcc/testsuite/gcc.dg/pr94368.c [new file with mode: 0644]

index 4686c59f90c2198cc78f1c1a33eef3b160476c9b..88e279eb4747b444da779ea71805c9d1807294ce 100644 (file)
@@ -1,3 +1,11 @@
+2020-03-31  Jakub Jelinek  <jakub@redhat.com>
+
+       PR target/94368
+       * config/aarch64/constraints.md (Uph): New constraint.
+       * config/aarch64/atomics.md (cas_short_expected_imm): New mode attr.
+       (@aarch64_compare_and_swap<mode>): Use it instead of n in operand 2's
+       constraint.
+
 2020-03-31  Marc Glisse  <marc.glisse@inria.fr>
            Jakub Jelinek  <jakub@redhat.com>
 
index 996947799b5ef8445e9786b94e1ce62fd16e5b5c..b27a80fda965b4df3eca8127b7b02a6649123726 100644 (file)
@@ -38,6 +38,8 @@
 
 (define_mode_attr cas_short_expected_pred
   [(QI "aarch64_reg_or_imm") (HI "aarch64_plushi_operand")])
+(define_mode_attr cas_short_expected_imm
+  [(QI "n") (HI "Uph")])
 
 (define_insn_and_split "@aarch64_compare_and_swap<mode>"
   [(set (reg:CC CC_REGNUM)                                     ;; bool out
@@ -47,7 +49,8 @@
       (match_operand:SHORT 1 "aarch64_sync_memory_operand" "+Q"))) ;; memory
    (set (match_dup 1)
     (unspec_volatile:SHORT
-      [(match_operand:SHORT 2 "<cas_short_expected_pred>" "rn")        ;; expected
+      [(match_operand:SHORT 2 "<cas_short_expected_pred>"
+                             "r<cas_short_expected_imm>")      ;; expected
        (match_operand:SHORT 3 "aarch64_reg_or_zero" "rZ")      ;; desired
        (match_operand:SI 4 "const_int_operand")                        ;; is_weak
        (match_operand:SI 5 "const_int_operand")                        ;; mod_s
index 7ae072be3951bbbf21023ab48c32ec681863d9d6..d993268a187fad9c80c32b16d8e95b26783bde24 100644 (file)
   (and (match_code "const_int")
        (match_test "(unsigned) exact_log2 (ival) <= 4")))
 
+(define_constraint "Uph"
+  "@internal
+  A constraint that matches HImode integers zero extendable to
+  SImode plus_operand."
+  (and (match_code "const_int")
+       (match_test "aarch64_plushi_immediate (op, VOIDmode)")))
+
 (define_memory_constraint "Q"
  "A memory address which uses a single base register with no offset."
  (and (match_code "mem")
index 32d5309882efb3664d006010d86c6c45f0f3cb87..aa269f6071ff0652722777f824ae6529b37ff853 100644 (file)
@@ -1,5 +1,8 @@
 2020-03-31  Jakub Jelinek  <jakub@redhat.com>
 
+       PR target/94368
+       * gcc.dg/pr94368.c: New test.
+
        PR middle-end/94412
        * gcc.c-torture/execute/pr94412.c: New test.
 
diff --git a/gcc/testsuite/gcc.dg/pr94368.c b/gcc/testsuite/gcc.dg/pr94368.c
new file mode 100644 (file)
index 0000000..1267b82
--- /dev/null
@@ -0,0 +1,25 @@
+/* PR target/94368 */
+/* { dg-do compile { target fpic } } */
+/* { dg-options "-fpic -O1 -fcommon" } */
+
+int b, c, d, e, f, h;
+short g;
+int foo (int) __attribute__ ((__const__));
+
+void
+bar (void)
+{
+  while (1)
+    {
+      while (1)
+       {
+         __atomic_load_n (&e, 0);
+         if (foo (2))
+           __sync_val_compare_and_swap (&c, 0, f);
+         b = 1;
+         if (h == e)
+           break;
+       }
+      __sync_val_compare_and_swap (&g, -1, f);
+    }
+}
This page took 0.132428 seconds and 5 git commands to generate.