forked from OSchip/llvm-project
[SimplifyCFG][BasicBlockUtils] Port SplitBlockPredecessors()/SplitLandingPadPredecessors() to DomTreeUpdater
This is not nice, but it's the best transient solution possible, and is better than just duplicating the whole function. The problem is, this function is widely used, and it is not at all obvious that all the users could be painlessly switched to operate on DomTreeUpdater, and somehow i don't feel like porting all those users first. This function is one of last three that not operate on DomTreeUpdater.
This commit is contained in:
parent
286cf6cb02
commit
c6654a4cda
|
@ -289,6 +289,28 @@ BasicBlock *splitBlockBefore(BasicBlock *Old, Instruction *SplitPt,
|
|||
DomTreeUpdater *DTU, LoopInfo *LI,
|
||||
MemorySSAUpdater *MSSAU, const Twine &BBName = "");
|
||||
|
||||
/// This method introduces at least one new basic block into the function and
|
||||
/// moves some of the predecessors of BB to be predecessors of the new block.
|
||||
/// The new predecessors are indicated by the Preds array. The new block is
|
||||
/// given a suffix of 'Suffix'. Returns new basic block to which predecessors
|
||||
/// from Preds are now pointing.
|
||||
///
|
||||
/// If BB is a landingpad block then additional basicblock might be introduced.
|
||||
/// It will have Suffix+".split_lp". See SplitLandingPadPredecessors for more
|
||||
/// details on this case.
|
||||
///
|
||||
/// This currently updates the LLVM IR, DominatorTree, LoopInfo, and LCCSA but
|
||||
/// no other analyses. In particular, it does not preserve LoopSimplify
|
||||
/// (because it's complicated to handle the case where one of the edges being
|
||||
/// split is an exit of a loop with other exits).
|
||||
///
|
||||
/// FIXME: deprecated, switch to the DomTreeUpdater-based one.
|
||||
BasicBlock *SplitBlockPredecessors(BasicBlock *BB, ArrayRef<BasicBlock *> Preds,
|
||||
const char *Suffix, DominatorTree *DT,
|
||||
LoopInfo *LI = nullptr,
|
||||
MemorySSAUpdater *MSSAU = nullptr,
|
||||
bool PreserveLCSSA = false);
|
||||
|
||||
/// This method introduces at least one new basic block into the function and
|
||||
/// moves some of the predecessors of BB to be predecessors of the new block.
|
||||
/// The new predecessors are indicated by the Preds array. The new block is
|
||||
|
@ -305,11 +327,32 @@ BasicBlock *splitBlockBefore(BasicBlock *Old, Instruction *SplitPt,
|
|||
/// split is an exit of a loop with other exits).
|
||||
BasicBlock *SplitBlockPredecessors(BasicBlock *BB, ArrayRef<BasicBlock *> Preds,
|
||||
const char *Suffix,
|
||||
DominatorTree *DT = nullptr,
|
||||
DomTreeUpdater *DTU = nullptr,
|
||||
LoopInfo *LI = nullptr,
|
||||
MemorySSAUpdater *MSSAU = nullptr,
|
||||
bool PreserveLCSSA = false);
|
||||
|
||||
/// This method transforms the landing pad, OrigBB, by introducing two new basic
|
||||
/// blocks into the function. One of those new basic blocks gets the
|
||||
/// predecessors listed in Preds. The other basic block gets the remaining
|
||||
/// predecessors of OrigBB. The landingpad instruction OrigBB is clone into both
|
||||
/// of the new basic blocks. The new blocks are given the suffixes 'Suffix1' and
|
||||
/// 'Suffix2', and are returned in the NewBBs vector.
|
||||
///
|
||||
/// This currently updates the LLVM IR, DominatorTree, LoopInfo, and LCCSA but
|
||||
/// no other analyses. In particular, it does not preserve LoopSimplify
|
||||
/// (because it's complicated to handle the case where one of the edges being
|
||||
/// split is an exit of a loop with other exits).
|
||||
///
|
||||
/// FIXME: deprecated, switch to the DomTreeUpdater-based one.
|
||||
void SplitLandingPadPredecessors(BasicBlock *OrigBB,
|
||||
ArrayRef<BasicBlock *> Preds,
|
||||
const char *Suffix, const char *Suffix2,
|
||||
SmallVectorImpl<BasicBlock *> &NewBBs,
|
||||
DominatorTree *DT, LoopInfo *LI = nullptr,
|
||||
MemorySSAUpdater *MSSAU = nullptr,
|
||||
bool PreserveLCSSA = false);
|
||||
|
||||
/// This method transforms the landing pad, OrigBB, by introducing two new basic
|
||||
/// blocks into the function. One of those new basic blocks gets the
|
||||
/// predecessors listed in Preds. The other basic block gets the remaining
|
||||
|
@ -324,7 +367,7 @@ BasicBlock *SplitBlockPredecessors(BasicBlock *BB, ArrayRef<BasicBlock *> Preds,
|
|||
void SplitLandingPadPredecessors(
|
||||
BasicBlock *OrigBB, ArrayRef<BasicBlock *> Preds, const char *Suffix,
|
||||
const char *Suffix2, SmallVectorImpl<BasicBlock *> &NewBBs,
|
||||
DominatorTree *DT = nullptr, LoopInfo *LI = nullptr,
|
||||
DomTreeUpdater *DTU = nullptr, LoopInfo *LI = nullptr,
|
||||
MemorySSAUpdater *MSSAU = nullptr, bool PreserveLCSSA = false);
|
||||
|
||||
/// This method duplicates the specified return instruction into a predecessor
|
||||
|
|
|
@ -659,11 +659,31 @@ BasicBlock *llvm::splitBlockBefore(BasicBlock *Old, Instruction *SplitPt,
|
|||
/// Update DominatorTree, LoopInfo, and LCCSA analysis information.
|
||||
static void UpdateAnalysisInformation(BasicBlock *OldBB, BasicBlock *NewBB,
|
||||
ArrayRef<BasicBlock *> Preds,
|
||||
DominatorTree *DT, LoopInfo *LI,
|
||||
MemorySSAUpdater *MSSAU,
|
||||
DomTreeUpdater *DTU, DominatorTree *DT,
|
||||
LoopInfo *LI, MemorySSAUpdater *MSSAU,
|
||||
bool PreserveLCSSA, bool &HasLoopExit) {
|
||||
// Update dominator tree if available.
|
||||
if (DT) {
|
||||
if (DTU) {
|
||||
// Recalculation of DomTree is needed when updating a forward DomTree and
|
||||
// the Entry BB is replaced.
|
||||
if (NewBB == &NewBB->getParent()->getEntryBlock() && DTU->hasDomTree()) {
|
||||
// The entry block was removed and there is no external interface for
|
||||
// the dominator tree to be notified of this change. In this corner-case
|
||||
// we recalculate the entire tree.
|
||||
DTU->recalculate(*NewBB->getParent());
|
||||
} else {
|
||||
// Split block expects NewBB to have a non-empty set of predecessors.
|
||||
SmallVector<DominatorTree::UpdateType, 8> Updates;
|
||||
SmallSetVector<BasicBlock *, 8> UniquePreds(Preds.begin(), Preds.end());
|
||||
Updates.push_back({DominatorTree::Insert, NewBB, OldBB});
|
||||
Updates.reserve(Updates.size() + 2 * UniquePreds.size());
|
||||
for (auto *UniquePred : UniquePreds) {
|
||||
Updates.push_back({DominatorTree::Insert, UniquePred, NewBB});
|
||||
Updates.push_back({DominatorTree::Delete, UniquePred, OldBB});
|
||||
}
|
||||
DTU->applyUpdates(Updates);
|
||||
}
|
||||
} else if (DT) {
|
||||
if (OldBB == DT->getRootNode()->getBlock()) {
|
||||
assert(NewBB == &NewBB->getParent()->getEntryBlock());
|
||||
DT->setNewRoot(NewBB);
|
||||
|
@ -681,6 +701,8 @@ static void UpdateAnalysisInformation(BasicBlock *OldBB, BasicBlock *NewBB,
|
|||
if (!LI)
|
||||
return;
|
||||
|
||||
if (DTU && DTU->hasDomTree())
|
||||
DT = &DTU->getDomTree();
|
||||
assert(DT && "DT should be available to update LoopInfo!");
|
||||
Loop *L = LI->getLoopFor(OldBB);
|
||||
|
||||
|
@ -814,11 +836,17 @@ static void UpdatePHINodes(BasicBlock *OrigBB, BasicBlock *NewBB,
|
|||
}
|
||||
}
|
||||
|
||||
BasicBlock *llvm::SplitBlockPredecessors(BasicBlock *BB,
|
||||
ArrayRef<BasicBlock *> Preds,
|
||||
const char *Suffix, DominatorTree *DT,
|
||||
LoopInfo *LI, MemorySSAUpdater *MSSAU,
|
||||
bool PreserveLCSSA) {
|
||||
static void SplitLandingPadPredecessorsImpl(
|
||||
BasicBlock *OrigBB, ArrayRef<BasicBlock *> Preds, const char *Suffix1,
|
||||
const char *Suffix2, SmallVectorImpl<BasicBlock *> &NewBBs,
|
||||
DomTreeUpdater *DTU, DominatorTree *DT, LoopInfo *LI,
|
||||
MemorySSAUpdater *MSSAU, bool PreserveLCSSA);
|
||||
|
||||
static BasicBlock *
|
||||
SplitBlockPredecessorsImpl(BasicBlock *BB, ArrayRef<BasicBlock *> Preds,
|
||||
const char *Suffix, DomTreeUpdater *DTU,
|
||||
DominatorTree *DT, LoopInfo *LI,
|
||||
MemorySSAUpdater *MSSAU, bool PreserveLCSSA) {
|
||||
// Do not attempt to split that which cannot be split.
|
||||
if (!BB->canSplitPredecessors())
|
||||
return nullptr;
|
||||
|
@ -829,8 +857,8 @@ BasicBlock *llvm::SplitBlockPredecessors(BasicBlock *BB,
|
|||
SmallVector<BasicBlock*, 2> NewBBs;
|
||||
std::string NewName = std::string(Suffix) + ".split-lp";
|
||||
|
||||
SplitLandingPadPredecessors(BB, Preds, Suffix, NewName.c_str(), NewBBs, DT,
|
||||
LI, MSSAU, PreserveLCSSA);
|
||||
SplitLandingPadPredecessorsImpl(BB, Preds, Suffix, NewName.c_str(), NewBBs,
|
||||
DTU, DT, LI, MSSAU, PreserveLCSSA);
|
||||
return NewBBs[0];
|
||||
}
|
||||
|
||||
|
@ -882,7 +910,7 @@ BasicBlock *llvm::SplitBlockPredecessors(BasicBlock *BB,
|
|||
|
||||
// Update DominatorTree, LoopInfo, and LCCSA analysis information.
|
||||
bool HasLoopExit = false;
|
||||
UpdateAnalysisInformation(BB, NewBB, Preds, DT, LI, MSSAU, PreserveLCSSA,
|
||||
UpdateAnalysisInformation(BB, NewBB, Preds, DTU, DT, LI, MSSAU, PreserveLCSSA,
|
||||
HasLoopExit);
|
||||
|
||||
if (!Preds.empty()) {
|
||||
|
@ -902,13 +930,29 @@ BasicBlock *llvm::SplitBlockPredecessors(BasicBlock *BB,
|
|||
return NewBB;
|
||||
}
|
||||
|
||||
void llvm::SplitLandingPadPredecessors(BasicBlock *OrigBB,
|
||||
ArrayRef<BasicBlock *> Preds,
|
||||
const char *Suffix1, const char *Suffix2,
|
||||
SmallVectorImpl<BasicBlock *> &NewBBs,
|
||||
DominatorTree *DT, LoopInfo *LI,
|
||||
MemorySSAUpdater *MSSAU,
|
||||
bool PreserveLCSSA) {
|
||||
BasicBlock *llvm::SplitBlockPredecessors(BasicBlock *BB,
|
||||
ArrayRef<BasicBlock *> Preds,
|
||||
const char *Suffix, DominatorTree *DT,
|
||||
LoopInfo *LI, MemorySSAUpdater *MSSAU,
|
||||
bool PreserveLCSSA) {
|
||||
return SplitBlockPredecessorsImpl(BB, Preds, Suffix, /*DTU=*/nullptr, DT, LI,
|
||||
MSSAU, PreserveLCSSA);
|
||||
}
|
||||
BasicBlock *llvm::SplitBlockPredecessors(BasicBlock *BB,
|
||||
ArrayRef<BasicBlock *> Preds,
|
||||
const char *Suffix,
|
||||
DomTreeUpdater *DTU, LoopInfo *LI,
|
||||
MemorySSAUpdater *MSSAU,
|
||||
bool PreserveLCSSA) {
|
||||
return SplitBlockPredecessorsImpl(BB, Preds, Suffix, DTU,
|
||||
/*DT=*/nullptr, LI, MSSAU, PreserveLCSSA);
|
||||
}
|
||||
|
||||
static void SplitLandingPadPredecessorsImpl(
|
||||
BasicBlock *OrigBB, ArrayRef<BasicBlock *> Preds, const char *Suffix1,
|
||||
const char *Suffix2, SmallVectorImpl<BasicBlock *> &NewBBs,
|
||||
DomTreeUpdater *DTU, DominatorTree *DT, LoopInfo *LI,
|
||||
MemorySSAUpdater *MSSAU, bool PreserveLCSSA) {
|
||||
assert(OrigBB->isLandingPad() && "Trying to split a non-landing pad!");
|
||||
|
||||
// Create a new basic block for OrigBB's predecessors listed in Preds. Insert
|
||||
|
@ -933,8 +977,8 @@ void llvm::SplitLandingPadPredecessors(BasicBlock *OrigBB,
|
|||
}
|
||||
|
||||
bool HasLoopExit = false;
|
||||
UpdateAnalysisInformation(OrigBB, NewBB1, Preds, DT, LI, MSSAU, PreserveLCSSA,
|
||||
HasLoopExit);
|
||||
UpdateAnalysisInformation(OrigBB, NewBB1, Preds, DTU, DT, LI, MSSAU,
|
||||
PreserveLCSSA, HasLoopExit);
|
||||
|
||||
// Update the PHI nodes in OrigBB with the values coming from NewBB1.
|
||||
UpdatePHINodes(OrigBB, NewBB1, Preds, BI1, HasLoopExit);
|
||||
|
@ -969,7 +1013,7 @@ void llvm::SplitLandingPadPredecessors(BasicBlock *OrigBB,
|
|||
|
||||
// Update DominatorTree, LoopInfo, and LCCSA analysis information.
|
||||
HasLoopExit = false;
|
||||
UpdateAnalysisInformation(OrigBB, NewBB2, NewBB2Preds, DT, LI, MSSAU,
|
||||
UpdateAnalysisInformation(OrigBB, NewBB2, NewBB2Preds, DTU, DT, LI, MSSAU,
|
||||
PreserveLCSSA, HasLoopExit);
|
||||
|
||||
// Update the PHI nodes in OrigBB with the values coming from NewBB2.
|
||||
|
@ -1006,6 +1050,29 @@ void llvm::SplitLandingPadPredecessors(BasicBlock *OrigBB,
|
|||
}
|
||||
}
|
||||
|
||||
void llvm::SplitLandingPadPredecessors(BasicBlock *OrigBB,
|
||||
ArrayRef<BasicBlock *> Preds,
|
||||
const char *Suffix1, const char *Suffix2,
|
||||
SmallVectorImpl<BasicBlock *> &NewBBs,
|
||||
DominatorTree *DT, LoopInfo *LI,
|
||||
MemorySSAUpdater *MSSAU,
|
||||
bool PreserveLCSSA) {
|
||||
return SplitLandingPadPredecessorsImpl(
|
||||
OrigBB, Preds, Suffix1, Suffix2, NewBBs,
|
||||
/*DTU=*/nullptr, DT, LI, MSSAU, PreserveLCSSA);
|
||||
}
|
||||
void llvm::SplitLandingPadPredecessors(BasicBlock *OrigBB,
|
||||
ArrayRef<BasicBlock *> Preds,
|
||||
const char *Suffix1, const char *Suffix2,
|
||||
SmallVectorImpl<BasicBlock *> &NewBBs,
|
||||
DomTreeUpdater *DTU, LoopInfo *LI,
|
||||
MemorySSAUpdater *MSSAU,
|
||||
bool PreserveLCSSA) {
|
||||
return SplitLandingPadPredecessorsImpl(OrigBB, Preds, Suffix1, Suffix2,
|
||||
NewBBs, DTU, /*DT=*/nullptr, LI, MSSAU,
|
||||
PreserveLCSSA);
|
||||
}
|
||||
|
||||
ReturnInst *llvm::FoldReturnIntoUncondBranch(ReturnInst *RI, BasicBlock *BB,
|
||||
BasicBlock *Pred,
|
||||
DomTreeUpdater *DTU) {
|
||||
|
|
|
@ -1113,7 +1113,7 @@ bool SimplifyCFGOpt::FoldValueComparisonIntoPredecessors(Instruction *TI,
|
|||
if (!SafeToMergeTerminators(TI, PTI, &FailBlocks)) {
|
||||
for (auto *Succ : FailBlocks) {
|
||||
if (!SplitBlockPredecessors(Succ, TI->getParent(), ".fold.split",
|
||||
DTU ? &DTU->getDomTree() : nullptr))
|
||||
DTU))
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
@ -1977,8 +1977,7 @@ static bool SinkCommonCodeFromPredecessors(BasicBlock *BB,
|
|||
LLVM_DEBUG(dbgs() << "SINK: Splitting edge\n");
|
||||
// We have a conditional edge and we're going to sink some instructions.
|
||||
// Insert a new block postdominating all blocks we're going to sink from.
|
||||
if (!SplitBlockPredecessors(BB, UnconditionalPreds, ".sink.split",
|
||||
DTU ? &DTU->getDomTree() : nullptr))
|
||||
if (!SplitBlockPredecessors(BB, UnconditionalPreds, ".sink.split", DTU))
|
||||
// Edges couldn't be split.
|
||||
return false;
|
||||
Changed = true;
|
||||
|
@ -3391,8 +3390,7 @@ static bool mergeConditionalStoreToAddress(
|
|||
// branch to QFB and PostBB.
|
||||
BasicBlock *TruePred = QTB ? QTB : QFB->getSinglePredecessor();
|
||||
BasicBlock *NewBB =
|
||||
SplitBlockPredecessors(PostBB, {QFB, TruePred}, "condstore.split",
|
||||
DTU ? &DTU->getDomTree() : nullptr);
|
||||
SplitBlockPredecessors(PostBB, {QFB, TruePred}, "condstore.split", DTU);
|
||||
if (!NewBB)
|
||||
return false;
|
||||
PostBB = NewBB;
|
||||
|
@ -4827,9 +4825,8 @@ static void createUnreachableSwitchDefault(SwitchInst *Switch,
|
|||
DomTreeUpdater *DTU) {
|
||||
LLVM_DEBUG(dbgs() << "SimplifyCFG: switch default is dead.\n");
|
||||
auto *BB = Switch->getParent();
|
||||
BasicBlock *NewDefaultBlock =
|
||||
SplitBlockPredecessors(Switch->getDefaultDest(), Switch->getParent(), "",
|
||||
DTU ? &DTU->getDomTree() : nullptr);
|
||||
BasicBlock *NewDefaultBlock = SplitBlockPredecessors(
|
||||
Switch->getDefaultDest(), Switch->getParent(), "", DTU);
|
||||
auto *OrigDefaultBlock = Switch->getDefaultDest();
|
||||
Switch->setDefaultDest(&*NewDefaultBlock);
|
||||
if (DTU)
|
||||
|
|
Loading…
Reference in New Issue