mirror of
				https://github.com/c64scene-ar/llvm-6502.git
				synced 2025-11-03 14:21:30 +00:00 
			
		
		
		
	git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@213466 91177308-0d34-0410-b5e6-96231b3b80d8
		
			
				
	
	
		
			174 lines
		
	
	
		
			4.7 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			174 lines
		
	
	
		
			4.7 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
//===- RegionInfo.cpp - SESE region detection analysis --------------------===//
 | 
						|
//
 | 
						|
//                     The LLVM Compiler Infrastructure
 | 
						|
//
 | 
						|
// This file is distributed under the University of Illinois Open Source
 | 
						|
// License. See LICENSE.TXT for details.
 | 
						|
//
 | 
						|
//===----------------------------------------------------------------------===//
 | 
						|
// Detects single entry single exit regions in the control flow graph.
 | 
						|
//===----------------------------------------------------------------------===//
 | 
						|
 | 
						|
#include "llvm/Analysis/RegionInfo.h"
 | 
						|
#include "llvm/Analysis/RegionInfoImpl.h"
 | 
						|
#include "llvm/ADT/PostOrderIterator.h"
 | 
						|
#include "llvm/ADT/Statistic.h"
 | 
						|
#include "llvm/Analysis/LoopInfo.h"
 | 
						|
#include "llvm/Analysis/RegionIterator.h"
 | 
						|
#include "llvm/Support/CommandLine.h"
 | 
						|
#include "llvm/Support/Debug.h"
 | 
						|
#include "llvm/Support/ErrorHandling.h"
 | 
						|
#include <algorithm>
 | 
						|
#include <iterator>
 | 
						|
#include <set>
 | 
						|
 | 
						|
using namespace llvm;
 | 
						|
 | 
						|
#define DEBUG_TYPE "region"
 | 
						|
 | 
						|
namespace llvm {
 | 
						|
template class RegionBase<RegionTraits<Function>>;
 | 
						|
template class RegionNodeBase<RegionTraits<Function>>;
 | 
						|
template class RegionInfoBase<RegionTraits<Function>>;
 | 
						|
}
 | 
						|
 | 
						|
STATISTIC(numRegions,       "The # of regions");
 | 
						|
STATISTIC(numSimpleRegions, "The # of simple regions");
 | 
						|
 | 
						|
// Always verify if expensive checking is enabled.
 | 
						|
 | 
						|
static cl::opt<bool,true>
 | 
						|
VerifyRegionInfoX(
 | 
						|
  "verify-region-info",
 | 
						|
  cl::location(RegionInfoBase<RegionTraits<Function>>::VerifyRegionInfo),
 | 
						|
  cl::desc("Verify region info (time consuming)"));
 | 
						|
 | 
						|
 | 
						|
static cl::opt<Region::PrintStyle, true> printStyleX("print-region-style",
 | 
						|
  cl::location(RegionInfo::printStyle),
 | 
						|
  cl::Hidden,
 | 
						|
  cl::desc("style of printing regions"),
 | 
						|
  cl::values(
 | 
						|
    clEnumValN(Region::PrintNone, "none",  "print no details"),
 | 
						|
    clEnumValN(Region::PrintBB, "bb",
 | 
						|
               "print regions in detail with block_iterator"),
 | 
						|
    clEnumValN(Region::PrintRN, "rn",
 | 
						|
               "print regions in detail with element_iterator"),
 | 
						|
    clEnumValEnd));
 | 
						|
 | 
						|
 | 
						|
//===----------------------------------------------------------------------===//
 | 
						|
// Region implementation
 | 
						|
//
 | 
						|
 | 
						|
Region::Region(BasicBlock *Entry, BasicBlock *Exit,
 | 
						|
               RegionInfo* RI,
 | 
						|
               DominatorTree *DT, Region *Parent) :
 | 
						|
  RegionBase<RegionTraits<Function>>(Entry, Exit, RI, DT, Parent) {
 | 
						|
 | 
						|
}
 | 
						|
 | 
						|
Region::~Region() { }
 | 
						|
 | 
						|
//===----------------------------------------------------------------------===//
 | 
						|
// RegionInfo implementation
 | 
						|
//
 | 
						|
 | 
						|
RegionInfo::RegionInfo() :
 | 
						|
  RegionInfoBase<RegionTraits<Function>>() {
 | 
						|
 | 
						|
}
 | 
						|
 | 
						|
