530 lines
		
	
	
		
			15 KiB
		
	
	
	
		
			C++
		
	
	
	
			
		
		
	
	
			530 lines
		
	
	
		
			15 KiB
		
	
	
	
		
			C++
		
	
	
	
| //===- DomTreeUpdater.cpp - DomTree/Post DomTree Updater --------*- C++ -*-===//
 | |
| //
 | |
| //                     The LLVM Compiler Infrastructure
 | |
| //
 | |
| // This file is distributed under the University of Illinois Open Source
 | |
| // License. See LICENSE.TXT for details.
 | |
| //
 | |
| //===----------------------------------------------------------------------===//
 | |
| //
 | |
| // This file implements the DomTreeUpdater class, which provides a uniform way
 | |
| // to update dominator tree related data structures.
 | |
| //
 | |
| //===----------------------------------------------------------------------===//
 | |
| 
 | |
| #include "llvm/IR/DomTreeUpdater.h"
 | |
| #include "llvm/Analysis/PostDominators.h"
 | |
| #include "llvm/IR/Dominators.h"
 | |
| #include "llvm/Support/GenericDomTree.h"
 | |
| #include <algorithm>
 | |
| #include <functional>
 | |
| 
 | |
| namespace llvm {
 | |
| 
 | |
| bool DomTreeUpdater::isUpdateValid(
 | |
|     const DominatorTree::UpdateType Update) const {
 | |
|   const auto *From = Update.getFrom();
 | |
|   const auto *To = Update.getTo();
 | |
|   const auto Kind = Update.getKind();
 | |
| 
 | |
|   // Discard updates by inspecting the current state of successors of From.
 | |
|   // Since isUpdateValid() must be called *after* the Terminator of From is
 | |
|   // altered we can determine if the update is unnecessary for batch updates
 | |
|   // or invalid for a single update.
 | |
|   const bool HasEdge = llvm::any_of(
 | |
|       successors(From), [To](const BasicBlock *B) { return B == To; });
 | |
| 
 | |
|   // If the IR does not match the update,
 | |
|   // 1. In batch updates, this update is unnecessary.
 | |
|   // 2. When called by insertEdge*()/deleteEdge*(), this update is invalid.
 | |
|   // Edge does not exist in IR.
 | |
|   if (Kind == DominatorTree::Insert && !HasEdge)
 | |
|     return false;
 | |
| 
 | |
|   // Edge exists in IR.
 | |
|   if (Kind == DominatorTree::Delete && HasEdge)
 | |
|     return false;
 | |
| 
 | |
|   return true;
 | |
| }
 | |
| 
 | |
| bool DomTreeUpdater::isSelfDominance(
 | |
|     const DominatorTree::UpdateType Update) const {
 | |
|   // Won't affect DomTree and PostDomTree.
 | |
|   return Update.getFrom() == Update.getTo();
 | |
| }
 | |
| 
 | |
| bool DomTreeUpdater::applyLazyUpdate(DominatorTree::UpdateKind Kind,
 | |
|                                      BasicBlock *From, BasicBlock *To) {
 | |
|   assert((DT || PDT) &&
 | |
|          "Call applyLazyUpdate() when both DT and PDT are nullptrs.");
 | |
|   assert(Strategy == DomTreeUpdater::UpdateStrategy::Lazy &&
 | |
|          "Call applyLazyUpdate() with Eager strategy error");
 | |
|   // Analyze pending updates to determine if the update is unnecessary.
 | |
|   const DominatorTree::UpdateType Update = {Kind, From, To};
 | |
|   const DominatorTree::UpdateType Invert = {Kind != DominatorTree::Insert
 | |
|                                                 ? DominatorTree::Insert
 | |
|                                                 : DominatorTree::Delete,
 | |
|                                             From, To};
 | |
|   // Only check duplicates in updates that are not applied by both trees.
 | |
|   auto I =
 | |
|       PendUpdates.begin() + std::max(PendDTUpdateIndex, PendPDTUpdateIndex);
 | |
|   const auto E = PendUpdates.end();
 | |
| 
 | |
|   assert(I <= E && "Iterator out of range.");
 | |
| 
 | |
|   for (; I != E; ++I) {
 | |
|     if (Update == *I)
 | |
|       return false; // Discard duplicate updates.
 | |
| 
 | |
|     if (Invert == *I) {
 | |
|       // Update and Invert are both valid (equivalent to a no-op). Remove
 | |
|       // Invert from PendUpdates and discard the Update.
 | |
|       PendUpdates.erase(I);
 | |
|       return false;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   PendUpdates.push_back(Update); // Save the valid update.
 | |
|   return true;
 | |
| }
 | |
| 
 | |
| void DomTreeUpdater::applyDomTreeUpdates() {
 | |
|   // No pending DomTreeUpdates.
 | |
|   if (Strategy != UpdateStrategy::Lazy || !DT)
 | |
|     return;
 | |
| 
 | |
|   // Only apply updates not are applied by DomTree.
 | |
|   if (hasPendingDomTreeUpdates()) {
 | |
|     const auto I = PendUpdates.begin() + PendDTUpdateIndex;
 | |
|     const auto E = PendUpdates.end();
 | |
|     assert(I < E && "Iterator range invalid; there should be DomTree updates.");
 | |
|     DT->applyUpdates(ArrayRef<DominatorTree::UpdateType>(I, E));
 | |
|     PendDTUpdateIndex = PendUpdates.size();
 | |
|   }
 | |
| }
 | |
| 
 | |
| void DomTreeUpdater::flush() {
 | |
|   applyDomTreeUpdates();
 | |
|   applyPostDomTreeUpdates();
 | |
|   dropOutOfDateUpdates();
 | |
| }
 | |
| 
 | |
| void DomTreeUpdater::applyPostDomTreeUpdates() {
 | |
|   // No pending PostDomTreeUpdates.
 | |
|   if (Strategy != UpdateStrategy::Lazy || !PDT)
 | |
|     return;
 | |
| 
 | |
|   // Only apply updates not are applied by PostDomTree.
 | |
|   if (hasPendingPostDomTreeUpdates()) {
 | |
|     const auto I = PendUpdates.begin() + PendPDTUpdateIndex;
 | |
|     const auto E = PendUpdates.end();
 | |
|     assert(I < E &&
 | |
|            "Iterator range invalid; there should be PostDomTree updates.");
 | |
|     PDT->applyUpdates(ArrayRef<DominatorTree::UpdateType>(I, E));
 | |
|     PendPDTUpdateIndex = PendUpdates.size();
 | |
|   }
 | |
| }
 | |
| 
 | |
| void DomTreeUpdater::tryFlushDeletedBB() {
 | |
|   if (!hasPendingUpdates())
 | |
|     forceFlushDeletedBB();
 | |
| }
 | |
| 
 | |
| bool DomTreeUpdater::forceFlushDeletedBB() {
 | |
|   if (DeletedBBs.empty())
 | |
|     return false;
 | |
| 
 | |
|   for (auto *BB : DeletedBBs) {
 | |
|     // After calling deleteBB or callbackDeleteBB under Lazy UpdateStrategy,
 | |
|     // validateDeleteBB() removes all instructions of DelBB and adds an
 | |
|     // UnreachableInst as its terminator. So we check whether the BasicBlock to
 | |
|     // delete only has an UnreachableInst inside.
 | |
|     assert(BB->getInstList().size() == 1 &&
 | |
|            isa<UnreachableInst>(BB->getTerminator()) &&
 | |
|            "DelBB has been modified while awaiting deletion.");
 | |
|     BB->removeFromParent();
 | |
|     eraseDelBBNode(BB);
 | |
|     delete BB;
 | |
|   }
 | |
|   DeletedBBs.clear();
 | |
|   Callbacks.clear();
 | |
|   return true;
 | |
| }
 | |
| 
 | |
| void DomTreeUpdater::recalculate(Function &F) {
 | |
| 
 | |
|   if (Strategy == UpdateStrategy::Eager) {
 | |
|     if (DT)
 | |
|       DT->recalculate(F);
 | |
|     if (PDT)
 | |
|       PDT->recalculate(F);
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   // There is little performance gain if we pend the recalculation under
 | |
|   // Lazy UpdateStrategy so we recalculate available trees immediately.
 | |
| 
 | |
|   // Prevent forceFlushDeletedBB() from erasing DomTree or PostDomTree nodes.
 | |
|   IsRecalculatingDomTree = IsRecalculatingPostDomTree = true;
 | |
| 
 | |
|   // Because all trees are going to be up-to-date after recalculation,
 | |
|   // flush awaiting deleted BasicBlocks.
 | |
|   forceFlushDeletedBB();
 | |
|   if (DT)
 | |
|     DT->recalculate(F);
 | |
|   if (PDT)
 | |
|     PDT->recalculate(F);
 | |
| 
 | |
|   // Resume forceFlushDeletedBB() to erase DomTree or PostDomTree nodes.
 | |
|   IsRecalculatingDomTree = IsRecalculatingPostDomTree = false;
 | |
|   PendDTUpdateIndex = PendPDTUpdateIndex = PendUpdates.size();
 | |
|   dropOutOfDateUpdates();
 | |
| }
 | |
| 
 | |
| bool DomTreeUpdater::hasPendingUpdates() const {
 | |
|   return hasPendingDomTreeUpdates() || hasPendingPostDomTreeUpdates();
 | |
| }
 | |
| 
 | |
| bool DomTreeUpdater::hasPendingDomTreeUpdates() const {
 | |
|   if (!DT)
 | |
|     return false;
 | |
|   return PendUpdates.size() != PendDTUpdateIndex;
 | |
| }
 | |
| 
 | |
| bool DomTreeUpdater::hasPendingPostDomTreeUpdates() const {
 | |
|   if (!PDT)
 | |
|     return false;
 | |
|   return PendUpdates.size() != PendPDTUpdateIndex;
 | |
| }
 | |
| 
 | |
| bool DomTreeUpdater::isBBPendingDeletion(llvm::BasicBlock *DelBB) const {
 | |
|   if (Strategy == UpdateStrategy::Eager || DeletedBBs.empty())
 | |
|     return false;
 | |
|   return DeletedBBs.count(DelBB) != 0;
 | |
| }
 | |
| 
 | |
| // The DT and PDT require the nodes related to updates
 | |
| // are not deleted when update functions are called.
 | |
| // So BasicBlock deletions must be pended when the
 | |
| // UpdateStrategy is Lazy. When the UpdateStrategy is
 | |
| // Eager, the BasicBlock will be deleted immediately.
 | |
| void DomTreeUpdater::deleteBB(BasicBlock *DelBB) {
 | |
|   validateDeleteBB(DelBB);
 | |
|   if (Strategy == UpdateStrategy::Lazy) {
 | |
|     DeletedBBs.insert(DelBB);
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   DelBB->removeFromParent();
 | |
|   eraseDelBBNode(DelBB);
 | |
|   delete DelBB;
 | |
| }
 | |
| 
 | |
| void DomTreeUpdater::callbackDeleteBB(
 | |
|     BasicBlock *DelBB, std::function<void(BasicBlock *)> Callback) {
 | |
|   validateDeleteBB(DelBB);
 | |
|   if (Strategy == UpdateStrategy::Lazy) {
 | |
|     Callbacks.push_back(CallBackOnDeletion(DelBB, Callback));
 | |
|     DeletedBBs.insert(DelBB);
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   DelBB->removeFromParent();
 | |
|   eraseDelBBNode(DelBB);
 | |
|   Callback(DelBB);
 | |
|   delete DelBB;
 | |
| }
 | |
| 
 | |
| void DomTreeUpdater::eraseDelBBNode(BasicBlock *DelBB) {
 | |
|   if (DT && !IsRecalculatingDomTree)
 | |
|     if (DT->getNode(DelBB))
 | |
|       DT->eraseNode(DelBB);
 | |
| 
 | |
|   if (PDT && !IsRecalculatingPostDomTree)
 | |
|     if (PDT->getNode(DelBB))
 | |
|       PDT->eraseNode(DelBB);
 | |
| }
 | |
| 
 | |
| void DomTreeUpdater::validateDeleteBB(BasicBlock *DelBB) {
 | |
|   assert(DelBB && "Invalid push_back of nullptr DelBB.");
 | |
|   assert(pred_empty(DelBB) && "DelBB has one or more predecessors.");
 | |
|   // DelBB is unreachable and all its instructions are dead.
 | |
|   while (!DelBB->empty()) {
 | |
|     Instruction &I = DelBB->back();
 | |
|     // Replace used instructions with an arbitrary value (undef).
 | |
|     if (!I.use_empty())
 | |
|       I.replaceAllUsesWith(llvm::UndefValue::get(I.getType()));
 | |
|     DelBB->getInstList().pop_back();
 | |
|   }
 | |
|   // Make sure DelBB has a valid terminator instruction. As long as DelBB is a
 | |
|   // Child of Function F it must contain valid IR.
 | |
|   new UnreachableInst(DelBB->getContext(), DelBB);
 | |
| }
 | |
| 
 | |
| void DomTreeUpdater::applyUpdates(ArrayRef<DominatorTree::UpdateType> Updates,
 | |
|                                   bool ForceRemoveDuplicates) {
 | |
|   if (!DT && !PDT)
 | |
|     return;
 | |
| 
 | |
|   if (Strategy == UpdateStrategy::Lazy || ForceRemoveDuplicates) {
 | |
|     SmallVector<DominatorTree::UpdateType, 8> Seen;
 | |
|     for (const auto U : Updates)
 | |
|       // For Lazy UpdateStrategy, avoid duplicates to applyLazyUpdate() to save
 | |
|       // on analysis.
 | |
|       if (llvm::none_of(
 | |
|               Seen,
 | |
|               [U](const DominatorTree::UpdateType S) { return S == U; }) &&
 | |
|           isUpdateValid(U) && !isSelfDominance(U)) {
 | |
|         Seen.push_back(U);
 | |
|         if (Strategy == UpdateStrategy::Lazy)
 | |
|           applyLazyUpdate(U.getKind(), U.getFrom(), U.getTo());
 | |
|       }
 | |
|     if (Strategy == UpdateStrategy::Lazy)
 | |
|       return;
 | |
| 
 | |
|     if (DT)
 | |
|       DT->applyUpdates(Seen);
 | |
|     if (PDT)
 | |
|       PDT->applyUpdates(Seen);
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   if (DT)
 | |
|     DT->applyUpdates(Updates);
 | |
|   if (PDT)
 | |
|     PDT->applyUpdates(Updates);
 | |
| }
 | |
| 
 | |
| DominatorTree &DomTreeUpdater::getDomTree() {
 | |
|   assert(DT && "Invalid acquisition of a null DomTree");
 | |
|   applyDomTreeUpdates();
 | |
|   dropOutOfDateUpdates();
 | |
|   return *DT;
 | |
| }
 | |
| 
 | |
| PostDominatorTree &DomTreeUpdater::getPostDomTree() {
 | |
|   assert(PDT && "Invalid acquisition of a null PostDomTree");
 | |
|   applyPostDomTreeUpdates();
 | |
|   dropOutOfDateUpdates();
 | |
|   return *PDT;
 | |
| }
 | |
| 
 | |
| void DomTreeUpdater::insertEdge(BasicBlock *From, BasicBlock *To) {
 | |
| 
 | |
| #ifndef NDEBUG
 | |
|   assert(isUpdateValid({DominatorTree::Insert, From, To}) &&
 | |
|          "Inserted edge does not appear in the CFG");
 | |
| #endif
 | |
| 
 | |
|   if (!DT && !PDT)
 | |
|     return;
 | |
| 
 | |
|   // Won't affect DomTree and PostDomTree; discard update.
 | |
|   if (From == To)
 | |
|     return;
 | |
| 
 | |
|   if (Strategy == UpdateStrategy::Eager) {
 | |
|     if (DT)
 | |
|       DT->insertEdge(From, To);
 | |
|     if (PDT)
 | |
|       PDT->insertEdge(From, To);
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   applyLazyUpdate(DominatorTree::Insert, From, To);
 | |
| }
 | |
| 
 | |
| void DomTreeUpdater::insertEdgeRelaxed(BasicBlock *From, BasicBlock *To) {
 | |
|   if (From == To)
 | |
|     return;
 | |
| 
 | |
|   if (!DT && !PDT)
 | |
|     return;
 | |
| 
 | |
|   if (!isUpdateValid({DominatorTree::Insert, From, To}))
 | |
|     return;
 | |
| 
 | |
|   if (Strategy == UpdateStrategy::Eager) {
 | |
|     if (DT)
 | |
|       DT->insertEdge(From, To);
 | |
|     if (PDT)
 | |
|       PDT->insertEdge(From, To);
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   applyLazyUpdate(DominatorTree::Insert, From, To);
 | |
| }
 | |
| 
 | |
| void DomTreeUpdater::deleteEdge(BasicBlock *From, BasicBlock *To) {
 | |
| 
 | |
| #ifndef NDEBUG
 | |
|   assert(isUpdateValid({DominatorTree::Delete, From, To}) &&
 | |
|          "Deleted edge still exists in the CFG!");
 | |
| #endif
 | |
| 
 | |
|   if (!DT && !PDT)
 | |
|     return;
 | |
| 
 | |
|   // Won't affect DomTree and PostDomTree; discard update.
 | |
|   if (From == To)
 | |
|     return;
 | |
| 
 | |
|   if (Strategy == UpdateStrategy::Eager) {
 | |
|     if (DT)
 | |
|       DT->deleteEdge(From, To);
 | |
|     if (PDT)
 | |
|       PDT->deleteEdge(From, To);
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   applyLazyUpdate(DominatorTree::Delete, From, To);
 | |
| }
 | |
| 
 | |
| void DomTreeUpdater::deleteEdgeRelaxed(BasicBlock *From, BasicBlock *To) {
 | |
|   if (From == To)
 | |
|     return;
 | |
| 
 | |
|   if (!DT && !PDT)
 | |
|     return;
 | |
| 
 | |
|   if (!isUpdateValid({DominatorTree::Delete, From, To}))
 | |
|     return;
 | |
| 
 | |
|   if (Strategy == UpdateStrategy::Eager) {
 | |
|     if (DT)
 | |
|       DT->deleteEdge(From, To);
 | |
|     if (PDT)
 | |
|       PDT->deleteEdge(From, To);
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   applyLazyUpdate(DominatorTree::Delete, From, To);
 | |
| }
 | |
| 
 | |
| void DomTreeUpdater::dropOutOfDateUpdates() {
 | |
|   if (Strategy == DomTreeUpdater::UpdateStrategy::Eager)
 | |
|     return;
 | |
| 
 | |
|   tryFlushDeletedBB();
 | |
| 
 | |
|   // Drop all updates applied by both trees.
 | |
|   if (!DT)
 | |
|     PendDTUpdateIndex = PendUpdates.size();
 | |
|   if (!PDT)
 | |
|     PendPDTUpdateIndex = PendUpdates.size();
 | |
| 
 | |
|   const size_t dropIndex = std::min(PendDTUpdateIndex, PendPDTUpdateIndex);
 | |
|   const auto B = PendUpdates.begin();
 | |
|   const auto E = PendUpdates.begin() + dropIndex;
 | |
|   assert(B <= E && "Iterator out of range.");
 | |
|   PendUpdates.erase(B, E);
 | |
|   // Calculate current index.
 | |
|   PendDTUpdateIndex -= dropIndex;
 | |
|   PendPDTUpdateIndex -= dropIndex;
 | |
| }
 | |
| 
 | |
| #if !defined(NDEBUG) || defined(LLVM_ENABLE_DUMP)
 | |
| LLVM_DUMP_METHOD void DomTreeUpdater::dump() const {
 | |
|   raw_ostream &OS = llvm::dbgs();
 | |
| 
 | |
|   OS << "Available Trees: ";
 | |
|   if (DT || PDT) {
 | |
|     if (DT)
 | |
|       OS << "DomTree ";
 | |
|     if (PDT)
 | |
|       OS << "PostDomTree ";
 | |
|     OS << "\n";
 | |
|   } else
 | |
|     OS << "None\n";
 | |
| 
 | |
|   OS << "UpdateStrategy: ";
 | |
|   if (Strategy == UpdateStrategy::Eager) {
 | |
|     OS << "Eager\n";
 | |
|     return;
 | |
|   } else
 | |
|     OS << "Lazy\n";
 | |
|   int Index = 0;
 | |
| 
 | |
|   auto printUpdates =
 | |
|       [&](ArrayRef<DominatorTree::UpdateType>::const_iterator begin,
 | |
|           ArrayRef<DominatorTree::UpdateType>::const_iterator end) {
 | |
|         if (begin == end)
 | |
|           OS << "  None\n";
 | |
|         Index = 0;
 | |
|         for (auto It = begin, ItEnd = end; It != ItEnd; ++It) {
 | |
|           auto U = *It;
 | |
|           OS << "  " << Index << " : ";
 | |
|           ++Index;
 | |
|           if (U.getKind() == DominatorTree::Insert)
 | |
|             OS << "Insert, ";
 | |
|           else
 | |
|             OS << "Delete, ";
 | |
|           BasicBlock *From = U.getFrom();
 | |
|           if (From) {
 | |
|             auto S = From->getName();
 | |
|             if (!From->hasName())
 | |
|               S = "(no name)";
 | |
|             OS << S << "(" << From << "), ";
 | |
|           } else {
 | |
|             OS << "(badref), ";
 | |
|           }
 | |
|           BasicBlock *To = U.getTo();
 | |
|           if (To) {
 | |
|             auto S = To->getName();
 | |
|             if (!To->hasName())
 | |
|               S = "(no_name)";
 | |
|             OS << S << "(" << To << ")\n";
 | |
|           } else {
 | |
|             OS << "(badref)\n";
 | |
|           }
 | |
|         }
 | |
|       };
 | |
| 
 | |
|   if (DT) {
 | |
|     const auto I = PendUpdates.begin() + PendDTUpdateIndex;
 | |
|     assert(PendUpdates.begin() <= I && I <= PendUpdates.end() &&
 | |
|            "Iterator out of range.");
 | |
|     OS << "Applied but not cleared DomTreeUpdates:\n";
 | |
|     printUpdates(PendUpdates.begin(), I);
 | |
|     OS << "Pending DomTreeUpdates:\n";
 | |
|     printUpdates(I, PendUpdates.end());
 | |
|   }
 | |
| 
 | |
|   if (PDT) {
 | |
|     const auto I = PendUpdates.begin() + PendPDTUpdateIndex;
 | |
|     assert(PendUpdates.begin() <= I && I <= PendUpdates.end() &&
 | |
|            "Iterator out of range.");
 | |
|     OS << "Applied but not cleared PostDomTreeUpdates:\n";
 | |
|     printUpdates(PendUpdates.begin(), I);
 | |
|     OS << "Pending PostDomTreeUpdates:\n";
 | |
|     printUpdates(I, PendUpdates.end());
 | |
|   }
 | |
| 
 | |
|   OS << "Pending DeletedBBs:\n";
 | |
|   Index = 0;
 | |
|   for (auto BB : DeletedBBs) {
 | |
|     OS << "  " << Index << " : ";
 | |
|     ++Index;
 | |
|     if (BB->hasName())
 | |
|       OS << BB->getName() << "(";
 | |
|     else
 | |
|       OS << "(no_name)(";
 | |
|     OS << BB << ")\n";
 | |
|   }
 | |
| 
 | |
|   OS << "Pending Callbacks:\n";
 | |
|   Index = 0;
 | |
|   for (auto BB : Callbacks) {
 | |
|     OS << "  " << Index << " : ";
 | |
|     ++Index;
 | |
|     if (BB->hasName())
 | |
|       OS << BB->getName() << "(";
 | |
|     else
 | |
|       OS << "(no_name)(";
 | |
|     OS << BB << ")\n";
 | |
|   }
 | |
| }
 | |
| #endif
 | |
| } // namespace llvm
 |