re PR tree-optimization/89143 (comparison of abs(i) against excessive constant less...
authorJakub Jelinek <jakub@redhat.com>
Fri, 1 Feb 2019 08:44:23 +0000 (09:44 +0100)
committerJakub Jelinek <jakub@gcc.gnu.org>
Fri, 1 Feb 2019 08:44:23 +0000 (09:44 +0100)
PR tree-optimization/89143
* wide-int-range.h (wide_int_range_absu): Declare.
* wide-int-range.cc (wide_int_range_absu): New function.
* tree-vrp.c (extract_range_from_unary_expr): Handle ABSU_EXPR.

* gcc.dg/tree-ssa/vrp121.c: New test.

From-SVN: r268445

gcc/ChangeLog
gcc/testsuite/ChangeLog
gcc/testsuite/gcc.dg/tree-ssa/vrp121.c [new file with mode: 0644]
gcc/tree-vrp.c
gcc/wide-int-range.cc
gcc/wide-int-range.h

index 485bac73f2bdb2ef944f76b97a1b58d2b73861e5..0da61908fee02feba34559dfc8ec6c517b30af6f 100644 (file)
@@ -1,5 +1,10 @@
 2019-02-01  Jakub Jelinek  <jakub@redhat.com>
 
+       PR tree-optimization/89143
+       * wide-int-range.h (wide_int_range_absu): Declare.
+       * wide-int-range.cc (wide_int_range_absu): New function.
+       * tree-vrp.c (extract_range_from_unary_expr): Handle ABSU_EXPR.
+
        PR tree-optimization/88107
        * tree-cfg.c (find_outermost_region_in_block): Add ALL argument,
        instead of assertion that eh_region_outermost is non-NULL, if it
index fb18cf118736e89b36ebe45b86aa3b5982bc397b..9d50c3e082a6201d30d501332ee2e21f8b0e58f0 100644 (file)
@@ -1,5 +1,8 @@
 2019-02-01  Jakub Jelinek  <jakub@redhat.com>
 
+       PR tree-optimization/89143
+       * gcc.dg/tree-ssa/vrp121.c: New test.
+
        PR tree-optimization/88107
        * gcc.dg/gomp/pr88107.c: New test.
 
diff --git a/gcc/testsuite/gcc.dg/tree-ssa/vrp121.c b/gcc/testsuite/gcc.dg/tree-ssa/vrp121.c
new file mode 100644 (file)
index 0000000..3af5a9a
--- /dev/null
@@ -0,0 +1,67 @@
+/* PR tree-optimization/89143 */
+/* { dg-do compile } */
+/* { dg-options "-O2 -fdump-tree-optimized" } */
+/* { dg-final { scan-tree-dump-not "link_error \\\(" "optimized" } } */
+
+void link_error (void);
+
+void
+f1 (signed char i)
+{
+  if (__builtin_abs (i) < 0 || __builtin_abs (i) > __SCHAR_MAX__ + 1)
+    link_error ();
+}
+
+void
+f2 (signed char i)
+{
+  if (i < 0 || i > 15)
+    __builtin_unreachable ();
+  if (__builtin_abs (i) < 0 || __builtin_abs (i) > 15)
+    link_error ();
+}
+
+void
+f3 (signed char i)
+{
+  if (i < 19 || i > 25)
+    __builtin_unreachable ();
+  if (__builtin_abs (i) < 19 || __builtin_abs (i) > 25)
+    link_error ();
+}
+
+void
+f4 (signed char i)
+{
+  if (i > -60)
+    __builtin_unreachable ();
+  if (__builtin_abs (i) < 60 || __builtin_abs (i) > __SCHAR_MAX__ + 1)
+    link_error ();
+}
+
+void
+f5 (signed char i)
+{
+  if (i < -__SCHAR_MAX__ || i > -30)
+    __builtin_unreachable ();
+  if (__builtin_abs (i) < 30 || __builtin_abs (i) > __SCHAR_MAX__)
+    link_error ();
+}
+
+void
+f6 (signed char i)
+{
+  if (i < -__SCHAR_MAX__ || i > 30)
+    __builtin_unreachable ();
+  if (__builtin_abs (i) < 0 || __builtin_abs (i) > __SCHAR_MAX__)
+    link_error ();
+}
+
+void
+f7 (signed char i)
+{
+  if (i < -31 || i > 30)
+    __builtin_unreachable ();
+  if (__builtin_abs (i) < 0 || __builtin_abs (i) > 31)
+    link_error ();
+}
index 8d18e19d6e4b3b3ec61105cad362ebee9e880c64..bde0751ec1e6cd19ea8b52702f86ce9ec3a641f1 100644 (file)
@@ -2196,6 +2196,16 @@ extract_range_from_unary_expr (value_range_base *vr,
        vr->set_varying ();
       return;
     }
+  else if (code == ABSU_EXPR)
+    {
+      wide_int wmin, wmax;
+      wide_int vr0_min, vr0_max;
+      extract_range_into_wide_ints (&vr0, SIGNED, prec, vr0_min, vr0_max);
+      wide_int_range_absu (wmin, wmax, prec, vr0_min, vr0_max);
+      vr->set (VR_RANGE, wide_int_to_tree (type, wmin),
+              wide_int_to_tree (type, wmax));
+      return;
+    }
 
   /* For unhandled operations fall back to varying.  */
   vr->set_varying ();
index f18cb29447f947e4eb91dceb1c9ed311ed37ed05..90c58f6bb6ecda09a7d86b103d53b1ef62be6d84 100644 (file)
@@ -735,6 +735,37 @@ wide_int_range_abs (wide_int &min, wide_int &max,
   return true;
 }
 
+/* Calculate ABSU_EXPR on a range and store the result in [MIN, MAX].  */
+
+void
+wide_int_range_absu (wide_int &min, wide_int &max,
+                    unsigned prec, const wide_int &vr0_min,
+                    const wide_int &vr0_max)
+{
+  /* Pass through VR0 the easy cases.  */
+  if (wi::ges_p (vr0_min, 0))
+    {
+      min = vr0_min;
+      max = vr0_max;
+      return;
+    }
+
+  min = wi::abs (vr0_min);
+  max = wi::abs (vr0_max);
+
+  /* If the range contains zero then we know that the minimum value in the
+     range will be zero.  */
+  if (wi::ges_p (vr0_max, 0))
+    {
+      if (wi::gtu_p (min, max))
+       max = min;
+      min = wi::zero (prec);
+    }
+  else
+    /* Otherwise, swap MIN and MAX.  */
+    std::swap (min, max);
+}
+
 /* Convert range in [VR0_MIN, VR0_MAX] with INNER_SIGN and INNER_PREC,
    to a range in [MIN, MAX] with OUTER_SIGN and OUTER_PREC.
 
index 7fbdffd84b311bf75b0f71fca71a7ff1b326002b..fc9af72b12732432ef4fd46b993c061f567704d7 100644 (file)
@@ -107,6 +107,10 @@ extern bool wide_int_range_abs (wide_int &min, wide_int &max,
                                const wide_int &vr0_min,
                                const wide_int &vr0_max,
                                bool overflow_undefined);
+extern void wide_int_range_absu (wide_int &min, wide_int &max,
+                                unsigned prec,
+                                const wide_int &vr0_min,
+                                const wide_int &vr0_max);
 extern bool wide_int_range_convert (wide_int &min, wide_int &max,
                                    signop inner_sign,
                                    unsigned inner_prec,