RAGreedy: Keep track of allocated PhysRegs internally

Do not use MachineRegisterInfo::setPhysRegUsed()/isPhysRegUsed()
anymore. This bitset changes function-global state and is set by the
VirtRegRewriter anyway.
Simply use a bitvector private to RAGreedy.

Differential Revision: http://reviews.llvm.org/D10910

llvm-svn: 242169
This commit is contained in:
Matthias Braun 2015-07-14 17:38:17 +00:00
parent 4ba525b727
commit 953393a72c
4 changed files with 40 additions and 18 deletions

View File

@ -203,6 +203,11 @@ public:
assert(idx < Size && "idx out of bounds"); assert(idx < Size && "idx out of bounds");
return LIUs[idx]; return LIUs[idx];
} }
const LiveIntervalUnion& operator[](unsigned Idx) const {
assert(Idx < Size && "Idx out of bounds");
return LIUs[Idx];
}
}; };
}; };

View File

@ -114,6 +114,9 @@ public:
/// the assignment and updates VirtRegMap accordingly. /// the assignment and updates VirtRegMap accordingly.
void unassign(LiveInterval &VirtReg); void unassign(LiveInterval &VirtReg);
/// Returns true if the given \p PhysReg has any live intervals assigned.
bool isPhysRegUsed(unsigned PhysReg) const;
//===--------------------------------------------------------------------===// //===--------------------------------------------------------------------===//
// Low-level interface. // Low-level interface.
//===--------------------------------------------------------------------===// //===--------------------------------------------------------------------===//

View File

