123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565 |
- #pragma once
- #ifdef __GNUC__
- #pragma GCC diagnostic push
- #pragma GCC diagnostic ignored "-Wunused-parameter"
- #endif
- //===- LoopPassManager.h - Loop pass management -----------------*- C++ -*-===//
- //
- // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
- // See https://llvm.org/LICENSE.txt for license information.
- // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
- //
- //===----------------------------------------------------------------------===//
- /// \file
- ///
- /// This header provides classes for managing a pipeline of passes over loops
- /// in LLVM IR.
- ///
- /// The primary loop pass pipeline is managed in a very particular way to
- /// provide a set of core guarantees:
- /// 1) Loops are, where possible, in simplified form.
- /// 2) Loops are *always* in LCSSA form.
- /// 3) A collection of Loop-specific analysis results are available:
- /// - LoopInfo
- /// - DominatorTree
- /// - ScalarEvolution
- /// - AAManager
- /// 4) All loop passes preserve #1 (where possible), #2, and #3.
- /// 5) Loop passes run over each loop in the loop nest from the innermost to
- /// the outermost. Specifically, all inner loops are processed before
- /// passes run over outer loops. When running the pipeline across an inner
- /// loop creates new inner loops, those are added and processed in this
- /// order as well.
- ///
- /// This process is designed to facilitate transformations which simplify,
- /// reduce, and remove loops. For passes which are more oriented towards
- /// optimizing loops, especially optimizing loop *nests* instead of single
- /// loops in isolation, this framework is less interesting.
- ///
- //===----------------------------------------------------------------------===//
- #ifndef LLVM_TRANSFORMS_SCALAR_LOOPPASSMANAGER_H
- #define LLVM_TRANSFORMS_SCALAR_LOOPPASSMANAGER_H
- #include "llvm/ADT/PriorityWorklist.h"
- #include "llvm/Analysis/LoopAnalysisManager.h"
- #include "llvm/Analysis/LoopInfo.h"
- #include "llvm/Analysis/LoopNestAnalysis.h"
- #include "llvm/IR/PassManager.h"
- #include "llvm/Transforms/Utils/LCSSA.h"
- #include "llvm/Transforms/Utils/LoopSimplify.h"
- #include "llvm/Transforms/Utils/LoopUtils.h"
- #include <memory>
- namespace llvm {
- // Forward declarations of an update tracking API used in the pass manager.
- class LPMUpdater;
- class PassInstrumentation;
- namespace {
- template <typename PassT>
- using HasRunOnLoopT = decltype(std::declval<PassT>().run(
- std::declval<Loop &>(), std::declval<LoopAnalysisManager &>(),
- std::declval<LoopStandardAnalysisResults &>(),
- std::declval<LPMUpdater &>()));
- } // namespace
- // Explicit specialization and instantiation declarations for the pass manager.
- // See the comments on the definition of the specialization for details on how
- // it differs from the primary template.
- template <>
- class PassManager<Loop, LoopAnalysisManager, LoopStandardAnalysisResults &,
- LPMUpdater &>
- : public PassInfoMixin<
- PassManager<Loop, LoopAnalysisManager, LoopStandardAnalysisResults &,
- LPMUpdater &>> {
- public:
- explicit PassManager() = default;
- // FIXME: These are equivalent to the default move constructor/move
- // assignment. However, using = default triggers linker errors due to the
- // explicit instantiations below. Find a way to use the default and remove the
- // duplicated code here.
- PassManager(PassManager &&Arg)
- : IsLoopNestPass(std::move(Arg.IsLoopNestPass)),
- LoopPasses(std::move(Arg.LoopPasses)),
- LoopNestPasses(std::move(Arg.LoopNestPasses)) {}
- PassManager &operator=(PassManager &&RHS) {
- IsLoopNestPass = std::move(RHS.IsLoopNestPass);
- LoopPasses = std::move(RHS.LoopPasses);
- LoopNestPasses = std::move(RHS.LoopNestPasses);
- return *this;
- }
- PreservedAnalyses run(Loop &L, LoopAnalysisManager &AM,
- LoopStandardAnalysisResults &AR, LPMUpdater &U);
- void printPipeline(raw_ostream &OS,
- function_ref<StringRef(StringRef)> MapClassName2PassName);
- /// Add either a loop pass or a loop-nest pass to the pass manager. Append \p
- /// Pass to the list of loop passes if it has a dedicated \fn run() method for
- /// loops and to the list of loop-nest passes if the \fn run() method is for
- /// loop-nests instead. Also append whether \p Pass is loop-nest pass or not
- /// to the end of \var IsLoopNestPass so we can easily identify the types of
- /// passes in the pass manager later.
- template <typename PassT>
- LLVM_ATTRIBUTE_MINSIZE
- std::enable_if_t<is_detected<HasRunOnLoopT, PassT>::value>
- addPass(PassT &&Pass) {
- using LoopPassModelT =
- detail::PassModel<Loop, PassT, PreservedAnalyses, LoopAnalysisManager,
- LoopStandardAnalysisResults &, LPMUpdater &>;
- IsLoopNestPass.push_back(false);
- // Do not use make_unique or emplace_back, they cause too many template
- // instantiations, causing terrible compile times.
- LoopPasses.push_back(std::unique_ptr<LoopPassConceptT>(
- new LoopPassModelT(std::forward<PassT>(Pass))));
- }
- template <typename PassT>
- LLVM_ATTRIBUTE_MINSIZE
- std::enable_if_t<!is_detected<HasRunOnLoopT, PassT>::value>
- addPass(PassT &&Pass) {
- using LoopNestPassModelT =
- detail::PassModel<LoopNest, PassT, PreservedAnalyses,
- LoopAnalysisManager, LoopStandardAnalysisResults &,
- LPMUpdater &>;
- IsLoopNestPass.push_back(true);
- // Do not use make_unique or emplace_back, they cause too many template
- // instantiations, causing terrible compile times.
- LoopNestPasses.push_back(std::unique_ptr<LoopNestPassConceptT>(
- new LoopNestPassModelT(std::forward<PassT>(Pass))));
- }
- // Specializations of `addPass` for `RepeatedPass`. These are necessary since
- // `RepeatedPass` has a templated `run` method that will result in incorrect
- // detection of `HasRunOnLoopT`.
- template <typename PassT>
- LLVM_ATTRIBUTE_MINSIZE
- std::enable_if_t<is_detected<HasRunOnLoopT, PassT>::value>
- addPass(RepeatedPass<PassT> &&Pass) {
- using RepeatedLoopPassModelT =
- detail::PassModel<Loop, RepeatedPass<PassT>, PreservedAnalyses,
- LoopAnalysisManager, LoopStandardAnalysisResults &,
- LPMUpdater &>;
- IsLoopNestPass.push_back(false);
- // Do not use make_unique or emplace_back, they cause too many template
- // instantiations, causing terrible compile times.
- LoopPasses.push_back(std::unique_ptr<LoopPassConceptT>(
- new RepeatedLoopPassModelT(std::move(Pass))));
- }
- template <typename PassT>
- LLVM_ATTRIBUTE_MINSIZE
- std::enable_if_t<!is_detected<HasRunOnLoopT, PassT>::value>
- addPass(RepeatedPass<PassT> &&Pass) {
- using RepeatedLoopNestPassModelT =
- detail::PassModel<LoopNest, RepeatedPass<PassT>, PreservedAnalyses,
- LoopAnalysisManager, LoopStandardAnalysisResults &,
- LPMUpdater &>;
- IsLoopNestPass.push_back(true);
- // Do not use make_unique or emplace_back, they cause too many template
- // instantiations, causing terrible compile times.
- LoopNestPasses.push_back(std::unique_ptr<LoopNestPassConceptT>(
- new RepeatedLoopNestPassModelT(std::move(Pass))));
- }
- bool isEmpty() const { return LoopPasses.empty() && LoopNestPasses.empty(); }
- static bool isRequired() { return true; }
- size_t getNumLoopPasses() const { return LoopPasses.size(); }
- size_t getNumLoopNestPasses() const { return LoopNestPasses.size(); }
- protected:
- using LoopPassConceptT =
- detail::PassConcept<Loop, LoopAnalysisManager,
- LoopStandardAnalysisResults &, LPMUpdater &>;
- using LoopNestPassConceptT =
- detail::PassConcept<LoopNest, LoopAnalysisManager,
- LoopStandardAnalysisResults &, LPMUpdater &>;
- // BitVector that identifies whether the passes are loop passes or loop-nest
- // passes (true for loop-nest passes).
- BitVector IsLoopNestPass;
- std::vector<std::unique_ptr<LoopPassConceptT>> LoopPasses;
- std::vector<std::unique_ptr<LoopNestPassConceptT>> LoopNestPasses;
- /// Run either a loop pass or a loop-nest pass. Returns `std::nullopt` if
- /// PassInstrumentation's BeforePass returns false. Otherwise, returns the
- /// preserved analyses of the pass.
- template <typename IRUnitT, typename PassT>
- std::optional<PreservedAnalyses>
- runSinglePass(IRUnitT &IR, PassT &Pass, LoopAnalysisManager &AM,
- LoopStandardAnalysisResults &AR, LPMUpdater &U,
- PassInstrumentation &PI);
- PreservedAnalyses runWithLoopNestPasses(Loop &L, LoopAnalysisManager &AM,
- LoopStandardAnalysisResults &AR,
- LPMUpdater &U);
- PreservedAnalyses runWithoutLoopNestPasses(Loop &L, LoopAnalysisManager &AM,
- LoopStandardAnalysisResults &AR,
- LPMUpdater &U);
- private:
- static const Loop &getLoopFromIR(Loop &L) { return L; }
- static const Loop &getLoopFromIR(LoopNest &LN) {
- return LN.getOutermostLoop();
- }
- };
- /// The Loop pass manager.
- ///
- /// See the documentation for the PassManager template for details. It runs
- /// a sequence of Loop passes over each Loop that the manager is run over. This
- /// typedef serves as a convenient way to refer to this construct.
- typedef PassManager<Loop, LoopAnalysisManager, LoopStandardAnalysisResults &,
- LPMUpdater &>
- LoopPassManager;
- /// A partial specialization of the require analysis template pass to forward
- /// the extra parameters from a transformation's run method to the
- /// AnalysisManager's getResult.
- template <typename AnalysisT>
- struct RequireAnalysisPass<AnalysisT, Loop, LoopAnalysisManager,
- LoopStandardAnalysisResults &, LPMUpdater &>
- : PassInfoMixin<
- RequireAnalysisPass<AnalysisT, Loop, LoopAnalysisManager,
- LoopStandardAnalysisResults &, LPMUpdater &>> {
- PreservedAnalyses run(Loop &L, LoopAnalysisManager &AM,
- LoopStandardAnalysisResults &AR, LPMUpdater &) {
- (void)AM.template getResult<AnalysisT>(L, AR);
- return PreservedAnalyses::all();
- }
- void printPipeline(raw_ostream &OS,
- function_ref<StringRef(StringRef)> MapClassName2PassName) {
- auto ClassName = AnalysisT::name();
- auto PassName = MapClassName2PassName(ClassName);
- OS << "require<" << PassName << ">";
- }
- };
- /// An alias template to easily name a require analysis loop pass.
- template <typename AnalysisT>
- using RequireAnalysisLoopPass =
- RequireAnalysisPass<AnalysisT, Loop, LoopAnalysisManager,
- LoopStandardAnalysisResults &, LPMUpdater &>;
- class FunctionToLoopPassAdaptor;
- /// This class provides an interface for updating the loop pass manager based
- /// on mutations to the loop nest.
- ///
- /// A reference to an instance of this class is passed as an argument to each
- /// Loop pass, and Loop passes should use it to update LPM infrastructure if
- /// they modify the loop nest structure.
- ///
- /// \c LPMUpdater comes with two modes: the loop mode and the loop-nest mode. In
- /// loop mode, all the loops in the function will be pushed into the worklist
- /// and when new loops are added to the pipeline, their subloops are also
- /// inserted recursively. On the other hand, in loop-nest mode, only top-level
- /// loops are contained in the worklist and the addition of new (top-level)
- /// loops will not trigger the addition of their subloops.
- class LPMUpdater {
- public:
- /// This can be queried by loop passes which run other loop passes (like pass
- /// managers) to know whether the loop needs to be skipped due to updates to
- /// the loop nest.
- ///
- /// If this returns true, the loop object may have been deleted, so passes
- /// should take care not to touch the object.
- bool skipCurrentLoop() const { return SkipCurrentLoop; }
- /// Loop passes should use this method to indicate they have deleted a loop
- /// from the nest.
- ///
- /// Note that this loop must either be the current loop or a subloop of the
- /// current loop. This routine must be called prior to removing the loop from
- /// the loop nest.
- ///
- /// If this is called for the current loop, in addition to clearing any
- /// state, this routine will mark that the current loop should be skipped by
- /// the rest of the pass management infrastructure.
- void markLoopAsDeleted(Loop &L, llvm::StringRef Name) {
- LAM.clear(L, Name);
- assert((&L == CurrentL || CurrentL->contains(&L)) &&
- "Cannot delete a loop outside of the "
- "subloop tree currently being processed.");
- if (&L == CurrentL)
- SkipCurrentLoop = true;
- }
- void setParentLoop(Loop *L) {
- #ifdef LLVM_ENABLE_ABI_BREAKING_CHECKS
- ParentL = L;
- #endif
- }
- /// Loop passes should use this method to indicate they have added new child
- /// loops of the current loop.
- ///
- /// \p NewChildLoops must contain only the immediate children. Any nested
- /// loops within them will be visited in postorder as usual for the loop pass
- /// manager.
- void addChildLoops(ArrayRef<Loop *> NewChildLoops) {
- assert(!LoopNestMode &&
- "Child loops should not be pushed in loop-nest mode.");
- // Insert ourselves back into the worklist first, as this loop should be
- // revisited after all the children have been processed.
- Worklist.insert(CurrentL);
- #ifndef NDEBUG
- for (Loop *NewL : NewChildLoops)
- assert(NewL->getParentLoop() == CurrentL && "All of the new loops must "
- "be immediate children of "
- "the current loop!");
- #endif
- appendLoopsToWorklist(NewChildLoops, Worklist);
- // Also skip further processing of the current loop--it will be revisited
- // after all of its newly added children are accounted for.
- SkipCurrentLoop = true;
- }
- /// Loop passes should use this method to indicate they have added new
- /// sibling loops to the current loop.
- ///
- /// \p NewSibLoops must only contain the immediate sibling loops. Any nested
- /// loops within them will be visited in postorder as usual for the loop pass
- /// manager.
- void addSiblingLoops(ArrayRef<Loop *> NewSibLoops) {
- #if defined(LLVM_ENABLE_ABI_BREAKING_CHECKS) && !defined(NDEBUG)
- for (Loop *NewL : NewSibLoops)
- assert(NewL->getParentLoop() == ParentL &&
- "All of the new loops must be siblings of the current loop!");
- #endif
- if (LoopNestMode)
- Worklist.insert(NewSibLoops);
- else
- appendLoopsToWorklist(NewSibLoops, Worklist);
- // No need to skip the current loop or revisit it, as sibling loops
- // shouldn't impact anything.
- }
- /// Restart the current loop.
- ///
- /// Loop passes should call this method to indicate the current loop has been
- /// sufficiently changed that it should be re-visited from the begining of
- /// the loop pass pipeline rather than continuing.
- void revisitCurrentLoop() {
- // Tell the currently in-flight pipeline to stop running.
- SkipCurrentLoop = true;
- // And insert ourselves back into the worklist.
- Worklist.insert(CurrentL);
- }
- bool isLoopNestChanged() const {
- return LoopNestChanged;
- }
- /// Loopnest passes should use this method to indicate if the
- /// loopnest has been modified.
- void markLoopNestChanged(bool Changed) {
- LoopNestChanged = Changed;
- }
- private:
- friend class llvm::FunctionToLoopPassAdaptor;
- /// The \c FunctionToLoopPassAdaptor's worklist of loops to process.
- SmallPriorityWorklist<Loop *, 4> &Worklist;
- /// The analysis manager for use in the current loop nest.
- LoopAnalysisManager &LAM;
- Loop *CurrentL;
- bool SkipCurrentLoop;
- const bool LoopNestMode;
- bool LoopNestChanged;
- #ifdef LLVM_ENABLE_ABI_BREAKING_CHECKS
- // In debug builds we also track the parent loop to implement asserts even in
- // the face of loop deletion.
- Loop *ParentL;
- #endif
- LPMUpdater(SmallPriorityWorklist<Loop *, 4> &Worklist,
- LoopAnalysisManager &LAM, bool LoopNestMode = false,
- bool LoopNestChanged = false)
- : Worklist(Worklist), LAM(LAM), LoopNestMode(LoopNestMode),
- LoopNestChanged(LoopNestChanged) {}
- };
- template <typename IRUnitT, typename PassT>
- std::optional<PreservedAnalyses> LoopPassManager::runSinglePass(
- IRUnitT &IR, PassT &Pass, LoopAnalysisManager &AM,
- LoopStandardAnalysisResults &AR, LPMUpdater &U, PassInstrumentation &PI) {
- // Get the loop in case of Loop pass and outermost loop in case of LoopNest
- // pass which is to be passed to BeforePass and AfterPass call backs.
- const Loop &L = getLoopFromIR(IR);
- // Check the PassInstrumentation's BeforePass callbacks before running the
- // pass, skip its execution completely if asked to (callback returns false).
- if (!PI.runBeforePass<Loop>(*Pass, L))
- return std::nullopt;
- PreservedAnalyses PA = Pass->run(IR, AM, AR, U);
- // do not pass deleted Loop into the instrumentation
- if (U.skipCurrentLoop())
- PI.runAfterPassInvalidated<IRUnitT>(*Pass, PA);
- else
- PI.runAfterPass<Loop>(*Pass, L, PA);
- return PA;
- }
- /// Adaptor that maps from a function to its loops.
- ///
- /// Designed to allow composition of a LoopPass(Manager) and a
- /// FunctionPassManager. Note that if this pass is constructed with a \c
- /// FunctionAnalysisManager it will run the \c LoopAnalysisManagerFunctionProxy
- /// analysis prior to running the loop passes over the function to enable a \c
- /// LoopAnalysisManager to be used within this run safely.
- ///
- /// The adaptor comes with two modes: the loop mode and the loop-nest mode, and
- /// the worklist updater lived inside will be in the same mode as the adaptor
- /// (refer to the documentation of \c LPMUpdater for more detailed explanation).
- /// Specifically, in loop mode, all loops in the funciton will be pushed into
- /// the worklist and processed by \p Pass, while only top-level loops are
- /// processed in loop-nest mode. Please refer to the various specializations of
- /// \fn createLoopFunctionToLoopPassAdaptor to see when loop mode and loop-nest
- /// mode are used.
- class FunctionToLoopPassAdaptor
- : public PassInfoMixin<FunctionToLoopPassAdaptor> {
- public:
- using PassConceptT =
- detail::PassConcept<Loop, LoopAnalysisManager,
- LoopStandardAnalysisResults &, LPMUpdater &>;
- explicit FunctionToLoopPassAdaptor(std::unique_ptr<PassConceptT> Pass,
- bool UseMemorySSA = false,
- bool UseBlockFrequencyInfo = false,
- bool UseBranchProbabilityInfo = false,
- bool LoopNestMode = false)
- : Pass(std::move(Pass)), UseMemorySSA(UseMemorySSA),
- UseBlockFrequencyInfo(UseBlockFrequencyInfo),
- UseBranchProbabilityInfo(UseBranchProbabilityInfo),
- LoopNestMode(LoopNestMode) {
- LoopCanonicalizationFPM.addPass(LoopSimplifyPass());
- LoopCanonicalizationFPM.addPass(LCSSAPass());
- }
- /// Runs the loop passes across every loop in the function.
- PreservedAnalyses run(Function &F, FunctionAnalysisManager &AM);
- void printPipeline(raw_ostream &OS,
- function_ref<StringRef(StringRef)> MapClassName2PassName);
- static bool isRequired() { return true; }
- bool isLoopNestMode() const { return LoopNestMode; }
- private:
- std::unique_ptr<PassConceptT> Pass;
- FunctionPassManager LoopCanonicalizationFPM;
- bool UseMemorySSA = false;
- bool UseBlockFrequencyInfo = false;
- bool UseBranchProbabilityInfo = false;
- const bool LoopNestMode;
- };
- /// A function to deduce a loop pass type and wrap it in the templated
- /// adaptor.
- ///
- /// If \p Pass is a loop pass, the returned adaptor will be in loop mode.
- template <typename LoopPassT>
- inline std::enable_if_t<is_detected<HasRunOnLoopT, LoopPassT>::value,
- FunctionToLoopPassAdaptor>
- createFunctionToLoopPassAdaptor(LoopPassT &&Pass, bool UseMemorySSA = false,
- bool UseBlockFrequencyInfo = false,
- bool UseBranchProbabilityInfo = false) {
- using PassModelT =
- detail::PassModel<Loop, LoopPassT, PreservedAnalyses, LoopAnalysisManager,
- LoopStandardAnalysisResults &, LPMUpdater &>;
- // Do not use make_unique, it causes too many template instantiations,
- // causing terrible compile times.
- return FunctionToLoopPassAdaptor(
- std::unique_ptr<FunctionToLoopPassAdaptor::PassConceptT>(
- new PassModelT(std::forward<LoopPassT>(Pass))),
- UseMemorySSA, UseBlockFrequencyInfo, UseBranchProbabilityInfo, false);
- }
- /// If \p Pass is a loop-nest pass, \p Pass will first be wrapped into a
- /// \c LoopPassManager and the returned adaptor will be in loop-nest mode.
- template <typename LoopNestPassT>
- inline std::enable_if_t<!is_detected<HasRunOnLoopT, LoopNestPassT>::value,
- FunctionToLoopPassAdaptor>
- createFunctionToLoopPassAdaptor(LoopNestPassT &&Pass, bool UseMemorySSA = false,
- bool UseBlockFrequencyInfo = false,
- bool UseBranchProbabilityInfo = false) {
- LoopPassManager LPM;
- LPM.addPass(std::forward<LoopNestPassT>(Pass));
- using PassModelT =
- detail::PassModel<Loop, LoopPassManager, PreservedAnalyses,
- LoopAnalysisManager, LoopStandardAnalysisResults &,
- LPMUpdater &>;
- // Do not use make_unique, it causes too many template instantiations,
- // causing terrible compile times.
- return FunctionToLoopPassAdaptor(
- std::unique_ptr<FunctionToLoopPassAdaptor::PassConceptT>(
- new PassModelT(std::move(LPM))),
- UseMemorySSA, UseBlockFrequencyInfo, UseBranchProbabilityInfo, true);
- }
- /// If \p Pass is an instance of \c LoopPassManager, the returned adaptor will
- /// be in loop-nest mode if the pass manager contains only loop-nest passes.
- template <>
- inline FunctionToLoopPassAdaptor
- createFunctionToLoopPassAdaptor<LoopPassManager>(
- LoopPassManager &&LPM, bool UseMemorySSA, bool UseBlockFrequencyInfo,
- bool UseBranchProbabilityInfo) {
- // Check if LPM contains any loop pass and if it does not, returns an adaptor
- // in loop-nest mode.
- using PassModelT =
- detail::PassModel<Loop, LoopPassManager, PreservedAnalyses,
- LoopAnalysisManager, LoopStandardAnalysisResults &,
- LPMUpdater &>;
- bool LoopNestMode = (LPM.getNumLoopPasses() == 0);
- // Do not use make_unique, it causes too many template instantiations,
- // causing terrible compile times.
- return FunctionToLoopPassAdaptor(
- std::unique_ptr<FunctionToLoopPassAdaptor::PassConceptT>(
- new PassModelT(std::move(LPM))),
- UseMemorySSA, UseBlockFrequencyInfo, UseBranchProbabilityInfo,
- LoopNestMode);
- }
- /// Pass for printing a loop's contents as textual IR.
- class PrintLoopPass : public PassInfoMixin<PrintLoopPass> {
- raw_ostream &OS;
- std::string Banner;
- public:
- PrintLoopPass();
- PrintLoopPass(raw_ostream &OS, const std::string &Banner = "");
- PreservedAnalyses run(Loop &L, LoopAnalysisManager &,
- LoopStandardAnalysisResults &, LPMUpdater &);
- };
- }
- #endif // LLVM_TRANSFORMS_SCALAR_LOOPPASSMANAGER_H
- #ifdef __GNUC__
- #pragma GCC diagnostic pop
- #endif
|