diff --git a/llvm/lib/CodeGen/AllocationOrder.h b/llvm/lib/CodeGen/AllocationOrder.h --- a/llvm/lib/CodeGen/AllocationOrder.h +++ b/llvm/lib/CodeGen/AllocationOrder.h @@ -17,8 +17,8 @@ #define LLVM_LIB_CODEGEN_ALLOCATIONORDER_H #include "llvm/ADT/ArrayRef.h" -#include "llvm/ADT/SmallVector.h" #include "llvm/ADT/STLExtras.h" +#include "llvm/ADT/SmallVector.h" #include "llvm/MC/MCRegister.h" namespace llvm { @@ -30,12 +30,52 @@ class LLVM_LIBRARY_VISIBILITY AllocationOrder { const SmallVector Hints; ArrayRef Order; - int Pos = 0; - - // If HardHints is true, *only* Hints will be returned. - const bool HardHints; + // How far into the Order we can iterate. This is 0 if the AllocationOrder is + // constructed with HardHints = true, Order.size() otherwise. While + // technically a size_t, it will participate in comparisons with the + // Iterator's Pos, which must be signed, so it's typed here as signed, too, to + // avoid warnings and under the assumption that the size of Order is + // relatively small. + // IterationLimit defines an invalid iterator position. + const int IterationLimit; public: + /// Forward iterator for an AllocationOrder. + class Iterator final { + const AllocationOrder &AO; + int Pos = 0; + + public: + Iterator(const AllocationOrder &AO, int Pos) : AO(AO), Pos(Pos) {} + + /// Return true if the curent position is that of a preferred register. + bool isHint() const { return Pos < 0; } + + /// Return the next physical register in the allocation order. + unsigned operator*() const { + if (Pos < 0) + return AO.Hints.end()[Pos]; + assert(Pos < AO.IterationLimit); + return AO.Order[Pos]; + } + + /// Advance the iterator to the next position. If that's past the Hints + /// list, advance to the first value that's not also in the Hints list. + Iterator &operator++() { + if (Pos < AO.IterationLimit) + ++Pos; + while (Pos >= 0 && Pos < AO.IterationLimit && AO.isHint(AO.Order[Pos])) + ++Pos; + return *this; + } + + bool operator==(const Iterator &Other) const { + assert(&AO == &Other.AO); + return Pos == Other.Pos; + } + + bool operator!=(const Iterator &Other) const { return !(*this == Other); } + }; /// Create a new AllocationOrder for VirtReg. /// @param VirtReg Virtual register to allocate for. @@ -50,34 +90,25 @@ AllocationOrder(SmallVector &&Hints, ArrayRef Order, bool HardHints) : Hints(std::move(Hints)), Order(Order), - Pos(-static_cast(this->Hints.size())), HardHints(HardHints) {} + IterationLimit(HardHints ? 0 : static_cast(Order.size())) {} - /// Get the allocation order without reordered hints. - ArrayRef getOrder() const { return Order; } - - /// Return the next physical register in the allocation order, or 0. - /// It is safe to call next() again after it returned 0, it will keep - /// returning 0 until rewind() is called. - MCPhysReg next(unsigned Limit = 0) { - if (Pos < 0) - return Hints.end()[Pos++]; - if (HardHints) - return 0; - if (!Limit) - Limit = Order.size(); - while (Pos < int(Limit)) { - unsigned Reg = Order[Pos++]; - if (!isHint(Reg)) - return Reg; - } - return 0; + Iterator begin() const { + return Iterator(*this, -(static_cast(Hints.size()))); } - /// Start over from the beginning. - void rewind() { Pos = -int(Hints.size()); } + Iterator end() const { return Iterator(*this, IterationLimit); } - /// Return true if the last register returned from next() was a preferred register. - bool isHint() const { return Pos <= 0; } + Iterator getOrderLimitEnd(unsigned OrderLimit) const { + assert(OrderLimit <= Order.size()); + if (OrderLimit == 0) + return end(); + Iterator Ret(*this, + std::min(static_cast(OrderLimit) - 1, IterationLimit)); + return ++Ret; + } + + /// Get the allocation order without reordered hints. + ArrayRef getOrder() const { return Order; } /// Return true if PhysReg is a preferred register. bool isHint(unsigned PhysReg) const { return is_contained(Hints, PhysReg); } diff --git a/llvm/lib/CodeGen/RegAllocBasic.cpp b/llvm/lib/CodeGen/RegAllocBasic.cpp --- a/llvm/lib/CodeGen/RegAllocBasic.cpp +++ b/llvm/lib/CodeGen/RegAllocBasic.cpp @@ -261,7 +261,8 @@ // Check for an available register in this class. auto Order = AllocationOrder::create(VirtReg.reg(), *VRM, RegClassInfo, Matrix); - while (Register PhysReg = Order.next()) { + for (Register PhysReg : Order) { + assert(PhysReg); // Check for interference in PhysReg switch (Matrix->checkInterference(VirtReg, PhysReg)) { case LiveRegMatrix::IK_Free: diff --git a/llvm/lib/CodeGen/RegAllocGreedy.cpp b/llvm/lib/CodeGen/RegAllocGreedy.cpp --- a/llvm/lib/CodeGen/RegAllocGreedy.cpp +++ b/llvm/lib/CodeGen/RegAllocGreedy.cpp @@ -756,12 +756,17 @@ AllocationOrder &Order, SmallVectorImpl &NewVRegs, const SmallVirtRegSet &FixedRegisters) { - Order.rewind(); Register PhysReg; - while ((PhysReg = Order.next())) - if (!Matrix->checkInterference(VirtReg, PhysReg)) - break; - if (!PhysReg || Order.isHint()) + for (auto I = Order.begin(), E = Order.end(); I != E && !PhysReg; ++I) { + assert(*I); + if (!Matrix->checkInterference(VirtReg, *I)) { + if (I.isHint()) + return *I; + else + PhysReg = *I; + } + } + if (!PhysReg) return PhysReg; // PhysReg is available, but there may be a better choice. @@ -803,11 +808,11 @@ auto Order = AllocationOrder::create(VirtReg.reg(), *VRM, RegClassInfo, Matrix); Register PhysReg; - while ((PhysReg = Order.next())) { - if (PhysReg == PrevReg) + for (auto I = Order.begin(), E = Order.end(); I != E && !PhysReg; ++I) { + if (*I == PrevReg) continue; - MCRegUnitIterator Units(PhysReg, TRI); + MCRegUnitIterator Units(*I, TRI); for (; Units.isValid(); ++Units) { // Instantiate a "subquery", not to be confused with the Queries array. LiveIntervalUnion::Query subQ(VirtReg, Matrix->getLiveUnions()[*Units]); @@ -816,7 +821,7 @@ } // If no units have interference, break out with the current PhysReg. if (!Units.isValid()) - break; + PhysReg = *I; } if (PhysReg) LLVM_DEBUG(dbgs() << "can reassign: " << VirtReg << " from " @@ -1133,8 +1138,10 @@ } } - Order.rewind(); - while (MCRegister PhysReg = Order.next(OrderLimit)) { + for (auto I = Order.begin(), E = Order.getOrderLimitEnd(OrderLimit); I != E; + ++I) { + MCRegister PhysReg = *I; + assert(PhysReg); if (TRI->getCostPerUse(PhysReg) >= CostPerUseLimit) continue; // The first use of a callee-saved register in a function has cost 1. @@ -1155,7 +1162,7 @@ BestPhys = PhysReg; // Stop if the hint can be used. - if (Order.isHint()) + if (I.isHint()) break; } @@ -1850,8 +1857,8 @@ unsigned &NumCands, bool IgnoreCSR, bool *CanCauseEvictionChain) { unsigned BestCand = NoCand; - Order.rewind(); - while (unsigned PhysReg = Order.next()) { + for (auto PhysReg : Order) { + assert(PhysReg); if (IgnoreCSR && isUnusedCalleeSavedReg(PhysReg)) continue; @@ -2289,8 +2296,8 @@ (1.0f / MBFI->getEntryFreq()); SmallVector GapWeight; - Order.rewind(); - while (unsigned PhysReg = Order.next()) { + for (auto PhysReg : Order) { + assert(PhysReg); // Keep track of the largest spill weight that would need to be evicted in // order to make use of PhysReg between UseSlots[I] and UseSlots[I + 1]. calcGapWeights(PhysReg, GapWeight); @@ -2607,8 +2614,8 @@ FixedRegisters.insert(VirtReg.reg()); SmallVector CurrentNewVRegs; - Order.rewind(); - while (Register PhysReg = Order.next()) { + for (auto PhysReg : Order) { + assert(PhysReg); LLVM_DEBUG(dbgs() << "Try to assign: " << VirtReg << " to " << printReg(PhysReg, TRI) << '\n'); RecoloringCandidates.clear(); diff --git a/llvm/unittests/CodeGen/AllocationOrderTest.cpp b/llvm/unittests/CodeGen/AllocationOrderTest.cpp --- a/llvm/unittests/CodeGen/AllocationOrderTest.cpp +++ b/llvm/unittests/CodeGen/AllocationOrderTest.cpp @@ -12,11 +12,14 @@ using namespace llvm; namespace { -std::vector loadOrder(AllocationOrder &O, unsigned Limit = 0) { +std::vector loadOrder(const AllocationOrder &O, unsigned Limit = 0) { std::vector Ret; - O.rewind(); - while (auto R = O.next(Limit)) - Ret.push_back(R); + if (Limit == 0) + for (auto R : O) + Ret.push_back(R); + else + for (auto I = O.begin(), E = O.getOrderLimitEnd(Limit); I != E; ++I) + Ret.push_back(*I); return Ret; } } // namespace @@ -48,6 +51,7 @@ AllocationOrder O(std::move(Hints), Order, false); EXPECT_EQ((std::vector{1, 2, 3, 4, 5, 6, 7}), loadOrder(O, 0)); EXPECT_EQ((std::vector{1, 2, 3, 4}), loadOrder(O, 1)); + EXPECT_EQ(O.end(), O.getOrderLimitEnd(0)); } TEST(AllocationOrderTest, LimitsDuplicates) { @@ -96,19 +100,19 @@ SmallVector Hints = {1, 2, 3}; SmallVector Order = {4, 1, 5, 6}; AllocationOrder O(std::move(Hints), Order, false); - O.rewind(); - auto V = O.next(); - EXPECT_TRUE(O.isHint()); + auto I = O.begin(); + auto V = *I; + EXPECT_TRUE(I.isHint()); EXPECT_EQ(V, 1U); - O.next(); - EXPECT_TRUE(O.isHint()); - O.next(); - EXPECT_TRUE(O.isHint()); - V = O.next(); - EXPECT_FALSE(O.isHint()); + ++I; + EXPECT_TRUE(I.isHint()); + ++I; + EXPECT_TRUE(I.isHint()); + V = *(++I); + EXPECT_FALSE(I.isHint()); EXPECT_EQ(V, 4U); - V = O.next(); + V = *(++I); EXPECT_TRUE(O.isHint(1)); - EXPECT_FALSE(O.isHint()); + EXPECT_FALSE(I.isHint()); EXPECT_EQ(V, 5U); }