@ -131,6 +131,14 @@ void LiveRegMatrix::unassign(LiveInterval &VirtReg) {
DEBUG(dbgs() << '\n'); DEBUG(dbgs() << '\n');
} }
bool LiveRegMatrix::isPhysRegUsed(unsigned PhysReg) const {
for (MCRegUnitIterator Unit(PhysReg, TRI); Unit.isValid(); ++Unit) {
if (!Matrix[*Unit].empty())
return true;
}
return false;
}
bool LiveRegMatrix::checkRegMaskInterference(LiveInterval &VirtReg, bool LiveRegMatrix::checkRegMaskInterference(LiveInterval &VirtReg,
unsigned PhysReg) { unsigned PhysReg) {
// Check if the cached information is valid. // Check if the cached information is valid.

View File

@ -400,6 +400,8 @@ private:
typedef SmallVector<HintInfo, 4> HintsInfo; typedef SmallVector<HintInfo, 4> HintsInfo;
BlockFrequency getBrokenHintFreq(const HintsInfo &, unsigned); BlockFrequency getBrokenHintFreq(const HintsInfo &, unsigned);
void collectHintInfo(unsigned, HintsInfo &); void collectHintInfo(unsigned, HintsInfo &);
bool isUnusedCalleeSavedReg(unsigned PhysReg) const;
}; };
} // end anonymous namespace } // end anonymous namespace
@ -816,6 +818,16 @@ void RAGreedy::evictInterference(LiveInterval &VirtReg, unsigned PhysReg,
} }
} }
/// Returns true if the given \p PhysReg is a callee saved register and has not
/// been used for allocation yet.
bool RAGreedy::isUnusedCalleeSavedReg(unsigned PhysReg) const {
unsigned CSR = RegClassInfo.getLastCalleeSavedAlias(PhysReg);
if (CSR == 0)
return false;
return !Matrix->isPhysRegUsed(PhysReg);
}
/// tryEvict - Try to evict all interferences for a physreg. /// tryEvict - Try to evict all interferences for a physreg.
/// @param VirtReg Currently unassigned virtual register. /// @param VirtReg Currently unassigned virtual register.
/// @param Order Physregs to try. /// @param Order Physregs to try.
@ -861,13 +873,12 @@ unsigned RAGreedy::tryEvict(LiveInterval &VirtReg,
continue; continue;
// The first use of a callee-saved register in a function has cost 1. // The first use of a callee-saved register in a function has cost 1.
// Don't start using a CSR when the CostPerUseLimit is low. // Don't start using a CSR when the CostPerUseLimit is low.
if (CostPerUseLimit == 1) if (CostPerUseLimit == 1 && isUnusedCalleeSavedReg(PhysReg)) {
if (unsigned CSR = RegClassInfo.getLastCalleeSavedAlias(PhysReg)) DEBUG(dbgs() << PrintReg(PhysReg, TRI) << " would clobber CSR "
if (!MRI->isPhysRegUsed(CSR)) { << PrintReg(RegClassInfo.getLastCalleeSavedAlias(PhysReg), TRI)
DEBUG(dbgs() << PrintReg(PhysReg, TRI) << " would clobber CSR " << '\n');
<< PrintReg(CSR, TRI) << '\n'); continue;
continue; }
}
if (!canEvictInterference(VirtReg, PhysReg, false, BestCost)) if (!canEvictInterference(VirtReg, PhysReg, false, BestCost))
continue; continue;
@ -1348,9 +1359,8 @@ unsigned RAGreedy::calculateRegionSplitCost(LiveInterval &VirtReg,
unsigned BestCand = NoCand; unsigned BestCand = NoCand;
Order.rewind(); Order.rewind();
while (unsigned PhysReg = Order.next()) { while (unsigned PhysReg = Order.next()) {
if (unsigned CSR = RegClassInfo.getLastCalleeSavedAlias(PhysReg)) if (IgnoreCSR && isUnusedCalleeSavedReg(PhysReg))
if (IgnoreCSR && !MRI->isPhysRegUsed(CSR)) continue;
continue;
// Discard bad candidates before we run out of interference cache cursors. // Discard bad candidates before we run out of interference cache cursors.
// This will only affect register classes with a lot of registers (>32). // This will only affect register classes with a lot of registers (>32).
@ -2134,7 +2144,8 @@ unsigned RAGreedy::tryLastChanceRecoloring(LiveInterval &VirtReg,
unsigned ItVirtReg = (*It)->reg; unsigned ItVirtReg = (*It)->reg;
if (VRM->hasPhys(ItVirtReg)) if (VRM->hasPhys(ItVirtReg))
Matrix->unassign(**It); Matrix->unassign(**It);
Matrix->assign(**It, VirtRegToPhysReg[ItVirtReg]); unsigned ItPhysReg = VirtRegToPhysReg[ItVirtReg];
Matrix->assign(**It, ItPhysReg);
} }
} }
@ -2441,16 +2452,11 @@ unsigned RAGreedy::selectOrSplitImpl(LiveInterval &VirtReg,
// First try assigning a free register. // First try assigning a free register.
AllocationOrder Order(VirtReg.reg, *VRM, RegClassInfo); AllocationOrder Order(VirtReg.reg, *VRM, RegClassInfo);
if (unsigned PhysReg = tryAssign(VirtReg, Order, NewVRegs)) { if (unsigned PhysReg = tryAssign(VirtReg, Order, NewVRegs)) {
// We check other options if we are using a CSR for the first time.
bool CSRFirstUse = false;
if (unsigned CSR = RegClassInfo.getLastCalleeSavedAlias(PhysReg))
if (!MRI->isPhysRegUsed(CSR))
CSRFirstUse = true;
// When NewVRegs is not empty, we may have made decisions such as evicting // When NewVRegs is not empty, we may have made decisions such as evicting
// a virtual register, go with the earlier decisions and use the physical // a virtual register, go with the earlier decisions and use the physical
// register. // register.
if (CSRCost.getFrequency() && CSRFirstUse && NewVRegs.empty()) { if (CSRCost.getFrequency() && isUnusedCalleeSavedReg(PhysReg) &&
NewVRegs.empty()) {
unsigned CSRReg = tryAssignCSRFirstTime(VirtReg, Order, PhysReg, unsigned CSRReg = tryAssignCSRFirstTime(VirtReg, Order, PhysReg,
CostPerUseLimit, NewVRegs); CostPerUseLimit, NewVRegs);
if (CSRReg || !NewVRegs.empty()) if (CSRReg || !NewVRegs.empty())