aboutsummaryrefslogtreecommitdiff
path: root/gcc/range-op.cc
diff options
context:
space:
mode:
authorAndrew MacLeod <amacleod@redhat.com>2023-11-28 09:39:30 -0500
committerAndrew MacLeod <amacleod@redhat.com>2023-11-29 11:43:53 -0500
commitea19de921b01a6ab470929f8da4dde526edb08f1 (patch)
tree5088e0a596202dc03e252f4956c009488989a119 /gcc/range-op.cc
parent302461ad9a04d82fee904bddac69811d13d5bb6a (diff)
downloadgcc-ea19de921b01a6ab470929f8da4dde526edb08f1.zip
gcc-ea19de921b01a6ab470929f8da4dde526edb08f1.tar.gz
gcc-ea19de921b01a6ab470929f8da4dde526edb08f1.tar.bz2
Add operand_check_p to range-ops.
Add an optional method to verify operands are compatible, and check the operands before all range operations. * range-op-mixed.h (operator_equal::operand_check_p): New. (operator_not_equal::operand_check_p): New. (operator_lt::operand_check_p): New. (operator_le::operand_check_p): New. (operator_gt::operand_check_p): New. (operator_ge::operand_check_p): New. (operator_plus::operand_check_p): New. (operator_abs::operand_check_p): New. (operator_minus::operand_check_p): New. (operator_negate::operand_check_p): New. (operator_mult::operand_check_p): New. (operator_bitwise_not::operand_check_p): New. (operator_bitwise_xor::operand_check_p): New. (operator_bitwise_and::operand_check_p): New. (operator_bitwise_or::operand_check_p): New. (operator_min::operand_check_p): New. (operator_max::operand_check_p): New. * range-op.cc (range_op_handler::fold_range): Check operand parameter types. (range_op_handler::op1_range): Ditto. (range_op_handler::op2_range): Ditto. (range_op_handler::operand_check_p): New. (range_operator::operand_check_p): New. (operator_lshift::operand_check_p): New. (operator_rshift::operand_check_p): New. (operator_logical_and::operand_check_p): New. (operator_logical_or::operand_check_p): New. (operator_logical_not::operand_check_p): New. * range-op.h (range_operator::operand_check_p): New. (range_op_handler::operand_check_p): New.
Diffstat (limited to 'gcc/range-op.cc')
-rw-r--r--gcc/range-op.cc53
1 files changed, 49 insertions, 4 deletions
diff --git a/gcc/range-op.cc b/gcc/range-op.cc
index 6137f2a..a091815 100644
--- a/gcc/range-op.cc
+++ b/gcc/range-op.cc
@@ -201,6 +201,10 @@ range_op_handler::fold_range (vrange &r, tree type,
relation_trio rel) const
{
gcc_checking_assert (m_operator);
+#if CHECKING_P
+ if (!lh.undefined_p () && !rh.undefined_p ())
+ gcc_assert (m_operator->operand_check_p (type, lh.type (), rh.type ()));
+#endif
switch (dispatch_kind (r, lh, rh))
{
case RO_III:
@@ -237,9 +241,12 @@ range_op_handler::op1_range (vrange &r, tree type,
relation_trio rel) const
{
gcc_checking_assert (m_operator);
-
if (lhs.undefined_p ())
return false;
+#if CHECKING_P
+ if (!op2.undefined_p ())
+ gcc_assert (m_operator->operand_check_p (lhs.type (), type, op2.type ()));
+#endif
switch (dispatch_kind (r, lhs, op2))
{
case RO_III:
@@ -270,7 +277,10 @@ range_op_handler::op2_range (vrange &r, tree type,
gcc_checking_assert (m_operator);
if (lhs.undefined_p ())
return false;
-
+#if CHECKING_P
+ if (!op1.undefined_p ())
+ gcc_assert (m_operator->operand_check_p (lhs.type (), op1.type (), type));
+#endif
switch (dispatch_kind (r, lhs, op1))
{
case RO_III:
@@ -394,6 +404,13 @@ range_op_handler::overflow_free_p (const vrange &lh,
}
}
+bool
+range_op_handler::operand_check_p (tree t1, tree t2, tree t3) const
+{
+ gcc_checking_assert (m_operator);
+ return m_operator->operand_check_p (t1, t2, t3);
+}
+
// Update the known bitmasks in R when applying the operation CODE to
// LH and RH.
@@ -737,6 +754,14 @@ range_operator::update_bitmask (irange &, const irange &,
{
}
+// Check that operand types are OK. Default to always OK.
+
+bool
+range_operator::operand_check_p (tree, tree, tree) const
+{
+ return true;
+}
+
// Create and return a range from a pair of wide-ints that are known
// to have overflowed (or underflowed).
@@ -2466,6 +2491,9 @@ public:
void update_bitmask (irange &r, const irange &lh,
const irange &rh) const final override
{ update_known_bitmask (r, LSHIFT_EXPR, lh, rh); }
+ // Check compatibility of LHS and op1.
+ bool operand_check_p (tree t1, tree t2, tree) const final override
+ { return TYPE_PRECISION (t1) == TYPE_PRECISION (t2); }
} op_lshift;
class operator_rshift : public cross_product_operator
@@ -2495,6 +2523,9 @@ public:
void update_bitmask (irange &r, const irange &lh,
const irange &rh) const final override
{ update_known_bitmask (r, RSHIFT_EXPR, lh, rh); }
+ // Check compatibility of LHS and op1.
+ bool operand_check_p (tree t1, tree t2, tree) const final override
+ { return TYPE_PRECISION (t1) == TYPE_PRECISION (t2); }
} op_rshift;
@@ -3070,9 +3101,12 @@ public:
const irange &lhs,
const irange &op1,
relation_trio rel = TRIO_VARYING) const;
+ // Check compatibility of all operands.
+ bool operand_check_p (tree t1, tree t2, tree t3) const final override
+ { return (TYPE_PRECISION (t1) == TYPE_PRECISION (t2)
+ && TYPE_PRECISION (t1) == TYPE_PRECISION (t3)); }
} op_logical_and;
-
bool
operator_logical_and::fold_range (irange &r, tree type,
const irange &lh,
@@ -3082,6 +3116,11 @@ operator_logical_and::fold_range (irange &r, tree type,
if (empty_range_varying (r, type, lh, rh))
return true;
+ // Precision of LHS and both operands must match.
+ if (TYPE_PRECISION (lh.type ()) != TYPE_PRECISION (type)
+ || TYPE_PRECISION (type) != TYPE_PRECISION (rh.type ()))
+ return false;
+
// 0 && anything is 0.
if ((wi::eq_p (lh.lower_bound (), 0) && wi::eq_p (lh.upper_bound (), 0))
|| (wi::eq_p (lh.lower_bound (), 0) && wi::eq_p (rh.upper_bound (), 0)))
@@ -3567,6 +3606,10 @@ public:
const irange &lhs,
const irange &op1,
relation_trio rel = TRIO_VARYING) const;
+ // Check compatibility of all operands.
+ bool operand_check_p (tree t1, tree t2, tree t3) const final override
+ { return (TYPE_PRECISION (t1) == TYPE_PRECISION (t2)
+ && TYPE_PRECISION (t1) == TYPE_PRECISION (t3)); }
} op_logical_or;
bool
@@ -3993,6 +4036,9 @@ public:
const irange &lhs,
const irange &op2,
relation_trio rel = TRIO_VARYING) const;
+ // Check compatibility of LHS and op1.
+ bool operand_check_p (tree t1, tree t2, tree) const final override
+ { return TYPE_PRECISION (t1) == TYPE_PRECISION (t2); }
} op_logical_not;
// Folding a logical NOT, oddly enough, involves doing nothing on the
@@ -4036,7 +4082,6 @@ operator_logical_not::op1_range (irange &r,
return fold_range (r, type, lhs, op2);
}
-
bool
operator_bitwise_not::fold_range (irange &r, tree type,
const irange &lh,