aboutsummaryrefslogtreecommitdiff
path: root/gcc/tree-vrp.c
diff options
context:
space:
mode:
authorEric Botcazou <ebotcazou@adacore.com>2017-04-19 19:27:09 +0000
committerJeff Law <law@gcc.gnu.org>2017-04-19 13:27:09 -0600
commitfe1e7d0e9b7479f13b84670ac3b56cf5244d877c (patch)
treee92d2d185b565e9dbbadd2eab3b245262ecb273f /gcc/tree-vrp.c
parent52744dd73c677f0fcbf80249af7b4163f72c3326 (diff)
downloadgcc-fe1e7d0e9b7479f13b84670ac3b56cf5244d877c.zip
gcc-fe1e7d0e9b7479f13b84670ac3b56cf5244d877c.tar.gz
gcc-fe1e7d0e9b7479f13b84670ac3b56cf5244d877c.tar.bz2
re PR tree-optimization/80426 (wrong manipulation of range based on INT_MIN)
PR tree-optimization/80426 * tree-vrp.c (extract_range_from_binary_expr_1): For an additive operation on symbolic operands, also compute the overflow for the invariant part when the operation degenerates into a negation. PR tree-optimization/80426 * gcc.c-torture/execute/20170419-1.c: New test. Co-Authored-By: Jakub Jelinek <jakub@redhat.com> From-SVN: r247007
Diffstat (limited to 'gcc/tree-vrp.c')
-rw-r--r--gcc/tree-vrp.c30
1 files changed, 28 insertions, 2 deletions
diff --git a/gcc/tree-vrp.c b/gcc/tree-vrp.c
index 6d802de..697cd88 100644
--- a/gcc/tree-vrp.c
+++ b/gcc/tree-vrp.c
@@ -2461,7 +2461,20 @@ extract_range_from_binary_expr_1 (value_range *vr,
else if (min_op0)
wmin = min_op0;
else if (min_op1)
- wmin = minus_p ? wi::neg (min_op1) : min_op1;
+ {
+ if (minus_p)
+ {
+ wmin = wi::neg (min_op1);
+
+ /* Check for overflow. */
+ if (sgn == SIGNED && wi::neg_p (min_op1) && wi::neg_p (wmin))
+ min_ovf = 1;
+ else if (sgn == UNSIGNED && wi::ne_p (min_op1, 0))
+ min_ovf = -1;
+ }
+ else
+ wmin = min_op1;
+ }
else
wmin = wi::shwi (0, prec);
@@ -2489,7 +2502,20 @@ extract_range_from_binary_expr_1 (value_range *vr,
else if (max_op0)
wmax = max_op0;
else if (max_op1)
- wmax = minus_p ? wi::neg (max_op1) : max_op1;
+ {
+ if (minus_p)
+ {
+ wmax = wi::neg (max_op1);
+
+ /* Check for overflow. */
+ if (sgn == SIGNED && wi::neg_p (max_op1) && wi::neg_p (wmax))
+ max_ovf = 1;
+ else if (sgn == UNSIGNED && wi::ne_p (max_op1, 0))
+ max_ovf = -1;
+ }
+ else
+ wmax = max_op1;
+ }
else
wmax = wi::shwi (0, prec);