tree-optimization/94103 avoid CSE of loads with padding
authorRichard Biener <rguenther@suse.de>
Thu, 12 Mar 2020 13:18:35 +0000 (14:18 +0100)
committerRichard Biener <rguenther@suse.de>
Thu, 12 Mar 2020 13:19:36 +0000 (14:19 +0100)
VN currently replaces a load of a 16 byte entity 128 bits of precision
(TImode) with the result of a load of a 16 byte entity with 80 bits of
mode precision (XFmode).  That will go downhill since if the padding
bits are not actually filled with memory contents those bits are
missing.

2020-03-12  Richard Biener  <rguenther@suse.de>

PR tree-optimization/94103
* tree-ssa-sccvn.c (visit_reference_op_load): Avoid type
punning when the mode precision is not sufficient.

* gcc.target/i386/pr94103.c: New testcase.

gcc/ChangeLog
gcc/testsuite/ChangeLog
gcc/testsuite/gcc.target/i386/pr94103.c [new file with mode: 0644]
gcc/tree-ssa-sccvn.c

index 22ba5ed36e68302757346f2c8b0c0401a095b2b3..a43b453178d86324f7b233dd2fcfd2f019568565 100644 (file)
@@ -1,3 +1,9 @@
+2020-03-12  Richard Biener  <rguenther@suse.de>
+
+       PR tree-optimization/94103
+       * tree-ssa-sccvn.c (visit_reference_op_load): Avoid type
+       punning when the mode precision is not sufficient.
+
 2020-03-12  H.J. Lu  <hongjiu.lu@intel.com>
 
        PR target/89229
index a1f0e3a18c5c13e742646190d511ac361403acca..a5730ebbbc07052e180b14067c529256707b6dda 100644 (file)
@@ -1,3 +1,8 @@
+2020-03-12  Richard Biener  <rguenther@suse.de>
+
+       PR tree-optimization/94103
+       * gcc.target/i386/pr94103.c: New testcase.
+
 2020-03-12  Tobias Burnus  <tobias@codesourcery.com>
 
        PR middle-end/94120
diff --git a/gcc/testsuite/gcc.target/i386/pr94103.c b/gcc/testsuite/gcc.target/i386/pr94103.c
new file mode 100644 (file)
index 0000000..91b5fc6
--- /dev/null
@@ -0,0 +1,17 @@
+/* { dg-do run { target lp64 } } */
+/* { dg-options "-O3" } */
+
+int main()
+{
+  long double x;
+  unsigned long u[2] = {0xEEEEEEEEEEEEEEEEUL, 0xEEEEEEEEEEEEEEEEUL};
+  __builtin_memcpy(&x, &u, sizeof x);
+  __builtin_memcpy(&u, &x, sizeof u);
+  ++*(unsigned char *)&x;
+  (void)-x;
+  __builtin_memcpy(&u, &x, sizeof u);
+  if (u[1] != 0xEEEEEEEEEEEEEEEEUL
+      || u[0] != 0xEEEEEEEEEEEEEEEFUL)
+    __builtin_abort ();
+  return 0;
+}
index b7174cd603ff008ed1c988793d5690b79a9e8c9a..150ddad3e69f7ecd0965507144902f98724bbeba 100644 (file)
@@ -4899,13 +4899,22 @@ visit_reference_op_load (tree lhs, tree op, gimple *stmt)
   if (result
       && !useless_type_conversion_p (TREE_TYPE (result), TREE_TYPE (op)))
     {
-      /* We will be setting the value number of lhs to the value number
-        of VIEW_CONVERT_EXPR <TREE_TYPE (result)> (result).
-        So first simplify and lookup this expression to see if it
-        is already available.  */
-      gimple_match_op res_op (gimple_match_cond::UNCOND,
-                             VIEW_CONVERT_EXPR, TREE_TYPE (op), result);
-      result = vn_nary_build_or_lookup (&res_op);
+      /* Avoid the type punning in case the result mode has padding where
+        the op we lookup has not.  */
+      if (maybe_lt (GET_MODE_PRECISION (TYPE_MODE (TREE_TYPE (result))),
+                   GET_MODE_PRECISION (TYPE_MODE (TREE_TYPE (op)))))
+       result = NULL_TREE;
+      else
+       {
+         /* We will be setting the value number of lhs to the value number
+            of VIEW_CONVERT_EXPR <TREE_TYPE (result)> (result).
+            So first simplify and lookup this expression to see if it
+            is already available.  */
+         gimple_match_op res_op (gimple_match_cond::UNCOND,
+                                 VIEW_CONVERT_EXPR, TREE_TYPE (op), result);
+         result = vn_nary_build_or_lookup (&res_op);
+       }
+
       /* When building the conversion fails avoid inserting the reference
          again.  */
       if (!result)