Add support for aggressive load-use-store folding. This takes care of the

vast majority of code size regressions introduced by pre-alloc-splitting.


git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@63274 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
Owen Anderson 2009-01-29 05:28:55 +00:00
parent 6e1bb38fb8
commit 45e6855ea2

View File

@ -37,8 +37,9 @@
using namespace llvm; using namespace llvm;
static cl::opt<int> PreSplitLimit("pre-split-limit", cl::init(-1), cl::Hidden); static cl::opt<int> PreSplitLimit("pre-split-limit", cl::init(-1), cl::Hidden);
static cl::opt<int> DeadSplitLimit("dead-split-limit", cl::init(-1), cl::Hidden);
STATISTIC(NumTotalSplits, "Number of intervals split"); STATISTIC(NumSplits, "Number of intervals split");
STATISTIC(NumRemats, "Number of intervals split by rematerialization"); STATISTIC(NumRemats, "Number of intervals split by rematerialization");
STATISTIC(NumFolds, "Number of intervals split with spill folding"); STATISTIC(NumFolds, "Number of intervals split with spill folding");
STATISTIC(NumRenumbers, "Number of intervals renumbered into new registers"); STATISTIC(NumRenumbers, "Number of intervals renumbered into new registers");
@ -78,8 +79,6 @@ namespace {
// Def2SpillMap - A map from a def instruction index to spill index. // Def2SpillMap - A map from a def instruction index to spill index.
DenseMap<unsigned, unsigned> Def2SpillMap; DenseMap<unsigned, unsigned> Def2SpillMap;
unsigned NumSplits;
public: public:
static char ID; static char ID;
PreAllocSplitting() : MachineFunctionPass(&ID) {} PreAllocSplitting() : MachineFunctionPass(&ID) {}
@ -162,8 +161,8 @@ namespace {
void RenumberValno(VNInfo* VN); void RenumberValno(VNInfo* VN);
void ReconstructLiveInterval(LiveInterval* LI); void ReconstructLiveInterval(LiveInterval* LI);
bool removeDeadSpills(SmallPtrSet<LiveInterval*, 8>& split); bool removeDeadSpills(SmallPtrSet<LiveInterval*, 8>& split);
unsigned getNumberOfSpills(SmallPtrSet<MachineInstr*, 4>& MIs, unsigned getNumberOfNonSpills(SmallPtrSet<MachineInstr*, 4>& MIs,
unsigned Reg, int FrameIndex); unsigned Reg, int FrameIndex, bool& TwoAddr);
VNInfo* PerformPHIConstruction(MachineBasicBlock::iterator use, VNInfo* PerformPHIConstruction(MachineBasicBlock::iterator use,
MachineBasicBlock* MBB, MachineBasicBlock* MBB,
LiveInterval* LI, LiveInterval* LI,
@ -789,6 +788,10 @@ void PreAllocSplitting::RenumberValno(VNInfo* VN) {
MO.setReg(NewVReg); MO.setReg(NewVReg);
} }
// The renumbered vreg shares a stack slot with the old register.
if (IntervalSSMap.count(CurrLI->reg))
IntervalSSMap[NewVReg] = IntervalSSMap[CurrLI->reg];
NumRenumbers++; NumRenumbers++;
} }
@ -1042,19 +1045,24 @@ PreAllocSplitting::SplitRegLiveIntervals(const TargetRegisterClass **RCs,
return Change; return Change;
} }
unsigned PreAllocSplitting::getNumberOfSpills( unsigned PreAllocSplitting::getNumberOfNonSpills(
SmallPtrSet<MachineInstr*, 4>& MIs, SmallPtrSet<MachineInstr*, 4>& MIs,
unsigned Reg, int FrameIndex) { unsigned Reg, int FrameIndex,
unsigned Spills = 0; bool& FeedsTwoAddr) {
unsigned NonSpills = 0;
for (SmallPtrSet<MachineInstr*, 4>::iterator UI = MIs.begin(), UE = MIs.end(); for (SmallPtrSet<MachineInstr*, 4>::iterator UI = MIs.begin(), UE = MIs.end();
UI != UI; ++UI) { UI != UE; ++UI) {
int StoreFrameIndex; int StoreFrameIndex;
unsigned StoreVReg = TII->isStoreToStackSlot(*UI, StoreFrameIndex); unsigned StoreVReg = TII->isStoreToStackSlot(*UI, StoreFrameIndex);
if (StoreVReg == Reg && StoreFrameIndex == FrameIndex) if (StoreVReg != Reg || StoreFrameIndex != FrameIndex)
Spills++; NonSpills++;
int DefIdx = (*UI)->findRegisterDefOperandIdx(Reg);
if (DefIdx != -1 && (*UI)->isRegReDefinedByTwoAddr(DefIdx))
FeedsTwoAddr = true;
} }
return Spills; return NonSpills;
} }
/// removeDeadSpills - After doing splitting, filter through all intervals we've /// removeDeadSpills - After doing splitting, filter through all intervals we've
@ -1077,6 +1085,10 @@ bool PreAllocSplitting::removeDeadSpills(SmallPtrSet<LiveInterval*, 8>& split) {
for (LiveInterval::vni_iterator VI = (*LI)->vni_begin(), for (LiveInterval::vni_iterator VI = (*LI)->vni_begin(),
VE = (*LI)->vni_end(); VI != VE; ++VI) { VE = (*LI)->vni_end(); VI != VE; ++VI) {
if (DeadSplitLimit != -1 && (int)NumDeadSpills == DeadSplitLimit)
return changed;
VNInfo* CurrVN = *VI; VNInfo* CurrVN = *VI;
if (CurrVN->hasPHIKill) continue; if (CurrVN->hasPHIKill) continue;
@ -1096,9 +1108,67 @@ bool PreAllocSplitting::removeDeadSpills(SmallPtrSet<LiveInterval*, 8>& split) {
continue; continue;
} }
unsigned SpillCount = getNumberOfSpills(VNUseCount[CurrVN], bool FeedsTwoAddr = false;
(*LI)->reg, FrameIndex); unsigned NonSpillCount = getNumberOfNonSpills(VNUseCount[CurrVN],
if (SpillCount != VNUseCount[CurrVN].size()) continue; (*LI)->reg, FrameIndex,
FeedsTwoAddr);
if (NonSpillCount == 1 && !FeedsTwoAddr) {
SmallPtrSet<MachineInstr*, 4>::iterator UI = VNUseCount[CurrVN].begin();
int StoreFrameIndex;
unsigned StoreVReg = TII->isStoreToStackSlot(*UI, StoreFrameIndex);
while (UI != VNUseCount[CurrVN].end() &&
(StoreVReg == (*LI)->reg && StoreFrameIndex == FrameIndex)) {
++UI;
if (UI != VNUseCount[CurrVN].end())
StoreVReg = TII->isStoreToStackSlot(*UI, StoreFrameIndex);
}
if (UI == VNUseCount[CurrVN].end()) continue;
MachineInstr* use = *UI;
int OpIdx = use->findRegisterUseOperandIdx((*LI)->reg, false);
if (OpIdx == -1) continue;
SmallVector<unsigned, 1> Ops;
Ops.push_back(OpIdx);
if (!TII->canFoldMemoryOperand(use, Ops)) continue;
MachineInstr* NewMI =
TII->foldMemoryOperand(*use->getParent()->getParent(),
use, Ops, FrameIndex);
if (!NewMI) continue;
LIs->RemoveMachineInstrFromMaps(DefMI);
LIs->ReplaceMachineInstrInMaps(use, NewMI);
(*LI)->removeValNo(CurrVN);
DefMI->eraseFromParent();
MachineBasicBlock* MBB = use->getParent();
NewMI = MBB->insert(MBB->erase(use), NewMI);
VNUseCount[CurrVN].erase(use);
for (SmallPtrSet<MachineInstr*, 4>::iterator II =
VNUseCount[CurrVN].begin(), IE = VNUseCount[CurrVN].end();
II != IE; ++II) {
LIs->RemoveMachineInstrFromMaps(*II);
(*II)->eraseFromParent();
}
for (DenseMap<VNInfo*, SmallPtrSet<MachineInstr*, 4> >::iterator
VI = VNUseCount.begin(), VE = VNUseCount.end(); VI != VE; ++VI)
if (VI->second.erase(use))
VI->second.insert(NewMI);
NumDeadSpills++;
changed = true;
continue;
}
if (NonSpillCount) continue;
for (SmallPtrSet<MachineInstr*, 4>::iterator UI = for (SmallPtrSet<MachineInstr*, 4>::iterator UI =
VNUseCount[CurrVN].begin(), UE = VNUseCount[CurrVN].end(); VNUseCount[CurrVN].begin(), UE = VNUseCount[CurrVN].end();
@ -1193,7 +1263,6 @@ bool PreAllocSplitting::runOnMachineFunction(MachineFunction &MF) {
LSs = &getAnalysis<LiveStacks>(); LSs = &getAnalysis<LiveStacks>();
bool MadeChange = false; bool MadeChange = false;
NumSplits = 0;
// Make sure blocks are numbered in order. // Make sure blocks are numbered in order.
MF.RenumberBlocks(); MF.RenumberBlocks();
@ -1221,8 +1290,5 @@ bool PreAllocSplitting::runOnMachineFunction(MachineFunction &MF) {
MadeChange |= removeDeadSpills(Split); MadeChange |= removeDeadSpills(Split);
if (NumSplits)
NumTotalSplits += NumSplits;
return MadeChange; return MadeChange;
} }