Allow libcalls for complex memcpy when optimizing for size.
authorJim Wilson <jimw@sifive.com>
Tue, 5 Nov 2019 22:34:40 +0000 (22:34 +0000)
committerJim Wilson <wilson@gcc.gnu.org>
Tue, 5 Nov 2019 22:34:40 +0000 (14:34 -0800)
The RISC-V backend wants to use a libcall when optimizing for size if
more than 6 instructions are needed.  Emit_move_complex asks for no
libcalls.  This case requires 8 insns for rv64 and 16 insns for rv32,
so we get fallback code that emits a loop.  Commit_one_edge_insertion
doesn't allow code inserted for a phi node on an edge to end with a
branch, and so this triggers an assertion.  This problem goes away if
we allow libcalls when optimizing for size, which gives the code the
RISC-V backend wants, and avoids triggering the assert.

gcc/
PR middle-end/92263
* expr.c (emit_move_complex): Only use BLOCK_OP_NO_LIBCALL when
optimize_insn_for_speed_p is true.

gcc/testsuite/
PR middle-end/92263
* gcc.dg/pr92263.c: New.

From-SVN: r277861

gcc/ChangeLog
gcc/expr.c
gcc/testsuite/ChangeLog
gcc/testsuite/gcc.dg/pr92263.c [new file with mode: 0644]

index f3deffc701ff96dddff2c591a0223fb08c3d12fb..4312d12506149a7d64d2ede5003c408115a82727 100644 (file)
@@ -1,3 +1,9 @@
+2019-11-05  Jim Wilson  <jimw@sifive.com>
+
+       PR middle-end/92263
+       * expr.c (emit_move_complex): Only use BLOCK_OP_NO_LIBCALL when
+       optimize_insn_for_speed_p is true.
+
 2019-11-05  Martin Sebor  <msebor@redhat.com>
 
        PR middle-end/92333
index 06e934e5be6e1d500460bba3779057455c85ba7a..0fd5890f8b4484b74c7f690fb748ef414c812945 100644 (file)
@@ -3571,11 +3571,13 @@ emit_move_complex (machine_mode mode, rtx x, rtx y)
       rtx_insn *ret;
 
       /* For memory to memory moves, optimal behavior can be had with the
-        existing block move logic.  */
+        existing block move logic.  But use normal expansion if optimizing
+        for size.  */
       if (MEM_P (x) && MEM_P (y))
        {
          emit_block_move (x, y, gen_int_mode (GET_MODE_SIZE (mode), Pmode),
-                          BLOCK_OP_NO_LIBCALL);
+                          (optimize_insn_for_speed_p()
+                           ? BLOCK_OP_NO_LIBCALL : BLOCK_OP_NORMAL));
          return get_last_insn ();
        }
 
index 4932615040b2caa37bac50fb396bf0cd021bfbc8..f34ed0873f8f9caf31d1b192de78db9f6ac9306d 100644 (file)
@@ -1,3 +1,8 @@
+2019-11-05  Jim Wilson  <jimw@sifive.com>
+
+       PR middle-end/92263
+       * gcc.dg/pr92263.c: New.
+
 2019-11-05  Martin Sebor  <msebor@redhat.com>
 
        PR middle-end/92333
diff --git a/gcc/testsuite/gcc.dg/pr92263.c b/gcc/testsuite/gcc.dg/pr92263.c
new file mode 100644 (file)
index 0000000..a79dfd1
--- /dev/null
@@ -0,0 +1,28 @@
+/* { dg-do compile } */
+/* { dg-options "-fno-tree-dce -fno-tree-forwprop -Os -ffloat-store" } */
+
+extern long double cabsl (_Complex long double);
+
+typedef struct {
+  int nsant, nvqd;
+  _Complex long double *vqd;
+} vsorc_t;
+vsorc_t vsorc;
+
+void foo(int next_job, int ain_num, int iped, long t) {
+  long double zpnorm;
+
+  while (!next_job)
+    if (ain_num)
+    {
+      if (iped == 1)
+        zpnorm = 0.0;
+      int indx = vsorc.nvqd-1;
+      vsorc.vqd[indx] = t*1.0fj;
+      if (cabsl(vsorc.vqd[indx]) < 1.e-20)
+        vsorc.vqd[indx] = 0.0fj;
+      zpnorm = t;
+      if (zpnorm > 0.0)
+        iped = vsorc.nsant;
+    }
+}