RegionInfo::~RegionInfo() {
 | 
						|
 | 
						|
}
 | 
						|
 | 
						|
void RegionInfo::updateStatistics(Region *R) {
 | 
						|
  ++numRegions;
 | 
						|
 | 
						|
  // TODO: Slow. Should only be enabled if -stats is used.
 | 
						|
  if (R->isSimple())
 | 
						|
    ++numSimpleRegions;
 | 
						|
}
 | 
						|
 | 
						|
void RegionInfo::recalculate(Function &F, DominatorTree *DT_,
 | 
						|
                             PostDominatorTree *PDT_, DominanceFrontier *DF_) {
 | 
						|
  DT = DT_;
 | 
						|
  PDT = PDT_;
 | 
						|
  DF = DF_;
 | 
						|
 | 
						|
  TopLevelRegion = new Region(&F.getEntryBlock(), nullptr,
 | 
						|
                              this, DT, nullptr);
 | 
						|
  updateStatistics(TopLevelRegion);
 | 
						|
  calculate(F);
 | 
						|
}
 | 
						|
 | 
						|
//===----------------------------------------------------------------------===//
 | 
						|
// RegionInfoPass implementation
 | 
						|
//
 | 
						|
 | 
						|
RegionInfoPass::RegionInfoPass() : FunctionPass(ID) {
 | 
						|
  initializeRegionInfoPassPass(*PassRegistry::getPassRegistry());
 | 
						|
}
 | 
						|
 | 
						|
RegionInfoPass::~RegionInfoPass() {
 | 
						|
 | 
						|
}
 | 
						|
 | 
						|
bool RegionInfoPass::runOnFunction(Function &F) {
 | 
						|
  releaseMemory();
 | 
						|
 | 
						|
  auto DT = &getAnalysis<DominatorTreeWrapperPass>().getDomTree();
 | 
						|
  auto PDT = &getAnalysis<PostDominatorTree>();
 | 
						|
  auto DF = &getAnalysis<DominanceFrontier>();
 | 
						|
 | 
						|
  RI.recalculate(F, DT, PDT, DF);
 | 
						|
  return false;
 | 
						|
}
 | 
						|
 | 
						|
void RegionInfoPass::releaseMemory() {
 | 
						|
  RI.releaseMemory();
 | 
						|
}
 | 
						|
 | 
						|
void RegionInfoPass::verifyAnalysis() const {
 | 
						|
    RI.verifyAnalysis();
 | 
						|
}
 | 
						|
 | 
						|
void RegionInfoPass::getAnalysisUsage(AnalysisUsage &AU) const {
 | 
						|
  AU.setPreservesAll();
 | 
						|
  AU.addRequiredTransitive<DominatorTreeWrapperPass>();
 | 
						|
  AU.addRequired<PostDominatorTree>();
 | 
						|
  AU.addRequired<DominanceFrontier>();
 | 
						|
}
 | 
						|
 | 
						|
void RegionInfoPass::print(raw_ostream &OS, const Module *) const {
 | 
						|
  RI.print(OS);
 | 
						|
}
 | 
						|
 | 
						|
#if !defined(NDEBUG) || defined(LLVM_ENABLE_DUMP)
 | 
						|
void RegionInfoPass::dump() const {
 | 
						|
  RI.dump();
 | 
						|
}
 | 
						|
#endif
 | 
						|
 | 
						|
char RegionInfoPass::ID = 0;
 | 
						|
 | 
						|
INITIALIZE_PASS_BEGIN(RegionInfoPass, "regions",
 | 
						|
                "Detect single entry single exit regions", true, true)
 | 
						|
INITIALIZE_PASS_DEPENDENCY(DominatorTreeWrapperPass)
 | 
						|
INITIALIZE_PASS_DEPENDENCY(PostDominatorTree)
 | 
						|
INITIALIZE_PASS_DEPENDENCY(DominanceFrontier)
 | 
						|
INITIALIZE_PASS_END(RegionInfoPass, "regions",
 | 
						|
                "Detect single entry single exit regions", true, true)
 | 
						|
 | 
						|
// Create methods available outside of this file, to use them
 | 
						|
// "include/llvm/LinkAllPasses.h". Otherwise the pass would be deleted by
 | 
						|
// the link time optimization.
 | 
						|
 | 
						|
namespace llvm {
 | 
						|
  FunctionPass *createRegionInfoPass() {
 | 
						|
    return new RegionInfoPass();
 | 
						|
  }
 | 
						|
}
 | 
						|
 |