//===----- ScopDetection.cpp - Detect Scops --------------------*- C++ -*-===// // // The LLVM Compiler Infrastructure // // This file is distributed under the University of Illinois Open Source // License. See LICENSE.TXT for details. // //===----------------------------------------------------------------------===// // // Detect the maximal Scops of a function. // // A static control part (Scop) is a subgraph of the control flow graph (CFG) // that only has statically known control flow and can therefore be described // within the polyhedral model. // // Every Scop fullfills these restrictions: // // * It is a single entry single exit region // // * Only affine linear bounds in the loops // // Every natural loop in a Scop must have a number of loop iterations that can // be described as an affine linear function in surrounding loop iterators or // parameters. (A parameter is a scalar that does not change its value during // execution of the Scop). // // * Only comparisons of affine linear expressions in conditions // // * All loops and conditions perfectly nested // // The control flow needs to be structured such that it could be written using // just 'for' and 'if' statements, without the need for any 'goto', 'break' or // 'continue'. // // * Side effect free functions call // // Only function calls and intrinsics that do not have side effects are allowed // (readnone). // // The Scop detection finds the largest Scops by checking if the largest // region is a Scop. If this is not the case, its canonical subregions are // checked until a region is a Scop. It is now tried to extend this Scop by // creating a larger non canonical region. // //===----------------------------------------------------------------------===// #include "polly/CodeGen/BlockGenerators.h" #include "polly/LinkAllPasses.h" #include "polly/Options.h" #include "polly/ScopDetectionDiagnostic.h" #include "polly/ScopDetection.h" #include "polly/Support/SCEVValidator.h" #include "polly/Support/ScopHelper.h" #include "llvm/ADT/Statistic.h" #include "llvm/Analysis/AliasAnalysis.h" #include "llvm/Analysis/LoopInfo.h" #include "llvm/Analysis/RegionIterator.h" #include "llvm/Analysis/ScalarEvolution.h" #include "llvm/Analysis/ScalarEvolutionExpressions.h" #include "llvm/IR/DebugInfo.h" #include "llvm/IR/DiagnosticInfo.h" #include "llvm/IR/DiagnosticPrinter.h" #include "llvm/IR/LLVMContext.h" #include "llvm/Support/Debug.h" #include using namespace llvm; using namespace polly; #define DEBUG_TYPE "polly-detect" static cl::opt DetectScopsWithoutLoops("polly-detect-scops-in-functions-without-loops", cl::desc("Detect scops in functions without loops"), cl::Hidden, cl::init(false), cl::ZeroOrMore, cl::cat(PollyCategory)); static cl::opt DetectRegionsWithoutLoops("polly-detect-scops-in-regions-without-loops", cl::desc("Detect scops in regions without loops"), cl::Hidden, cl::init(false), cl::ZeroOrMore, cl::cat(PollyCategory)); static cl::opt OnlyFunction("polly-only-func", cl::desc("Only run on functions that contain a certain string"), cl::value_desc("string"), cl::ValueRequired, cl::init(""), cl::cat(PollyCategory)); static cl::opt OnlyRegion("polly-only-region", cl::desc("Only run on certain regions (The provided identifier must " "appear in the name of the region's entry block"), cl::value_desc("identifier"), cl::ValueRequired, cl::init(""), cl::cat(PollyCategory)); static cl::opt IgnoreAliasing("polly-ignore-aliasing", cl::desc("Ignore possible aliasing of the array bases"), cl::Hidden, cl::init(false), cl::ZeroOrMore, cl::cat(PollyCategory)); static cl::opt ReportLevel("polly-report", cl::desc("Print information about the activities of Polly"), cl::init(false), cl::ZeroOrMore, cl::cat(PollyCategory)); static cl::opt AllowNonAffine("polly-allow-nonaffine", cl::desc("Allow non affine access functions in arrays"), cl::Hidden, cl::init(false), cl::ZeroOrMore, cl::cat(PollyCategory)); static cl::opt TrackFailures("polly-detect-track-failures", cl::desc("Track failure strings in detecting scop regions"), cl::location(PollyTrackFailures), cl::Hidden, cl::ZeroOrMore, cl::init(false), cl::cat(PollyCategory)); static cl::opt PollyDelinearizeX("polly-delinearize", cl::desc("Delinearize array access functions"), cl::location(PollyDelinearize), cl::Hidden, cl::ZeroOrMore, cl::init(false), cl::cat(PollyCategory)); static cl::opt VerifyScops("polly-detect-verify", cl::desc("Verify the detected SCoPs after each transformation"), cl::Hidden, cl::init(false), cl::ZeroOrMore, cl::cat(PollyCategory)); bool polly::PollyTrackFailures = false; bool polly::PollyDelinearize = false; //===----------------------------------------------------------------------===// // Statistics. STATISTIC(ValidRegion, "Number of regions that a valid part of Scop"); class DiagnosticScopFound : public DiagnosticInfo { private: static int PluginDiagnosticKind; Function &F; std::string FileName; unsigned EntryLine, ExitLine; public: DiagnosticScopFound(Function &F, std::string FileName, unsigned EntryLine, unsigned ExitLine) : DiagnosticInfo(PluginDiagnosticKind, DS_Note), F(F), FileName(FileName), EntryLine(EntryLine), ExitLine(ExitLine) {} virtual void print(DiagnosticPrinter &DP) const; static bool classof(const DiagnosticInfo *DI) { return DI->getKind() == PluginDiagnosticKind; } }; int DiagnosticScopFound::PluginDiagnosticKind = 10; void DiagnosticScopFound::print(DiagnosticPrinter &DP) const { DP << "Polly detected an optimizable loop region (scop) in function '" << F << "'\n"; if (FileName.empty()) { DP << "Scop location is unknown. Compile with debug info " "(-g) to get more precise information. "; return; } DP << FileName << ":" << EntryLine << ": Start of scop\n"; DP << FileName << ":" << ExitLine << ": End of scop"; } //===----------------------------------------------------------------------===// // ScopDetection. template inline bool ScopDetection::invalid(DetectionContext &Context, bool Assert, Args &&... Arguments) const { if (!Context.Verifying) { RR RejectReason = RR(Arguments...); if (PollyTrackFailures) LastFailure = RejectReason.getMessage(); DEBUG(dbgs() << RejectReason.getMessage()); DEBUG(dbgs() << "\n"); } else { assert(!Assert && "Verification of detected scop failed"); } return false; } bool ScopDetection::isMaxRegionInScop(const Region &R, bool Verify) const { if (!ValidRegions.count(&R)) return false; if (Verify) return isValidRegion(const_cast(R)); return true; } std::string ScopDetection::regionIsInvalidBecause(const Region *R) const { if (!InvalidRegions.count(R)) return ""; return InvalidRegions.find(R)->second; } bool ScopDetection::isValidCFG(BasicBlock &BB, DetectionContext &Context) const { Region &RefRegion = Context.CurRegion; TerminatorInst *TI = BB.getTerminator(); // Return instructions are only valid if the region is the top level region. if (isa(TI) && !RefRegion.getExit() && TI->getNumOperands() == 0) return true; BranchInst *Br = dyn_cast(TI); if (!Br) return invalid(Context, /*Assert=*/true, &BB); if (Br->isUnconditional()) return true; Value *Condition = Br->getCondition(); // UndefValue is not allowed as condition. if (isa(Condition)) return invalid(Context, /*Assert=*/true, &BB); // Only Constant and ICmpInst are allowed as condition. if (!(isa(Condition) || isa(Condition))) return invalid(Context, /*Assert=*/true, &BB); // Allow perfectly nested conditions. assert(Br->getNumSuccessors() == 2 && "Unexpected number of successors"); if (ICmpInst *ICmp = dyn_cast(Condition)) { // Unsigned comparisons are not allowed. They trigger overflow problems // in the code generation. // // TODO: This is not sufficient and just hides bugs. However it does pretty // well. if (ICmp->isUnsigned()) return false; // Are both operands of the ICmp affine? if (isa(ICmp->getOperand(0)) || isa(ICmp->getOperand(1))) return invalid(Context, /*Assert=*/true, &BB); Loop *L = LI->getLoopFor(ICmp->getParent()); const SCEV *LHS = SE->getSCEVAtScope(ICmp->getOperand(0), L); const SCEV *RHS = SE->getSCEVAtScope(ICmp->getOperand(1), L); if (!isAffineExpr(&Context.CurRegion, LHS, *SE) || !isAffineExpr(&Context.CurRegion, RHS, *SE)) return invalid(Context, /*Assert=*/true, &BB, LHS, RHS); } // Allow loop exit conditions. Loop *L = LI->getLoopFor(&BB); if (L && L->getExitingBlock() == &BB) return true; // Allow perfectly nested conditions. Region *R = RI->getRegionFor(&BB); if (R->getEntry() != &BB) return invalid(Context, /*Assert=*/true, &BB); return true; } bool ScopDetection::isValidCallInst(CallInst &CI) { if (CI.mayHaveSideEffects() || CI.doesNotReturn()) return false; if (CI.doesNotAccessMemory()) return true; Function *CalledFunction = CI.getCalledFunction(); // Indirect calls are not supported. if (CalledFunction == 0) return false; // TODO: Intrinsics. return false; } bool ScopDetection::isInvariant(const Value &Val, const Region &Reg) const { // A reference to function argument or constant value is invariant. if (isa(Val) || isa(Val)) return true; const Instruction *I = dyn_cast(&Val); if (!I) return false; if (!Reg.contains(I)) return true; if (I->mayHaveSideEffects()) return false; // When Val is a Phi node, it is likely not invariant. We do not check whether // Phi nodes are actually invariant, we assume that Phi nodes are usually not // invariant. Recursively checking the operators of Phi nodes would lead to // infinite recursion. if (isa(*I)) return false; for (const Use &Operand : I->operands()) if (!isInvariant(*Operand, Reg)) return false; // When the instruction is a load instruction, check that no write to memory // in the region aliases with the load. if (const LoadInst *LI = dyn_cast(I)) { AliasAnalysis::Location Loc = AA->getLocation(LI); const Region::const_block_iterator BE = Reg.block_end(); // Check if any basic block in the region can modify the location pointed to // by 'Loc'. If so, 'Val' is (likely) not invariant in the region. for (const BasicBlock *BB : Reg.blocks()) if (AA->canBasicBlockModify(*BB, Loc)) return false; } return true; } bool ScopDetection::hasAffineMemoryAccesses(DetectionContext &Context) const { for (auto P : Context.NonAffineAccesses) { const SCEVUnknown *BasePointer = P.first; Value *BaseValue = BasePointer->getValue(); // First step: collect parametric terms in all array references. SmallVector Terms; for (const SCEVAddRecExpr *AF : Context.NonAffineAccesses[BasePointer]) AF->collectParametricTerms(*SE, Terms); // Also collect terms from the affine memory accesses. for (const SCEVAddRecExpr *AF : Context.AffineAccesses[BasePointer]) AF->collectParametricTerms(*SE, Terms); // Second step: find array shape. SmallVector Sizes; SE->findArrayDimensions(Terms, Sizes); // Third step: compute the access functions for each subscript. for (const SCEVAddRecExpr *AF : Context.NonAffineAccesses[BasePointer]) { if (Sizes.empty()) return invalid(Context, /*Assert=*/true, AF); SmallVector Subscripts; if (!AF->computeAccessFunctions(*SE, Subscripts, Sizes) || Sizes.empty() || Subscripts.empty()) return invalid(Context, /*Assert=*/true, AF); // Check that the delinearized subscripts are affine. for (const SCEV *S : Subscripts) if (!isAffineExpr(&Context.CurRegion, S, *SE, BaseValue)) return invalid(Context, /*Assert=*/true, AF); } } return true; } bool ScopDetection::isValidMemoryAccess(Instruction &Inst, DetectionContext &Context) const { Value *Ptr = getPointerOperand(Inst); Loop *L = LI->getLoopFor(Inst.getParent()); const SCEV *AccessFunction = SE->getSCEVAtScope(Ptr, L); const SCEVUnknown *BasePointer; Value *BaseValue; BasePointer = dyn_cast(SE->getPointerBase(AccessFunction)); if (!BasePointer) return invalid(Context, /*Assert=*/true); BaseValue = BasePointer->getValue(); if (isa(BaseValue)) return invalid(Context, /*Assert=*/true); // Check that the base address of the access is invariant in the current // region. if (!isInvariant(*BaseValue, Context.CurRegion)) // Verification of this property is difficult as the independent blocks // pass may introduce aliasing that we did not have when running the // scop detection. return invalid(Context, /*Assert=*/false, BaseValue); AccessFunction = SE->getMinusSCEV(AccessFunction, BasePointer); if (AllowNonAffine) { // Do not check whether AccessFunction is affine. } else if (!isAffineExpr(&Context.CurRegion, AccessFunction, *SE, BaseValue)) { const SCEVAddRecExpr *AF = dyn_cast(AccessFunction); if (!PollyDelinearize || !AF) return invalid(Context, /*Assert=*/true, AccessFunction); // Collect all non affine memory accesses, and check whether they are linear // at the end of scop detection. That way we can delinearize all the memory // accesses to the same array in a unique step. if (Context.NonAffineAccesses[BasePointer].size() == 0) Context.NonAffineAccesses[BasePointer] = AFs(); Context.NonAffineAccesses[BasePointer].push_back(AF); } else if (const SCEVAddRecExpr *AF = dyn_cast(AccessFunction)) { if (Context.AffineAccesses[BasePointer].size() == 0) Context.AffineAccesses[BasePointer] = AFs(); Context.AffineAccesses[BasePointer].push_back(AF); } // FIXME: Alias Analysis thinks IntToPtrInst aliases with alloca instructions // created by IndependentBlocks Pass. if (isa(BaseValue)) return invalid(Context, /*Assert=*/true, BaseValue); if (IgnoreAliasing) return true; // Check if the base pointer of the memory access does alias with // any other pointer. This cannot be handled at the moment. AliasSet &AS = Context.AST.getAliasSetForPointer(BaseValue, AliasAnalysis::UnknownSize, Inst.getMetadata(LLVMContext::MD_tbaa)); // INVALID triggers an assertion in verifying mode, if it detects that a // SCoP was detected by SCoP detection and that this SCoP was invalidated by // a pass that stated it would preserve the SCoPs. We disable this check as // the independent blocks pass may create memory references which seem to // alias, if -basicaa is not available. They actually do not, but as we can // not proof this without -basicaa we would fail. We disable this check to // not cause irrelevant verification failures. if (!AS.isMustAlias()) return invalid(Context, /*Assert=*/true, &AS); return true; } bool ScopDetection::isValidInstruction(Instruction &Inst, DetectionContext &Context) const { if (PHINode *PN = dyn_cast(&Inst)) if (!canSynthesize(PN, LI, SE, &Context.CurRegion)) { if (SCEVCodegen) return invalid(Context, /*Assert=*/true, &Inst); else return invalid(Context, /*Assert=*/true, &Inst); } // We only check the call instruction but not invoke instruction. if (CallInst *CI = dyn_cast(&Inst)) { if (isValidCallInst(*CI)) return true; return invalid(Context, /*Assert=*/true, &Inst); } if (!Inst.mayWriteToMemory() && !Inst.mayReadFromMemory()) { if (!isa(Inst)) return true; return invalid(Context, /*Assert=*/true, &Inst); } // Check the access function. if (isa(Inst) || isa(Inst)) return isValidMemoryAccess(Inst, Context); // We do not know this instruction, therefore we assume it is invalid. return invalid(Context, /*Assert=*/true, &Inst); } bool ScopDetection::isValidLoop(Loop *L, DetectionContext &Context) const { if (!SCEVCodegen) { // If code generation is not in scev based mode, we need to ensure that // each loop has a canonical induction variable. PHINode *IndVar = L->getCanonicalInductionVariable(); if (!IndVar) return invalid(Context, /*Assert=*/true, L); } // Is the loop count affine? const SCEV *LoopCount = SE->getBackedgeTakenCount(L); if (!isAffineExpr(&Context.CurRegion, LoopCount, *SE)) return invalid(Context, /*Assert=*/true, L, LoopCount); return true; } Region *ScopDetection::expandRegion(Region &R) { // Initial no valid region was found (greater than R) Region *LastValidRegion = nullptr; Region *ExpandedRegion = R.getExpandedRegion(); DEBUG(dbgs() << "\tExpanding " << R.getNameStr() << "\n"); while (ExpandedRegion) { DetectionContext Context(*ExpandedRegion, *AA, false /* verifying */); DEBUG(dbgs() << "\t\tTrying " << ExpandedRegion->getNameStr() << "\n"); // Check the exit first (cheap) if (isValidExit(Context)) { // If the exit is valid check all blocks // - if true, a valid region was found => store it + keep expanding // - if false, .tbd. => stop (should this really end the loop?) if (!allBlocksValid(Context)) break; // Delete unnecessary regions (allocated by getExpandedRegion) if (LastValidRegion) delete LastValidRegion; // Store this region, because it is the greatest valid (encountered so // far). LastValidRegion = ExpandedRegion; // Create and test the next greater region (if any) ExpandedRegion = ExpandedRegion->getExpandedRegion(); } else { // Create and test the next greater region (if any) Region *TmpRegion = ExpandedRegion->getExpandedRegion(); // Delete unnecessary regions (allocated by getExpandedRegion) delete ExpandedRegion; ExpandedRegion = TmpRegion; } } DEBUG({ if (LastValidRegion) dbgs() << "\tto " << LastValidRegion->getNameStr() << "\n"; else dbgs() << "\tExpanding " << R.getNameStr() << " failed\n"; }); return LastValidRegion; } static bool regionWithoutLoops(Region &R, LoopInfo *LI) { for (const BasicBlock *BB : R.blocks()) if (R.contains(LI->getLoopFor(BB))) return false; return true; } // Remove all direct and indirect children of region R from the region set Regs, // but do not recurse further if the first child has been found. // // Return the number of regions erased from Regs. static unsigned eraseAllChildren(std::set &Regs, const Region &R) { unsigned Count = 0; for (auto &SubRegion : R) { if (Regs.find(SubRegion.get()) != Regs.end()) { ++Count; Regs.erase(SubRegion.get()); } else { Count += eraseAllChildren(Regs, *SubRegion); } } return Count; } void ScopDetection::findScops(Region &R) { if (!DetectRegionsWithoutLoops && regionWithoutLoops(R, LI)) return; LastFailure = ""; if (isValidRegion(R)) { ++ValidRegion; ValidRegions.insert(&R); return; } InvalidRegions[&R] = LastFailure; for (auto &SubRegion : R) findScops(*SubRegion); // Try to expand regions. // // As the region tree normally only contains canonical regions, non canonical // regions that form a Scop are not found. Therefore, those non canonical // regions are checked by expanding the canonical ones. std::vector ToExpand; for (auto &SubRegion : R) ToExpand.push_back(SubRegion.get()); for (Region *CurrentRegion : ToExpand) { // Skip invalid regions. Regions may become invalid, if they are element of // an already expanded region. if (ValidRegions.find(CurrentRegion) == ValidRegions.end()) continue; Region *ExpandedR = expandRegion(*CurrentRegion); if (!ExpandedR) continue; R.addSubRegion(ExpandedR, true); ValidRegions.insert(ExpandedR); ValidRegions.erase(CurrentRegion); // Erase all (direct and indirect) children of ExpandedR from the valid // regions and update the number of valid regions. ValidRegion -= eraseAllChildren(ValidRegions, *ExpandedR); } } bool ScopDetection::allBlocksValid(DetectionContext &Context) const { Region &R = Context.CurRegion; for (const BasicBlock *BB : R.blocks()) { Loop *L = LI->getLoopFor(BB); if (L && L->getHeader() == BB && !isValidLoop(L, Context)) return false; } for (BasicBlock *BB : R.blocks()) if (!isValidCFG(*BB, Context)) return false; for (BasicBlock *BB : R.blocks()) for (BasicBlock::iterator I = BB->begin(), E = --BB->end(); I != E; ++I) if (!isValidInstruction(*I, Context)) return false; if (!hasAffineMemoryAccesses(Context)) return false; return true; } bool ScopDetection::isValidExit(DetectionContext &Context) const { Region &R = Context.CurRegion; // PHI nodes are not allowed in the exit basic block. if (BasicBlock *Exit = R.getExit()) { BasicBlock::iterator I = Exit->begin(); if (I != Exit->end() && isa(*I)) return invalid(Context, /*Assert=*/true); } return true; } bool ScopDetection::isValidRegion(Region &R) const { DetectionContext Context(R, *AA, false /*verifying*/); return isValidRegion(Context); } bool ScopDetection::isValidRegion(DetectionContext &Context) const { Region &R = Context.CurRegion; DEBUG(dbgs() << "Checking region: " << R.getNameStr() << "\n\t"); if (R.isTopLevelRegion()) { DEBUG(dbgs() << "Top level region is invalid"; dbgs() << "\n"); return false; } if (!R.getEntry()->getName().count(OnlyRegion)) { DEBUG({ dbgs() << "Region entry does not match -polly-region-only"; dbgs() << "\n"; }); return false; } if (!R.getEnteringBlock()) { BasicBlock *entry = R.getEntry(); Loop *L = LI->getLoopFor(entry); if (L) { if (!L->isLoopSimplifyForm()) return invalid(Context, /*Assert=*/true); for (pred_iterator PI = pred_begin(entry), PE = pred_end(entry); PI != PE; ++PI) { // Region entering edges come from the same loop but outside the region // are not allowed. if (L->contains(*PI) && !R.contains(*PI)) return invalid(Context, /*Assert=*/true); } } } // SCoP cannot contain the entry block of the function, because we need // to insert alloca instruction there when translate scalar to array. if (R.getEntry() == &(R.getEntry()->getParent()->getEntryBlock())) return invalid(Context, /*Assert=*/true); if (!isValidExit(Context)) return false; if (!allBlocksValid(Context)) return false; DEBUG(dbgs() << "OK\n"); return true; } bool ScopDetection::isValidFunction(llvm::Function &F) { return !InvalidFunctions.count(&F); } void ScopDetection::getDebugLocation(const Region *R, unsigned &LineBegin, unsigned &LineEnd, std::string &FileName) { LineBegin = -1; LineEnd = 0; for (const BasicBlock *BB : R->blocks()) for (const Instruction &Inst : *BB) { DebugLoc DL = Inst.getDebugLoc(); if (DL.isUnknown()) continue; DIScope Scope(DL.getScope(Inst.getContext())); if (FileName.empty()) FileName = Scope.getFilename(); unsigned NewLine = DL.getLine(); LineBegin = std::min(LineBegin, NewLine); LineEnd = std::max(LineEnd, NewLine); } } void ScopDetection::printLocations(llvm::Function &F) { for (const Region *R : *this) { unsigned LineEntry, LineExit; std::string FileName; getDebugLocation(R, LineEntry, LineExit, FileName); DiagnosticScopFound Diagnostic(F, FileName, LineEntry, LineExit); F.getContext().diagnose(Diagnostic); } } bool ScopDetection::runOnFunction(llvm::Function &F) { LI = &getAnalysis(); RI = &getAnalysis(); if (!DetectScopsWithoutLoops && LI->empty()) return false; AA = &getAnalysis(); SE = &getAnalysis(); Region *TopRegion = RI->getTopLevelRegion(); releaseMemory(); if (OnlyFunction != "" && !F.getName().count(OnlyFunction)) return false; if (!isValidFunction(F)) return false; findScops(*TopRegion); if (ReportLevel >= 1) printLocations(F); return false; } void polly::ScopDetection::verifyRegion(const Region &R) const { assert(isMaxRegionInScop(R) && "Expect R is a valid region."); DetectionContext Context(const_cast(R), *AA, true /*verifying*/); isValidRegion(Context); } void polly::ScopDetection::verifyAnalysis() const { if (!VerifyScops) return; for (const Region *R : ValidRegions) verifyRegion(*R); } void ScopDetection::getAnalysisUsage(AnalysisUsage &AU) const { AU.addRequired(); AU.addRequired(); AU.addRequired(); AU.addRequired(); // We also need AA and RegionInfo when we are verifying analysis. AU.addRequiredTransitive(); AU.addRequiredTransitive(); AU.setPreservesAll(); } void ScopDetection::print(raw_ostream &OS, const Module *) const { for (const Region *R : ValidRegions) OS << "Valid Region for Scop: " << R->getNameStr() << '\n'; OS << "\n"; } void ScopDetection::releaseMemory() { ValidRegions.clear(); InvalidRegions.clear(); // Do not clear the invalid function set. } char ScopDetection::ID = 0; Pass *polly::createScopDetectionPass() { return new ScopDetection(); } INITIALIZE_PASS_BEGIN(ScopDetection, "polly-detect", "Polly - Detect static control parts (SCoPs)", false, false); INITIALIZE_AG_DEPENDENCY(AliasAnalysis); INITIALIZE_PASS_DEPENDENCY(DominatorTreeWrapperPass); INITIALIZE_PASS_DEPENDENCY(LoopInfo); INITIALIZE_PASS_DEPENDENCY(PostDominatorTree); INITIALIZE_PASS_DEPENDENCY(RegionInfo); INITIALIZE_PASS_DEPENDENCY(ScalarEvolution); INITIALIZE_PASS_END(ScopDetection, "polly-detect", "Polly - Detect static control parts (SCoPs)", false, false)