Create two BlockInfo entries when a live range is discontinuous through a block.

Delete the Kill and Def markers in BlockInfo. They are no longer
necessary when BlockInfo describes a continuous live range.

This only affects the relatively rare kind of basic block where a live
range looks like this:

 |---x   o---|

Now live range splitting can pretend that it is looking at two blocks:

 |---x
         o---|

This allows the code to be simplified a bit.

llvm-svn: 132245
This commit is contained in:
Jakob Stoklund Olesen 2011-05-28 02:33:00 +00:00
parent 5cc91b2611
commit fd3f71ef3a
3 changed files with 72 additions and 56 deletions

View File

@ -552,7 +552,7 @@ bool RAGreedy::addSplitConstraints(InterferenceCache::Cursor Intf,
BC.Entry = SpillPlacement::MustSpill, ++Ins; BC.Entry = SpillPlacement::MustSpill, ++Ins;
else if (Intf.first() < BI.FirstUse) else if (Intf.first() < BI.FirstUse)
BC.Entry = SpillPlacement::PrefSpill, ++Ins; BC.Entry = SpillPlacement::PrefSpill, ++Ins;
else if (Intf.first() < (BI.LiveThrough ? BI.LastUse : BI.Kill)) else if (Intf.first() < BI.LastUse)
++Ins; ++Ins;
} }
@ -562,7 +562,7 @@ bool RAGreedy::addSplitConstraints(InterferenceCache::Cursor Intf,
BC.Exit = SpillPlacement::MustSpill, ++Ins; BC.Exit = SpillPlacement::MustSpill, ++Ins;
else if (Intf.last() > BI.LastUse) else if (Intf.last() > BI.LastUse)
BC.Exit = SpillPlacement::PrefSpill, ++Ins; BC.Exit = SpillPlacement::PrefSpill, ++Ins;
else if (Intf.last() > (BI.LiveThrough ? BI.FirstUse : BI.Def)) else if (Intf.last() > BI.FirstUse)
++Ins; ++Ins;
} }
@ -811,7 +811,7 @@ void RAGreedy::splitAroundRegion(LiveInterval &VirtReg,
DEBUG(dbgs() << ", no interference"); DEBUG(dbgs() << ", no interference");
if (!BI.LiveThrough) { if (!BI.LiveThrough) {
DEBUG(dbgs() << ", not live-through.\n"); DEBUG(dbgs() << ", not live-through.\n");
SE->useIntv(SE->enterIntvBefore(BI.Def), Stop); SE->useIntv(SE->enterIntvBefore(BI.FirstUse), Stop);
continue; continue;
} }
if (!RegIn) { if (!RegIn) {
@ -828,10 +828,10 @@ void RAGreedy::splitAroundRegion(LiveInterval &VirtReg,
// Block has interference. // Block has interference.
DEBUG(dbgs() << ", interference to " << Intf.last()); DEBUG(dbgs() << ", interference to " << Intf.last());
if (!BI.LiveThrough && Intf.last() <= BI.Def) { if (!BI.LiveThrough && Intf.last() <= BI.FirstUse) {
// The interference doesn't reach the outgoing segment. // The interference doesn't reach the outgoing segment.
DEBUG(dbgs() << " doesn't affect def from " << BI.Def << '\n'); DEBUG(dbgs() << " doesn't affect def from " << BI.FirstUse << '\n');
SE->useIntv(BI.Def, Stop); SE->useIntv(BI.FirstUse, Stop);
continue; continue;
} }
@ -887,7 +887,7 @@ void RAGreedy::splitAroundRegion(LiveInterval &VirtReg,
DEBUG(dbgs() << ", no interference"); DEBUG(dbgs() << ", no interference");
if (!BI.LiveThrough) { if (!BI.LiveThrough) {
DEBUG(dbgs() << ", killed in block.\n"); DEBUG(dbgs() << ", killed in block.\n");
SE->useIntv(Start, SE->leaveIntvAfter(BI.Kill)); SE->useIntv(Start, SE->leaveIntvAfter(BI.LastUse));
continue; continue;
} }
if (!RegOut) { if (!RegOut) {
@ -920,10 +920,10 @@ void RAGreedy::splitAroundRegion(LiveInterval &VirtReg,
// Block has interference. // Block has interference.
DEBUG(dbgs() << ", interference from " << Intf.first()); DEBUG(dbgs() << ", interference from " << Intf.first());
if (!BI.LiveThrough && Intf.first() >= BI.Kill) { if (!BI.LiveThrough && Intf.first() >= BI.LastUse) {
// The interference doesn't reach the outgoing segment. // The interference doesn't reach the outgoing segment.
DEBUG(dbgs() << " doesn't affect kill at " << BI.Kill << '\n'); DEBUG(dbgs() << " doesn't affect kill at " << BI.LastUse << '\n');
SE->useIntv(Start, BI.Kill); SE->useIntv(Start, BI.LastUse);
continue; continue;
} }

View File

@ -145,7 +145,7 @@ void SplitAnalysis::analyzeUses() {
/// where CurLI is live. /// where CurLI is live.
bool SplitAnalysis::calcLiveBlockInfo() { bool SplitAnalysis::calcLiveBlockInfo() {
ThroughBlocks.resize(MF.getNumBlockIDs()); ThroughBlocks.resize(MF.getNumBlockIDs());
NumThroughBlocks = 0; NumThroughBlocks = NumGapBlocks = 0;
if (CurLI->empty()) if (CurLI->empty())
return true; return true;
@ -164,55 +164,62 @@ bool SplitAnalysis::calcLiveBlockInfo() {
SlotIndex Start, Stop; SlotIndex Start, Stop;
tie(Start, Stop) = LIS.getSlotIndexes()->getMBBRange(BI.MBB); tie(Start, Stop) = LIS.getSlotIndexes()->getMBBRange(BI.MBB);
// LVI is the first live segment overlapping MBB. // If the block contains no uses, the range must be live through. At one
BI.LiveIn = LVI->start <= Start; // point, SimpleRegisterCoalescing could create dangling ranges that ended
if (!BI.LiveIn) // mid-block.
BI.Def = LVI->start; if (UseI == UseE || *UseI >= Stop) {
++NumThroughBlocks;
// Find the first and last uses in the block. ThroughBlocks.set(BI.MBB->getNumber());
bool Uses = UseI != UseE && *UseI < Stop; // The range shouldn't end mid-block if there are no uses. This shouldn't
if (Uses) { // happen.
if (LVI->end < Stop)
return false;
} else {
// This block has uses. Find the first and last uses in the block.
BI.FirstUse = *UseI; BI.FirstUse = *UseI;
assert(BI.FirstUse >= Start); assert(BI.FirstUse >= Start);
do ++UseI; do ++UseI;
while (UseI != UseE && *UseI < Stop); while (UseI != UseE && *UseI < Stop);
BI.LastUse = UseI[-1]; BI.LastUse = UseI[-1];
assert(BI.LastUse < Stop); assert(BI.LastUse < Stop);
}
// Look for gaps in the live range. // LVI is the first live segment overlapping MBB.
bool hasGap = false; BI.LiveIn = LVI->start <= Start;
BI.LiveOut = true;
while (LVI->end < Stop) {
SlotIndex LastStop = LVI->end;
if (++LVI == LVE || LVI->start >= Stop) {
BI.Kill = LastStop;
BI.LiveOut = false;
break;
}
if (LastStop < LVI->start) {
hasGap = true;
BI.Kill = LastStop;
BI.Def = LVI->start;
}
}
// Don't set LiveThrough when the block has a gap. // Look for gaps in the live range.
BI.LiveThrough = !hasGap && BI.LiveIn && BI.LiveOut; BI.LiveOut = true;
if (Uses) while (LVI->end < Stop) {
SlotIndex LastStop = LVI->end;
if (++LVI == LVE || LVI->start >= Stop) {
BI.LiveOut = false;
break;
}
if (LastStop < LVI->start) {
// There is a gap in the live range. Create duplicate entries for the
// live-in snippet and the live-out snippet.
++NumGapBlocks;
// Push the Live-in part.
BI.LiveThrough = false;
BI.LiveOut = false;
UseBlocks.push_back(BI);
UseBlocks.back().LastUse = LastStop;
// Set up BI for the live-out part.
BI.LiveIn = false;
BI.LiveOut = true;
BI.FirstUse = LVI->start;
}
}
// Don't set LiveThrough when the block has a gap.
BI.LiveThrough = BI.LiveIn && BI.LiveOut;
UseBlocks.push_back(BI); UseBlocks.push_back(BI);
else {
++NumThroughBlocks;
ThroughBlocks.set(BI.MBB->getNumber());
}
// FIXME: This should never happen. The live range stops or starts without a
// corresponding use. An earlier pass did something wrong.
if (!BI.LiveThrough && !Uses)
return false;
// LVI is now at LVE or LVI->end >= Stop. // LVI is now at LVE or LVI->end >= Stop.
if (LVI == LVE) if (LVI == LVE)
break; break;
}
// Live segment ends exactly at Stop. Move to the next segment. // Live segment ends exactly at Stop. Move to the next segment.
if (LVI->end == Stop && ++LVI == LVE) if (LVI->end == Stop && ++LVI == LVE)

View File

@ -63,17 +63,22 @@ public:
/// 1. | o---x | Internal to block. Variable is only live in this block. /// 1. | o---x | Internal to block. Variable is only live in this block.
/// 2. |---x | Live-in, kill. /// 2. |---x | Live-in, kill.
/// 3. | o---| Def, live-out. /// 3. | o---| Def, live-out.
/// 4. |---x o---| Live-in, kill, def, live-out. /// 4. |---x o---| Live-in, kill, def, live-out. Counted by NumGapBlocks.
/// 5. |---o---o---| Live-through with uses or defs. /// 5. |---o---o---| Live-through with uses or defs.
/// 6. |-----------| Live-through without uses. Transparent. /// 6. |-----------| Live-through without uses. Counted by NumThroughBlocks.
///
/// Two BlockInfo entries are created for template 4. One for the live-in
/// segment, and one for the live-out segment. These entries look as if the
/// block were split in the middle where the live range isn't live.
///
/// Live-through blocks without any uses don't get BlockInfo entries. They
/// are simply listed in ThroughBlocks instead.
/// ///
struct BlockInfo { struct BlockInfo {
MachineBasicBlock *MBB; MachineBasicBlock *MBB;
SlotIndex FirstUse; ///< First instr using current reg. SlotIndex FirstUse; ///< First instr using current reg.
SlotIndex LastUse; ///< Last instr using current reg. SlotIndex LastUse; ///< Last instr using current reg.
SlotIndex Kill; ///< Interval end point inside block. bool LiveThrough; ///< Live in whole block (Templ 5. above).
SlotIndex Def; ///< Interval start point inside block.
bool LiveThrough; ///< Live in whole block (Templ 5. or 6. above).
bool LiveIn; ///< Current reg is live in. bool LiveIn; ///< Current reg is live in.
bool LiveOut; ///< Current reg is live out. bool LiveOut; ///< Current reg is live out.
}; };
@ -91,6 +96,10 @@ private:
/// UseBlocks - Blocks where CurLI has uses. /// UseBlocks - Blocks where CurLI has uses.
SmallVector<BlockInfo, 8> UseBlocks; SmallVector<BlockInfo, 8> UseBlocks;
/// NumGapBlocks - Number of duplicate entries in UseBlocks for blocks where
/// the live range has a gap.
unsigned NumGapBlocks;
/// ThroughBlocks - Block numbers where CurLI is live through without uses. /// ThroughBlocks - Block numbers where CurLI is live through without uses.
BitVector ThroughBlocks; BitVector ThroughBlocks;
@ -160,7 +169,7 @@ public:
/// getNumLiveBlocks - Return the number of blocks where CurLI is live. /// getNumLiveBlocks - Return the number of blocks where CurLI is live.
unsigned getNumLiveBlocks() const { unsigned getNumLiveBlocks() const {
return getUseBlocks().size() + getNumThroughBlocks(); return getUseBlocks().size() - NumGapBlocks + getNumThroughBlocks();
} }
/// countLiveBlocks - Return the number of blocks where li is live. This is /// countLiveBlocks - Return the number of blocks where li is live. This is