mirror of
https://github.com/RPCS3/llvm-mirror.git
synced 2025-02-08 21:37:35 +00:00
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:
parent
a397908a1f
commit
e903448f74
@ -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];
|
||||||
|
}
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -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.
|
||||||
//===--------------------------------------------------------------------===//
|
//===--------------------------------------------------------------------===//
|
||||||
|
@ -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.
|
||||||
|
@ -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())
|
||||||
|
Loading…
x
Reference in New Issue
Block a user