Hoist spills when the same value is known to be in less loopy sibling registers.

Stack slot real estate is virtually free compared to registers, so it is
advantageous to spill earlier even though the same value is now kept in both a
register and a stack slot.

Also eliminate redundant spills by extending the stack slot live range
underneath reloaded registers.

This can trigger a dead code elimination, removing copies and even reloads that
were only feeding spills.

llvm-svn: 127868
This commit is contained in:
Jakob Stoklund Olesen 2011-03-18 04:23:06 +00:00
parent 01fb4b018e
commit 27320cb864
1 changed files with 148 additions and 22 deletions

View File

@ -83,7 +83,11 @@ class InlineSpiller : public Spiller {
}; };
// Values in RegsToSpill defined by sibling copies. // Values in RegsToSpill defined by sibling copies.
DenseMap<VNInfo*, SibValueInfo> SibValues; typedef DenseMap<VNInfo*, SibValueInfo> SibValueMap;
SibValueMap SibValues;
// Dead defs generated during spilling.
SmallVector<MachineInstr*, 8> DeadDefs;
~InlineSpiller() {} ~InlineSpiller() {}
@ -110,9 +114,18 @@ private:
bool isSnippet(const LiveInterval &SnipLI); bool isSnippet(const LiveInterval &SnipLI);
void collectRegsToSpill(); void collectRegsToSpill();
bool isRegToSpill(unsigned Reg) {
return std::find(RegsToSpill.begin(),
RegsToSpill.end(), Reg) != RegsToSpill.end();
}
bool isSibling(unsigned Reg);
void traceSiblingValue(unsigned, VNInfo*, VNInfo*); void traceSiblingValue(unsigned, VNInfo*, VNInfo*);
void analyzeSiblingValues(); void analyzeSiblingValues();
bool hoistSpill(LiveInterval &SpillLI, MachineInstr *CopyMI);
void eliminateRedundantSpills(unsigned Reg, VNInfo *VNI);
bool reMaterializeFor(MachineBasicBlock::iterator MI); bool reMaterializeFor(MachineBasicBlock::iterator MI);
void reMaterializeAll(); void reMaterializeAll();
@ -225,18 +238,13 @@ void InlineSpiller::collectRegsToSpill() {
for (MachineRegisterInfo::reg_iterator RI = MRI.reg_begin(Reg); for (MachineRegisterInfo::reg_iterator RI = MRI.reg_begin(Reg);
MachineInstr *MI = RI.skipInstruction();) { MachineInstr *MI = RI.skipInstruction();) {
unsigned SnipReg = isFullCopyOf(MI, Reg); unsigned SnipReg = isFullCopyOf(MI, Reg);
if (!SnipReg) if (!isSibling(SnipReg))
continue;
if (!TargetRegisterInfo::isVirtualRegister(SnipReg))
continue;
if (VRM.getOriginal(SnipReg) != Original)
continue; continue;
LiveInterval &SnipLI = LIS.getInterval(SnipReg); LiveInterval &SnipLI = LIS.getInterval(SnipReg);
if (!isSnippet(SnipLI)) if (!isSnippet(SnipLI))
continue; continue;
SnippetCopies.insert(MI); SnippetCopies.insert(MI);
if (std::find(RegsToSpill.begin(), RegsToSpill.end(), if (!isRegToSpill(SnipReg))
SnipReg) == RegsToSpill.end())
RegsToSpill.push_back(SnipReg); RegsToSpill.push_back(SnipReg);
DEBUG(dbgs() << "\talso spill snippet " << SnipLI << '\n'); DEBUG(dbgs() << "\talso spill snippet " << SnipLI << '\n');
@ -256,6 +264,11 @@ void InlineSpiller::collectRegsToSpill() {
// the value has already been spilled, or we may want to hoist the spill from a // the value has already been spilled, or we may want to hoist the spill from a
// loop. // loop.
bool InlineSpiller::isSibling(unsigned Reg) {
return TargetRegisterInfo::isVirtualRegister(Reg) &&
VRM.getOriginal(Reg) == Original;
}
/// traceSiblingValue - Trace a value that is about to be spilled back to the /// traceSiblingValue - Trace a value that is about to be spilled back to the
/// real defining instructions by looking through sibling copies. Always stay /// real defining instructions by looking through sibling copies. Always stay
/// within the range of OrigVNI so the registers are known to carry the same /// within the range of OrigVNI so the registers are known to carry the same
@ -271,14 +284,13 @@ void InlineSpiller::traceSiblingValue(unsigned UseReg, VNInfo *UseVNI,
DEBUG(dbgs() << "Tracing value " << PrintReg(UseReg) << ':' DEBUG(dbgs() << "Tracing value " << PrintReg(UseReg) << ':'
<< UseVNI->id << '@' << UseVNI->def << '\n'); << UseVNI->id << '@' << UseVNI->def << '\n');
SmallPtrSet<VNInfo*, 8> Visited; SmallPtrSet<VNInfo*, 8> Visited;
SmallVector<std::pair<unsigned,VNInfo*>, 8> WorkList; SmallVector<std::pair<unsigned, VNInfo*>, 8> WorkList;
WorkList.push_back(std::make_pair(UseReg, UseVNI)); WorkList.push_back(std::make_pair(UseReg, UseVNI));
// Best spill candidate seen so far. This must dominate UseVNI. // Best spill candidate seen so far. This must dominate UseVNI.
SibValueInfo SVI(UseReg, UseVNI); SibValueInfo SVI(UseReg, UseVNI);
MachineBasicBlock *UseMBB = LIS.getMBBFromIndex(UseVNI->def); MachineBasicBlock *UseMBB = LIS.getMBBFromIndex(UseVNI->def);
MachineBasicBlock *SpillMBB = UseMBB; unsigned SpillDepth = Loops.getLoopDepth(UseMBB);
unsigned SpillDepth = Loops.getLoopDepth(SpillMBB);
bool SeenOrigPHI = false; // Original PHI met. bool SeenOrigPHI = false; // Original PHI met.
do { do {
@ -289,13 +301,9 @@ void InlineSpiller::traceSiblingValue(unsigned UseReg, VNInfo *UseVNI,
continue; continue;
// Is this value a better spill candidate? // Is this value a better spill candidate?
if (VNI != SVI.SpillVNI) { if (!isRegToSpill(Reg)) {
MachineBasicBlock *MBB = LIS.getMBBFromIndex(VNI->def); MachineBasicBlock *MBB = LIS.getMBBFromIndex(VNI->def);
if (MBB == SpillMBB) { if (MBB != UseMBB && MDT.dominates(MBB, UseMBB)) {
// Prefer to spill a previous value in the same block.
if (VNI->def < SVI.SpillVNI->def)
SVI.SpillReg = Reg, SVI.SpillVNI = VNI;
} else if (MDT.dominates(MBB, UseMBB)) {
// This is a valid spill location dominating UseVNI. // This is a valid spill location dominating UseVNI.
// Prefer to spill at a smaller loop depth. // Prefer to spill at a smaller loop depth.
unsigned Depth = Loops.getLoopDepth(MBB); unsigned Depth = Loops.getLoopDepth(MBB);
@ -304,7 +312,6 @@ void InlineSpiller::traceSiblingValue(unsigned UseReg, VNInfo *UseVNI,
<< ':' << VNI->id << '@' << VNI->def << '\n'); << ':' << VNI->id << '@' << VNI->def << '\n');
SVI.SpillReg = Reg; SVI.SpillReg = Reg;
SVI.SpillVNI = VNI; SVI.SpillVNI = VNI;
SpillMBB = MBB;
SpillDepth = Depth; SpillDepth = Depth;
} }
} }
@ -335,8 +342,7 @@ void InlineSpiller::traceSiblingValue(unsigned UseReg, VNInfo *UseVNI,
// Trace through sibling copies. // Trace through sibling copies.
if (unsigned SrcReg = isFullCopyOf(MI, Reg)) { if (unsigned SrcReg = isFullCopyOf(MI, Reg)) {
if (TargetRegisterInfo::isVirtualRegister(SrcReg) && if (isSibling(SrcReg)) {
VRM.getOriginal(SrcReg) == Original) {
LiveInterval &SrcLI = LIS.getInterval(SrcReg); LiveInterval &SrcLI = LIS.getInterval(SrcReg);
VNInfo *SrcVNI = SrcLI.getVNInfoAt(VNI->def.getUseIndex()); VNInfo *SrcVNI = SrcLI.getVNInfoAt(VNI->def.getUseIndex());
assert(SrcVNI && "Copy from non-existing value"); assert(SrcVNI && "Copy from non-existing value");
@ -406,6 +412,108 @@ void InlineSpiller::analyzeSiblingValues() {
} }
} }
/// hoistSpill - Given a sibling copy that defines a value to be spilled, insert
/// a spill at a better location.
bool InlineSpiller::hoistSpill(LiveInterval &SpillLI, MachineInstr *CopyMI) {
SlotIndex Idx = LIS.getInstructionIndex(CopyMI);
VNInfo *VNI = SpillLI.getVNInfoAt(Idx.getDefIndex());
assert(VNI && VNI->def == Idx.getDefIndex() && "Not defined by copy");
SibValueMap::const_iterator I = SibValues.find(VNI);
if (I == SibValues.end())
return false;
const SibValueInfo &SVI = I->second;
// Let the normal folding code deal with the boring case.
if (!SVI.AllDefsAreReloads && SVI.SpillVNI == VNI)
return false;
// Conservatively extend the stack slot range to the range of the original
// value. We may be able to do better with stack slot coloring by being more
// careful here.
LiveInterval &StackInt = LSS.getInterval(StackSlot);
LiveInterval &OrigLI = LIS.getInterval(Original);
VNInfo *OrigVNI = OrigLI.getVNInfoAt(Idx);
StackInt.MergeValueInAsValue(OrigLI, OrigVNI, StackInt.getValNumInfo(0));
// Already spilled everywhere.
if (SVI.AllDefsAreReloads)
return true;
// We are going to spill SVI.SpillVNI immediately after its def, so clear out
// any later spills of the same value.
eliminateRedundantSpills(SVI.SpillReg, SVI.SpillVNI);
MachineBasicBlock *MBB = LIS.getMBBFromIndex(SVI.SpillVNI->def);
MachineBasicBlock::iterator MII;
if (SVI.SpillVNI->isPHIDef())
MII = MBB->SkipPHIsAndLabels(MBB->begin());
else {
MII = LIS.getInstructionFromIndex(SVI.SpillVNI->def);
++MII;
}
// Insert spill without kill flag immediately after def.
TII.storeRegToStackSlot(*MBB, MII, SVI.SpillReg, false, StackSlot, RC, &TRI);
--MII; // Point to store instruction.
LIS.InsertMachineInstrInMaps(MII);
VRM.addSpillSlotUse(StackSlot, MII);
DEBUG(dbgs() << "\thoisted: " << SVI.SpillVNI->def << '\t' << *MII);
return true;
}
/// eliminateRedundantSpills - Reg:VNI is known to be on the stack. Remove any
/// redundant spills of this value in Reg and sibling copies.
void InlineSpiller::eliminateRedundantSpills(unsigned Reg, VNInfo *VNI) {
SmallVector<std::pair<unsigned, VNInfo*>, 8> WorkList;
WorkList.push_back(std::make_pair(Reg, VNI));
LiveInterval &StackInt = LSS.getInterval(StackSlot);
do {
tie(Reg, VNI) = WorkList.pop_back_val();
DEBUG(dbgs() << "Checking redundant spills for " << PrintReg(Reg) << ':'
<< VNI->id << '@' << VNI->def << '\n');
// Regs to spill are taken care of.
if (isRegToSpill(Reg))
continue;
// Add all of VNI's live range to StackInt.
LiveInterval &LI = LIS.getInterval(Reg);
StackInt.MergeValueInAsValue(LI, VNI, StackInt.getValNumInfo(0));
// Find all spills and copies of VNI.
for (MachineRegisterInfo::use_nodbg_iterator UI = MRI.use_nodbg_begin(Reg);
MachineInstr *MI = UI.skipInstruction();) {
if (!MI->isCopy() && !MI->getDesc().mayStore())
continue;
SlotIndex Idx = LIS.getInstructionIndex(MI);
if (LI.getVNInfoAt(Idx) != VNI)
continue;
// Follow sibling copies down the dominator tree.
if (unsigned DstReg = isFullCopyOf(MI, Reg)) {
if (isSibling(DstReg)) {
LiveInterval &DstLI = LIS.getInterval(DstReg);
VNInfo *DstVNI = DstLI.getVNInfoAt(Idx.getDefIndex());
assert(DstVNI && "Missing defined value");
assert(DstVNI->def == Idx.getDefIndex() && "Wrong copy def slot");
WorkList.push_back(std::make_pair(DstReg, DstVNI));
}
continue;
}
// Erase spills.
int FI;
if (Reg == TII.isStoreToStackSlot(MI, FI) && FI == StackSlot) {
DEBUG(dbgs() << "Redundant spill " << Idx << '\t' << *MI);
// eliminateDeadDefs won't normally remove stores, so switch opcode.
MI->setDesc(TII.get(TargetOpcode::KILL));
DeadDefs.push_back(MI);
}
}
} while (!WorkList.empty());
}
/// reMaterializeFor - Attempt to rematerialize before MI instead of reloading. /// reMaterializeFor - Attempt to rematerialize before MI instead of reloading.
bool InlineSpiller::reMaterializeFor(MachineBasicBlock::iterator MI) { bool InlineSpiller::reMaterializeFor(MachineBasicBlock::iterator MI) {
SlotIndex UseIdx = LIS.getInstructionIndex(MI).getUseIndex(); SlotIndex UseIdx = LIS.getInstructionIndex(MI).getUseIndex();
@ -679,6 +787,19 @@ void InlineSpiller::spillAroundUses(unsigned Reg) {
SmallVector<unsigned, 8> Ops; SmallVector<unsigned, 8> Ops;
tie(Reads, Writes) = MI->readsWritesVirtualRegister(Reg, &Ops); tie(Reads, Writes) = MI->readsWritesVirtualRegister(Reg, &Ops);
// Check for a sibling copy.
unsigned SibReg = isFullCopyOf(MI, Reg);
if (!isSibling(SibReg))
SibReg = 0;
// Hoist the spill of a sib-reg copy.
if (SibReg && Writes && !Reads && hoistSpill(OldLI, MI)) {
// This COPY is now dead, the value is already in the stack slot.
MI->getOperand(0).setIsDead();
DeadDefs.push_back(MI);
continue;
}
// Attempt to fold memory ops. // Attempt to fold memory ops.
if (foldMemoryOperand(MI, Ops)) if (foldMemoryOperand(MI, Ops))
continue; continue;
@ -728,11 +849,10 @@ void InlineSpiller::spill(LiveRangeEdit &edit) {
<< LIS.getInterval(Original) << '\n'); << LIS.getInterval(Original) << '\n');
assert(edit.getParent().isSpillable() && assert(edit.getParent().isSpillable() &&
"Attempting to spill already spilled value."); "Attempting to spill already spilled value.");
assert(DeadDefs.empty() && "Previous spill didn't remove dead defs");
collectRegsToSpill(); collectRegsToSpill();
analyzeSiblingValues(); analyzeSiblingValues();
reMaterializeAll(); reMaterializeAll();
// Remat may handle everything. // Remat may handle everything.
@ -759,6 +879,12 @@ void InlineSpiller::spill(LiveRangeEdit &edit) {
for (unsigned i = 0, e = RegsToSpill.size(); i != e; ++i) for (unsigned i = 0, e = RegsToSpill.size(); i != e; ++i)
spillAroundUses(RegsToSpill[i]); spillAroundUses(RegsToSpill[i]);
// Hoisted spills may cause dead code.
if (!DeadDefs.empty()) {
DEBUG(dbgs() << "Eliminating " << DeadDefs.size() << " dead defs\n");
Edit->eliminateDeadDefs(DeadDefs, LIS, VRM, TII);
}
// Finally delete the SnippetCopies. // Finally delete the SnippetCopies.
for (MachineRegisterInfo::reg_iterator RI = MRI.reg_begin(edit.getReg()); for (MachineRegisterInfo::reg_iterator RI = MRI.reg_begin(edit.getReg());
MachineInstr *MI = RI.skipInstruction();) { MachineInstr *MI = RI.skipInstruction();) {