| 1 | //===- ConstantRangeList.cpp - ConstantRangeList implementation -----------===// |
| 2 | // |
| 3 | // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. |
| 4 | // See https://llvm.org/LICENSE.txt for license information. |
| 5 | // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception |
| 6 | // |
| 7 | //===----------------------------------------------------------------------===// |
| 8 | |
| 9 | #include "llvm/IR/ConstantRangeList.h" |
| 10 | #include <cstddef> |
| 11 | |
| 12 | using namespace llvm; |
| 13 | |
| 14 | bool ConstantRangeList::isOrderedRanges(ArrayRef<ConstantRange> RangesRef) { |
| 15 | if (RangesRef.empty()) |
| 16 | return true; |
| 17 | auto Range = RangesRef[0]; |
| 18 | if (Range.getLower().sge(RHS: Range.getUpper())) |
| 19 | return false; |
| 20 | for (unsigned i = 1; i < RangesRef.size(); i++) { |
| 21 | auto CurRange = RangesRef[i]; |
| 22 | auto PreRange = RangesRef[i - 1]; |
| 23 | if (CurRange.getLower().sge(RHS: CurRange.getUpper()) || |
| 24 | CurRange.getLower().sle(RHS: PreRange.getUpper())) |
| 25 | return false; |
| 26 | } |
| 27 | return true; |
| 28 | } |
| 29 | |
| 30 | std::optional<ConstantRangeList> |
| 31 | ConstantRangeList::getConstantRangeList(ArrayRef<ConstantRange> RangesRef) { |
| 32 | if (!isOrderedRanges(RangesRef)) |
| 33 | return std::nullopt; |
| 34 | return ConstantRangeList(RangesRef); |
| 35 | } |
| 36 | |
| 37 | void ConstantRangeList::insert(const ConstantRange &NewRange) { |
| 38 | if (NewRange.isEmptySet()) |
| 39 | return; |
| 40 | assert(!NewRange.isFullSet() && "Do not support full set" ); |
| 41 | assert(NewRange.getLower().slt(NewRange.getUpper())); |
| 42 | // Handle common cases. |
| 43 | if (empty() || Ranges.back().getUpper().slt(RHS: NewRange.getLower())) { |
| 44 | Ranges.push_back(Elt: NewRange); |
| 45 | return; |
| 46 | } |
| 47 | |
| 48 | assert(getBitWidth() == NewRange.getBitWidth()); |
| 49 | |
| 50 | if (NewRange.getUpper().slt(RHS: Ranges.front().getLower())) { |
| 51 | Ranges.insert(I: Ranges.begin(), Elt: NewRange); |
| 52 | return; |
| 53 | } |
| 54 | |
| 55 | auto LowerBound = lower_bound( |
| 56 | Range&: Ranges, Value: NewRange, C: [](const ConstantRange &a, const ConstantRange &b) { |
| 57 | return a.getLower().slt(RHS: b.getLower()); |
| 58 | }); |
| 59 | if (LowerBound != Ranges.end() && LowerBound->contains(CR: NewRange)) |
| 60 | return; |
| 61 | |
| 62 | // Slow insert. |
| 63 | SmallVector<ConstantRange, 2> ExistingTail(LowerBound, Ranges.end()); |
| 64 | Ranges.erase(CS: LowerBound, CE: Ranges.end()); |
| 65 | // Merge consecutive ranges. |
| 66 | if (!Ranges.empty() && NewRange.getLower().sle(RHS: Ranges.back().getUpper())) { |
| 67 | APInt NewLower = Ranges.back().getLower(); |
| 68 | APInt NewUpper = |
| 69 | APIntOps::smax(A: NewRange.getUpper(), B: Ranges.back().getUpper()); |
| 70 | Ranges.back() = ConstantRange(NewLower, NewUpper); |
| 71 | } else { |
| 72 | Ranges.push_back(Elt: NewRange); |
| 73 | } |
| 74 | for (auto Iter = ExistingTail.begin(); Iter != ExistingTail.end(); Iter++) { |
| 75 | if (Ranges.back().getUpper().slt(RHS: Iter->getLower())) { |
| 76 | Ranges.push_back(Elt: *Iter); |
| 77 | } else { |
| 78 | APInt NewLower = Ranges.back().getLower(); |
| 79 | APInt NewUpper = |
| 80 | APIntOps::smax(A: Iter->getUpper(), B: Ranges.back().getUpper()); |
| 81 | Ranges.back() = ConstantRange(NewLower, NewUpper); |
| 82 | } |
| 83 | } |
| 84 | } |
| 85 | |
| 86 | void ConstantRangeList::subtract(const ConstantRange &SubRange) { |
| 87 | if (SubRange.isEmptySet() || empty()) |
| 88 | return; |
| 89 | assert(!SubRange.isFullSet() && "Do not support full set" ); |
| 90 | assert(SubRange.getLower().slt(SubRange.getUpper())); |
| 91 | assert(getBitWidth() == SubRange.getBitWidth()); |
| 92 | // Handle common cases. |
| 93 | if (Ranges.back().getUpper().sle(RHS: SubRange.getLower())) |
| 94 | return; |
| 95 | if (SubRange.getUpper().sle(RHS: Ranges.front().getLower())) |
| 96 | return; |
| 97 | |
| 98 | SmallVector<ConstantRange, 2> Result; |
| 99 | auto AppendRangeIfNonEmpty = [&Result](APInt Start, APInt End) { |
| 100 | if (Start.slt(RHS: End)) |
| 101 | Result.push_back(Elt: ConstantRange(Start, End)); |
| 102 | }; |
| 103 | for (auto &Range : Ranges) { |
| 104 | if (SubRange.getUpper().sle(RHS: Range.getLower()) || |
| 105 | Range.getUpper().sle(RHS: SubRange.getLower())) { |
| 106 | // "Range" and "SubRange" do not overlap. |
| 107 | // L---U : Range |
| 108 | // L---U : SubRange (Case1) |
| 109 | // L---U : SubRange (Case2) |
| 110 | Result.push_back(Elt: Range); |
| 111 | } else if (Range.getLower().sle(RHS: SubRange.getLower()) && |
| 112 | SubRange.getUpper().sle(RHS: Range.getUpper())) { |
| 113 | // "Range" contains "SubRange". |
| 114 | // L---U : Range |
| 115 | // L-U : SubRange |
| 116 | // Note that ConstantRange::contains(ConstantRange) checks unsigned, |
| 117 | // but we need signed checking here. |
| 118 | AppendRangeIfNonEmpty(Range.getLower(), SubRange.getLower()); |
| 119 | AppendRangeIfNonEmpty(SubRange.getUpper(), Range.getUpper()); |
| 120 | } else if (SubRange.getLower().sle(RHS: Range.getLower()) && |
| 121 | Range.getUpper().sle(RHS: SubRange.getUpper())) { |
| 122 | // "SubRange" contains "Range". |
| 123 | // L-U : Range |
| 124 | // L---U : SubRange |
| 125 | continue; |
| 126 | } else if (Range.getLower().sge(RHS: SubRange.getLower()) && |
| 127 | Range.getLower().sle(RHS: SubRange.getUpper())) { |
| 128 | // "Range" and "SubRange" overlap at the left. |
| 129 | // L---U : Range |
| 130 | // L---U : SubRange |
| 131 | AppendRangeIfNonEmpty(SubRange.getUpper(), Range.getUpper()); |
| 132 | } else { |
| 133 | // "Range" and "SubRange" overlap at the right. |
| 134 | // L---U : Range |
| 135 | // L---U : SubRange |
| 136 | assert(SubRange.getLower().sge(Range.getLower()) && |
| 137 | SubRange.getLower().sle(Range.getUpper())); |
| 138 | AppendRangeIfNonEmpty(Range.getLower(), SubRange.getLower()); |
| 139 | } |
| 140 | } |
| 141 | |
| 142 | Ranges = Result; |
| 143 | } |
| 144 | |
| 145 | ConstantRangeList |
| 146 | ConstantRangeList::unionWith(const ConstantRangeList &CRL) const { |
| 147 | // Handle common cases. |
| 148 | if (empty()) |
| 149 | return CRL; |
| 150 | if (CRL.empty()) |
| 151 | return *this; |
| 152 | |
| 153 | assert(getBitWidth() == CRL.getBitWidth() && |
| 154 | "ConstantRangeList bitwidths don't agree!" ); |
| 155 | |
| 156 | ConstantRangeList Result; |
| 157 | size_t i = 0, j = 0; |
| 158 | // "PreviousRange" tracks the lowest unioned range that is being processed. |
| 159 | // Its lower is fixed and the upper may be updated over iterations. |
| 160 | ConstantRange PreviousRange(getBitWidth(), false); |
| 161 | if (Ranges[i].getLower().slt(RHS: CRL.Ranges[j].getLower())) { |
| 162 | PreviousRange = Ranges[i++]; |
| 163 | } else { |
| 164 | PreviousRange = CRL.Ranges[j++]; |
| 165 | } |
| 166 | |
| 167 | // Try to union "PreviousRange" and "CR". If they are disjoint, push |
| 168 | // "PreviousRange" to the result and assign it to "CR", a new union range. |
| 169 | // Otherwise, update the upper of "PreviousRange" to cover "CR". Note that, |
| 170 | // the lower of "PreviousRange" is always less or equal the lower of "CR". |
| 171 | auto UnionAndUpdateRange = [&PreviousRange, |
| 172 | &Result](const ConstantRange &CR) { |
| 173 | if (PreviousRange.getUpper().slt(RHS: CR.getLower())) { |
| 174 | Result.Ranges.push_back(Elt: PreviousRange); |
| 175 | PreviousRange = CR; |
| 176 | } else { |
| 177 | PreviousRange = ConstantRange( |
| 178 | PreviousRange.getLower(), |
| 179 | APIntOps::smax(A: PreviousRange.getUpper(), B: CR.getUpper())); |
| 180 | } |
| 181 | }; |
| 182 | while (i < size() || j < CRL.size()) { |
| 183 | if (j == CRL.size() || |
| 184 | (i < size() && Ranges[i].getLower().slt(RHS: CRL.Ranges[j].getLower()))) { |
| 185 | // Merge PreviousRange with this. |
| 186 | UnionAndUpdateRange(Ranges[i++]); |
| 187 | } else { |
| 188 | // Merge PreviousRange with CRL. |
| 189 | UnionAndUpdateRange(CRL.Ranges[j++]); |
| 190 | } |
| 191 | } |
| 192 | Result.Ranges.push_back(Elt: PreviousRange); |
| 193 | return Result; |
| 194 | } |
| 195 | |
| 196 | ConstantRangeList |
| 197 | ConstantRangeList::intersectWith(const ConstantRangeList &CRL) const { |
| 198 | // Handle common cases. |
| 199 | if (empty()) |
| 200 | return *this; |
| 201 | if (CRL.empty()) |
| 202 | return CRL; |
| 203 | |
| 204 | assert(getBitWidth() == CRL.getBitWidth() && |
| 205 | "ConstantRangeList bitwidths don't agree!" ); |
| 206 | |
| 207 | ConstantRangeList Result; |
| 208 | size_t i = 0, j = 0; |
| 209 | while (i < size() && j < CRL.size()) { |
| 210 | auto &Range = this->Ranges[i]; |
| 211 | auto &OtherRange = CRL.Ranges[j]; |
| 212 | |
| 213 | // The intersection of two Ranges is (max(lowers), min(uppers)), and it's |
| 214 | // possible that max(lowers) > min(uppers) if they don't have intersection. |
| 215 | // Add the intersection to result only if it's non-empty. |
| 216 | // To keep simple, we don't call ConstantRange::intersectWith() as it |
| 217 | // considers the complex upper wrapped case and may result two ranges, |
| 218 | // like (2, 8) && (6, 4) = {(2, 4), (6, 8)}. |
| 219 | APInt Start = APIntOps::smax(A: Range.getLower(), B: OtherRange.getLower()); |
| 220 | APInt End = APIntOps::smin(A: Range.getUpper(), B: OtherRange.getUpper()); |
| 221 | if (Start.slt(RHS: End)) |
| 222 | Result.Ranges.push_back(Elt: ConstantRange(Start, End)); |
| 223 | |
| 224 | // Move to the next Range in one list determined by the uppers. |
| 225 | // For example: A = {(0, 2), (4, 8)}; B = {(-2, 5), (6, 10)} |
| 226 | // We need to intersect three pairs: A0 && B0; A1 && B0; A1 && B1. |
| 227 | if (Range.getUpper().slt(RHS: OtherRange.getUpper())) |
| 228 | i++; |
| 229 | else |
| 230 | j++; |
| 231 | } |
| 232 | return Result; |
| 233 | } |
| 234 | |
| 235 | void ConstantRangeList::print(raw_ostream &OS) const { |
| 236 | interleaveComma(c: Ranges, os&: OS, each_fn: [&](ConstantRange CR) { |
| 237 | OS << "(" << CR.getLower() << ", " << CR.getUpper() << ")" ; |
| 238 | }); |
| 239 | } |
| 240 | |
| 241 | #if !defined(NDEBUG) || defined(LLVM_ENABLE_DUMP) |
| 242 | LLVM_DUMP_METHOD void ConstantRangeList::dump() const { |
| 243 | print(dbgs()); |
| 244 | dbgs() << '\n'; |
| 245 | } |
| 246 | #endif |
| 247 | |