m_base[x - 1] = src.m_base[src.m_num_ranges * 2 - 1];
m_num_ranges = lim;
+ m_kind = src.m_kind;
return *this;
}
irange::copy_to_legacy (const irange &src)
{
gcc_checking_assert (legacy_mode_p ());
- // Copy legacy to legacy.
- if (src.legacy_mode_p ())
+ // Handle legacy to legacy and other things that are easy to copy.
+ if (src.legacy_mode_p () || src.varying_p () || src.undefined_p ())
{
m_num_ranges = src.m_num_ranges;
m_base[0] = src.m_base[0];
return;
}
// Copy multi-range to legacy.
- if (src.undefined_p ())
- set_undefined ();
- else if (src.varying_p ())
- set_varying (src.type ());
- else if (src.maybe_anti_range ())
+ if (src.maybe_anti_range ())
{
int_range<3> r (src);
r.invert ();
m_base[0] = min;
m_base[1] = max;
m_num_ranges = 1;
+ m_kind = VR_RANGE;
+ normalize_kind ();
+
if (flag_checking)
verify_range ();
}
m_base[m_num_ranges * 2 + 1] = type_range.tree_upper_bound (0);
++m_num_ranges;
}
+
+ m_kind = VR_RANGE;
+ normalize_kind ();
+
if (flag_checking)
verify_range ();
}
m_base[0] = min;
m_base[1] = max;
m_num_ranges = 1;
- normalize_min_max ();
+ normalize_kind ();
if (flag_checking)
verify_range ();
}
void
irange::verify_range ()
{
+ if (m_kind == VR_UNDEFINED)
+ {
+ gcc_assert (m_num_ranges == 0);
+ return;
+ }
+ gcc_assert (m_num_ranges != 0);
+
+ if (m_kind == VR_VARYING)
+ {
+ gcc_checking_assert (m_num_ranges == 1);
+ gcc_checking_assert (varying_compatible_p ());
+ return;
+ }
if (!legacy_mode_p ())
{
- gcc_checking_assert (m_kind == VR_RANGE);
+ gcc_checking_assert (!varying_compatible_p ());
for (unsigned i = 0; i < m_num_ranges; ++i)
{
tree lb = tree_lower_bound (i);
}
return;
}
-
- switch (m_kind)
+ if (m_kind == VR_RANGE || m_kind == VR_ANTI_RANGE)
{
- case VR_UNDEFINED:
- gcc_assert (m_num_ranges == 0);
- break;
-
- case VR_VARYING:
gcc_assert (m_num_ranges == 1);
- break;
-
- case VR_ANTI_RANGE:
- case VR_RANGE:
- {
- gcc_assert (m_num_ranges == 1);
- int cmp = compare_values (tree_lower_bound (0), tree_upper_bound (0));
- gcc_assert (cmp == 0 || cmp == -1 || cmp == -2);
- return;
- }
-
- default:
- gcc_unreachable ();
+ int cmp = compare_values (tree_lower_bound (0), tree_upper_bound (0));
+ gcc_assert (cmp == 0 || cmp == -1 || cmp == -2);
}
}
m_base[j] = res [j];
m_num_ranges = i / 2;
+ m_kind = VR_RANGE;
+ normalize_kind ();
+
if (flag_checking)
verify_range ();
}
// At the exit of this loop, it is one of 2 things:
// ran out of r1, or r2, but either means we are done.
m_num_ranges = bld_pair;
+
+ m_kind = VR_RANGE;
+ normalize_kind ();
+
if (flag_checking)
verify_range ();
}
}
m_num_ranges = nitems / 2;
+ // We disallow undefined or varying coming in, so the result can
+ // only be a VR_RANGE.
+ gcc_checking_assert (m_kind == VR_RANGE);
+
if (flag_checking)
verify_range ();
}
void irange_set (tree, tree);
void irange_set_anti_range (tree, tree);
- void normalize_min_max ();
+ void normalize_kind ();
bool legacy_mode_p () const;
bool legacy_equal_p (const irange &) const;
private:
void irange_set_1bit_anti_range (tree, tree);
+ bool varying_compatible_p () const;
unsigned char m_num_ranges;
unsigned char m_max_ranges;
inline value_range_kind
irange::kind () const
{
- if (legacy_mode_p ())
- return m_kind;
-
- if (undefined_p ())
- return VR_UNDEFINED;
-
- if (varying_p ())
- return VR_VARYING;
-
- return VR_RANGE;
+ return m_kind;
}
// Number of sub-ranges in a range.
}
inline bool
-irange::varying_p () const
+irange::varying_compatible_p () const
{
- if (legacy_mode_p ())
- return m_kind == VR_VARYING;
-
if (m_num_ranges != 1)
return false;
tree l = m_base[0];
tree u = m_base[1];
tree t = TREE_TYPE (l);
+
+ if (m_kind == VR_VARYING && t == error_mark_node)
+ return true;
+
unsigned prec = TYPE_PRECISION (t);
signop sign = TYPE_SIGN (t);
if (INTEGRAL_TYPE_P (t))
return (wi::to_wide (l) == 0
&& wi::to_wide (u) == wi::max_value (prec, sign));
return true;
+}
+inline bool
+irange::varying_p () const
+{
+ return m_kind == VR_VARYING;
}
inline bool
irange::undefined_p () const
{
- if (!legacy_mode_p ())
- return m_num_ranges == 0;
-
- if (CHECKING_P && legacy_mode_p ())
- {
- if (m_kind == VR_UNDEFINED)
- gcc_checking_assert (m_num_ranges == 0);
- else
- gcc_checking_assert (m_num_ranges != 0);
- }
return m_kind == VR_UNDEFINED;
}
m_base = base;
m_num_ranges = 0;
m_max_ranges = nranges;
- if (legacy_mode_p ())
- m_kind = VR_UNDEFINED;
- else
- m_kind = VR_RANGE;
+ m_kind = VR_UNDEFINED;
}
// Constructors for int_range<>.
inline void
irange::set_undefined ()
{
+ m_kind = VR_UNDEFINED;
m_num_ranges = 0;
- if (legacy_mode_p ())
- m_kind = VR_UNDEFINED;
}
inline void
irange::set_varying (tree type)
{
- if (legacy_mode_p ())
- m_kind = VR_VARYING;
-
+ m_kind = VR_VARYING;
m_num_ranges = 1;
+
if (INTEGRAL_TYPE_P (type))
{
wide_int min = wi::min_value (TYPE_PRECISION (type), TYPE_SIGN (type));
// Normalize a range to VARYING or UNDEFINED if possible.
inline void
-irange::normalize_min_max ()
-{
- gcc_checking_assert (legacy_mode_p ());
- gcc_checking_assert (!undefined_p ());
- unsigned prec = TYPE_PRECISION (type ());
- signop sign = TYPE_SIGN (type ());
- if (wi::eq_p (wi::to_wide (min ()), wi::min_value (prec, sign))
- && wi::eq_p (wi::to_wide (max ()), wi::max_value (prec, sign)))
+irange::normalize_kind ()
+{
+ if (m_num_ranges == 0)
+ m_kind = VR_UNDEFINED;
+ else if (varying_compatible_p ())
{
if (m_kind == VR_RANGE)
- set_varying (type ());
+ m_kind = VR_VARYING;
else if (m_kind == VR_ANTI_RANGE)
set_undefined ();
else