2023-02-03 Jakub Jelinek PR tree-optimization/108647 * range-op.cc (operator_equal::op1_range, operator_not_equal::op1_range): Don't test op2 bound equality if op2.undefined_p (), instead set_varying. (operator_lt::op1_range, operator_le::op1_range, operator_gt::op1_range, operator_ge::op1_range): Return false if op2.undefined_p (). (operator_lt::op2_range, operator_le::op2_range, operator_gt::op2_range, operator_ge::op2_range): Return false if op1.undefined_p (). * g++.dg/torture/pr108647.C: New test. --- gcc/range-op.cc.jj 2023-02-03 10:51:40.699003658 +0100 +++ gcc/range-op.cc 2023-02-03 19:43:28.082326069 +0100 @@ -642,7 +642,8 @@ operator_equal::op1_range (irange &r, tr case BRS_FALSE: // If the result is false, the only time we know anything is // if OP2 is a constant. - if (wi::eq_p (op2.lower_bound(), op2.upper_bound())) + if (!op2.undefined_p () + && wi::eq_p (op2.lower_bound(), op2.upper_bound())) { r = op2; r.invert (); @@ -755,7 +756,8 @@ operator_not_equal::op1_range (irange &r case BRS_TRUE: // If the result is true, the only time we know anything is if // OP2 is a constant. - if (wi::eq_p (op2.lower_bound(), op2.upper_bound())) + if (!op2.undefined_p () + && wi::eq_p (op2.lower_bound(), op2.upper_bound())) { r = op2; r.invert (); @@ -923,10 +925,14 @@ operator_lt::op1_range (irange &r, tree switch (get_bool_state (r, lhs, type)) { case BRS_TRUE: + if (op2.undefined_p ()) + return false; build_lt (r, type, op2.upper_bound ()); break; case BRS_FALSE: + if (op2.undefined_p ()) + return false; build_ge (r, type, op2.lower_bound ()); break; @@ -945,10 +951,14 @@ operator_lt::op2_range (irange &r, tree switch (get_bool_state (r, lhs, type)) { case BRS_TRUE: + if (op1.undefined_p ()) + return false; build_gt (r, type, op1.lower_bound ()); break; case BRS_FALSE: + if (op1.undefined_p ()) + return false; build_le (r, type, op1.upper_bound ()); break; @@ -1034,10 +1044,14 @@ operator_le::op1_range (irange &r, tree switch (get_bool_state (r, lhs, type)) { case BRS_TRUE: + if (op2.undefined_p ()) + return false; build_le (r, type, op2.upper_bound ()); break; case BRS_FALSE: + if (op2.undefined_p ()) + return false; build_gt (r, type, op2.lower_bound ()); break; @@ -1056,10 +1070,14 @@ operator_le::op2_range (irange &r, tree switch (get_bool_state (r, lhs, type)) { case BRS_TRUE: + if (op1.undefined_p ()) + return false; build_ge (r, type, op1.lower_bound ()); break; case BRS_FALSE: + if (op1.undefined_p ()) + return false; build_lt (r, type, op1.upper_bound ()); break; @@ -1144,10 +1162,14 @@ operator_gt::op1_range (irange &r, tree switch (get_bool_state (r, lhs, type)) { case BRS_TRUE: + if (op2.undefined_p ()) + return false; build_gt (r, type, op2.lower_bound ()); break; case BRS_FALSE: + if (op2.undefined_p ()) + return false; build_le (r, type, op2.upper_bound ()); break; @@ -1166,10 +1188,14 @@ operator_gt::op2_range (irange &r, tree switch (get_bool_state (r, lhs, type)) { case BRS_TRUE: + if (op1.undefined_p ()) + return false; build_lt (r, type, op1.upper_bound ()); break; case BRS_FALSE: + if (op1.undefined_p ()) + return false; build_ge (r, type, op1.lower_bound ()); break; @@ -1255,10 +1281,14 @@ operator_ge::op1_range (irange &r, tree switch (get_bool_state (r, lhs, type)) { case BRS_TRUE: + if (op2.undefined_p ()) + return false; build_ge (r, type, op2.lower_bound ()); break; case BRS_FALSE: + if (op2.undefined_p ()) + return false; build_lt (r, type, op2.upper_bound ()); break; @@ -1277,10 +1307,14 @@ operator_ge::op2_range (irange &r, tree switch (get_bool_state (r, lhs, type)) { case BRS_TRUE: + if (op1.undefined_p ()) + return false; build_le (r, type, op1.upper_bound ()); break; case BRS_FALSE: + if (op1.undefined_p ()) + return false; build_gt (r, type, op1.lower_bound ()); break; --- gcc/testsuite/g++.dg/torture/pr108647.C.jj 2023-02-03 16:36:18.347255058 +0100 +++ gcc/testsuite/g++.dg/torture/pr108647.C 2023-02-03 16:32:16.338811259 +0100 @@ -0,0 +1,25 @@ +// PR tree-optimization/108647 +// { dg-do compile } + +bool a; +int b, c; + +inline const bool & +foo (bool &e, const bool &f) +{ + return f < e ? f : e; +} + +void +bar (signed char e, bool *f, bool *h, bool *g) +{ + for (;;) + if (g) + for (signed char j = 0; j < 6; + j += ((f[0] & c ? g[0] : int(0 >= e)) + ? 0 : foo (g[0], g[0] > h[0]) + 1)) + { + a = 0; + b = 0; + } +}