Details | Last modification | View Log | RSS feed
| Rev | Author | Line No. | Line |
|---|---|---|---|
| 14 | pmbaty | 1 | //===- LoopPassManager.h - Loop pass management -----------------*- C++ -*-===// |
| 2 | // |
||
| 3 | // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. |
||
| 4 | // See https://llvm.org/LICENSE.txt for license information. |
||
| 5 | // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception |
||
| 6 | // |
||
| 7 | //===----------------------------------------------------------------------===// |
||
| 8 | /// \file |
||
| 9 | /// |
||
| 10 | /// This header provides classes for managing a pipeline of passes over loops |
||
| 11 | /// in LLVM IR. |
||
| 12 | /// |
||
| 13 | /// The primary loop pass pipeline is managed in a very particular way to |
||
| 14 | /// provide a set of core guarantees: |
||
| 15 | /// 1) Loops are, where possible, in simplified form. |
||
| 16 | /// 2) Loops are *always* in LCSSA form. |
||
| 17 | /// 3) A collection of Loop-specific analysis results are available: |
||
| 18 | /// - LoopInfo |
||
| 19 | /// - DominatorTree |
||
| 20 | /// - ScalarEvolution |
||
| 21 | /// - AAManager |
||
| 22 | /// 4) All loop passes preserve #1 (where possible), #2, and #3. |
||
| 23 | /// 5) Loop passes run over each loop in the loop nest from the innermost to |
||
| 24 | /// the outermost. Specifically, all inner loops are processed before |
||
| 25 | /// passes run over outer loops. When running the pipeline across an inner |
||
| 26 | /// loop creates new inner loops, those are added and processed in this |
||
| 27 | /// order as well. |
||
| 28 | /// |
||
| 29 | /// This process is designed to facilitate transformations which simplify, |
||
| 30 | /// reduce, and remove loops. For passes which are more oriented towards |
||
| 31 | /// optimizing loops, especially optimizing loop *nests* instead of single |
||
| 32 | /// loops in isolation, this framework is less interesting. |
||
| 33 | /// |
||
| 34 | //===----------------------------------------------------------------------===// |
||
| 35 | |||
| 36 | #ifndef LLVM_TRANSFORMS_SCALAR_LOOPPASSMANAGER_H |
||
| 37 | #define LLVM_TRANSFORMS_SCALAR_LOOPPASSMANAGER_H |
||
| 38 | |||
| 39 | #include "llvm/ADT/PriorityWorklist.h" |
||
| 40 | #include "llvm/Analysis/LoopAnalysisManager.h" |
||
| 41 | #include "llvm/Analysis/LoopInfo.h" |
||
| 42 | #include "llvm/Analysis/LoopNestAnalysis.h" |
||
| 43 | #include "llvm/IR/PassManager.h" |
||
| 44 | #include "llvm/Transforms/Utils/LCSSA.h" |
||
| 45 | #include "llvm/Transforms/Utils/LoopSimplify.h" |
||
| 46 | #include "llvm/Transforms/Utils/LoopUtils.h" |
||
| 47 | #include <memory> |
||
| 48 | |||
| 49 | namespace llvm { |
||
| 50 | |||
| 51 | // Forward declarations of an update tracking API used in the pass manager. |
||
| 52 | class LPMUpdater; |
||
| 53 | class PassInstrumentation; |
||
| 54 | |||
| 55 | namespace { |
||
| 56 | |||
| 57 | template <typename PassT> |
||
| 58 | using HasRunOnLoopT = decltype(std::declval<PassT>().run( |
||
| 59 | std::declval<Loop &>(), std::declval<LoopAnalysisManager &>(), |
||
| 60 | std::declval<LoopStandardAnalysisResults &>(), |
||
| 61 | std::declval<LPMUpdater &>())); |
||
| 62 | |||
| 63 | } // namespace |
||
| 64 | |||
| 65 | // Explicit specialization and instantiation declarations for the pass manager. |
||
| 66 | // See the comments on the definition of the specialization for details on how |
||
| 67 | // it differs from the primary template. |
||
| 68 | template <> |
||
| 69 | class PassManager<Loop, LoopAnalysisManager, LoopStandardAnalysisResults &, |
||
| 70 | LPMUpdater &> |
||
| 71 | : public PassInfoMixin< |
||
| 72 | PassManager<Loop, LoopAnalysisManager, LoopStandardAnalysisResults &, |
||
| 73 | LPMUpdater &>> { |
||
| 74 | public: |
||
| 75 | explicit PassManager() = default; |
||
| 76 | |||
| 77 | // FIXME: These are equivalent to the default move constructor/move |
||
| 78 | // assignment. However, using = default triggers linker errors due to the |
||
| 79 | // explicit instantiations below. Find a way to use the default and remove the |
||
| 80 | // duplicated code here. |
||
| 81 | PassManager(PassManager &&Arg) |
||
| 82 | : IsLoopNestPass(std::move(Arg.IsLoopNestPass)), |
||
| 83 | LoopPasses(std::move(Arg.LoopPasses)), |
||
| 84 | LoopNestPasses(std::move(Arg.LoopNestPasses)) {} |
||
| 85 | |||
| 86 | PassManager &operator=(PassManager &&RHS) { |
||
| 87 | IsLoopNestPass = std::move(RHS.IsLoopNestPass); |
||
| 88 | LoopPasses = std::move(RHS.LoopPasses); |
||
| 89 | LoopNestPasses = std::move(RHS.LoopNestPasses); |
||
| 90 | return *this; |
||
| 91 | } |
||
| 92 | |||
| 93 | PreservedAnalyses run(Loop &L, LoopAnalysisManager &AM, |
||
| 94 | LoopStandardAnalysisResults &AR, LPMUpdater &U); |
||
| 95 | |||
| 96 | void printPipeline(raw_ostream &OS, |
||
| 97 | function_ref<StringRef(StringRef)> MapClassName2PassName); |
||
| 98 | /// Add either a loop pass or a loop-nest pass to the pass manager. Append \p |
||
| 99 | /// Pass to the list of loop passes if it has a dedicated \fn run() method for |
||
| 100 | /// loops and to the list of loop-nest passes if the \fn run() method is for |
||
| 101 | /// loop-nests instead. Also append whether \p Pass is loop-nest pass or not |
||
| 102 | /// to the end of \var IsLoopNestPass so we can easily identify the types of |
||
| 103 | /// passes in the pass manager later. |
||
| 104 | template <typename PassT> |
||
| 105 | LLVM_ATTRIBUTE_MINSIZE |
||
| 106 | std::enable_if_t<is_detected<HasRunOnLoopT, PassT>::value> |
||
| 107 | addPass(PassT &&Pass) { |
||
| 108 | using LoopPassModelT = |
||
| 109 | detail::PassModel<Loop, PassT, PreservedAnalyses, LoopAnalysisManager, |
||
| 110 | LoopStandardAnalysisResults &, LPMUpdater &>; |
||
| 111 | IsLoopNestPass.push_back(false); |
||
| 112 | // Do not use make_unique or emplace_back, they cause too many template |
||
| 113 | // instantiations, causing terrible compile times. |
||
| 114 | LoopPasses.push_back(std::unique_ptr<LoopPassConceptT>( |
||
| 115 | new LoopPassModelT(std::forward<PassT>(Pass)))); |
||
| 116 | } |
||
| 117 | |||
| 118 | template <typename PassT> |
||
| 119 | LLVM_ATTRIBUTE_MINSIZE |
||
| 120 | std::enable_if_t<!is_detected<HasRunOnLoopT, PassT>::value> |
||
| 121 | addPass(PassT &&Pass) { |
||
| 122 | using LoopNestPassModelT = |
||
| 123 | detail::PassModel<LoopNest, PassT, PreservedAnalyses, |
||
| 124 | LoopAnalysisManager, LoopStandardAnalysisResults &, |
||
| 125 | LPMUpdater &>; |
||
| 126 | IsLoopNestPass.push_back(true); |
||
| 127 | // Do not use make_unique or emplace_back, they cause too many template |
||
| 128 | // instantiations, causing terrible compile times. |
||
| 129 | LoopNestPasses.push_back(std::unique_ptr<LoopNestPassConceptT>( |
||
| 130 | new LoopNestPassModelT(std::forward<PassT>(Pass)))); |
||
| 131 | } |
||
| 132 | |||
| 133 | // Specializations of `addPass` for `RepeatedPass`. These are necessary since |
||
| 134 | // `RepeatedPass` has a templated `run` method that will result in incorrect |
||
| 135 | // detection of `HasRunOnLoopT`. |
||
| 136 | template <typename PassT> |
||
| 137 | LLVM_ATTRIBUTE_MINSIZE |
||
| 138 | std::enable_if_t<is_detected<HasRunOnLoopT, PassT>::value> |
||
| 139 | addPass(RepeatedPass<PassT> &&Pass) { |
||
| 140 | using RepeatedLoopPassModelT = |
||
| 141 | detail::PassModel<Loop, RepeatedPass<PassT>, PreservedAnalyses, |
||
| 142 | LoopAnalysisManager, LoopStandardAnalysisResults &, |
||
| 143 | LPMUpdater &>; |
||
| 144 | IsLoopNestPass.push_back(false); |
||
| 145 | // Do not use make_unique or emplace_back, they cause too many template |
||
| 146 | // instantiations, causing terrible compile times. |
||
| 147 | LoopPasses.push_back(std::unique_ptr<LoopPassConceptT>( |
||
| 148 | new RepeatedLoopPassModelT(std::move(Pass)))); |
||
| 149 | } |
||
| 150 | |||
| 151 | template <typename PassT> |
||
| 152 | LLVM_ATTRIBUTE_MINSIZE |
||
| 153 | std::enable_if_t<!is_detected<HasRunOnLoopT, PassT>::value> |
||
| 154 | addPass(RepeatedPass<PassT> &&Pass) { |
||
| 155 | using RepeatedLoopNestPassModelT = |
||
| 156 | detail::PassModel<LoopNest, RepeatedPass<PassT>, PreservedAnalyses, |
||
| 157 | LoopAnalysisManager, LoopStandardAnalysisResults &, |
||
| 158 | LPMUpdater &>; |
||
| 159 | IsLoopNestPass.push_back(true); |
||
| 160 | // Do not use make_unique or emplace_back, they cause too many template |
||
| 161 | // instantiations, causing terrible compile times. |
||
| 162 | LoopNestPasses.push_back(std::unique_ptr<LoopNestPassConceptT>( |
||
| 163 | new RepeatedLoopNestPassModelT(std::move(Pass)))); |
||
| 164 | } |
||
| 165 | |||
| 166 | bool isEmpty() const { return LoopPasses.empty() && LoopNestPasses.empty(); } |
||
| 167 | |||
| 168 | static bool isRequired() { return true; } |
||
| 169 | |||
| 170 | size_t getNumLoopPasses() const { return LoopPasses.size(); } |
||
| 171 | size_t getNumLoopNestPasses() const { return LoopNestPasses.size(); } |
||
| 172 | |||
| 173 | protected: |
||
| 174 | using LoopPassConceptT = |
||
| 175 | detail::PassConcept<Loop, LoopAnalysisManager, |
||
| 176 | LoopStandardAnalysisResults &, LPMUpdater &>; |
||
| 177 | using LoopNestPassConceptT = |
||
| 178 | detail::PassConcept<LoopNest, LoopAnalysisManager, |
||
| 179 | LoopStandardAnalysisResults &, LPMUpdater &>; |
||
| 180 | |||
| 181 | // BitVector that identifies whether the passes are loop passes or loop-nest |
||
| 182 | // passes (true for loop-nest passes). |
||
| 183 | BitVector IsLoopNestPass; |
||
| 184 | std::vector<std::unique_ptr<LoopPassConceptT>> LoopPasses; |
||
| 185 | std::vector<std::unique_ptr<LoopNestPassConceptT>> LoopNestPasses; |
||
| 186 | |||
| 187 | /// Run either a loop pass or a loop-nest pass. Returns `std::nullopt` if |
||
| 188 | /// PassInstrumentation's BeforePass returns false. Otherwise, returns the |
||
| 189 | /// preserved analyses of the pass. |
||
| 190 | template <typename IRUnitT, typename PassT> |
||
| 191 | std::optional<PreservedAnalyses> |
||
| 192 | runSinglePass(IRUnitT &IR, PassT &Pass, LoopAnalysisManager &AM, |
||
| 193 | LoopStandardAnalysisResults &AR, LPMUpdater &U, |
||
| 194 | PassInstrumentation &PI); |
||
| 195 | |||
| 196 | PreservedAnalyses runWithLoopNestPasses(Loop &L, LoopAnalysisManager &AM, |
||
| 197 | LoopStandardAnalysisResults &AR, |
||
| 198 | LPMUpdater &U); |
||
| 199 | PreservedAnalyses runWithoutLoopNestPasses(Loop &L, LoopAnalysisManager &AM, |
||
| 200 | LoopStandardAnalysisResults &AR, |
||
| 201 | LPMUpdater &U); |
||
| 202 | |||
| 203 | private: |
||
| 204 | static const Loop &getLoopFromIR(Loop &L) { return L; } |
||
| 205 | static const Loop &getLoopFromIR(LoopNest &LN) { |
||
| 206 | return LN.getOutermostLoop(); |
||
| 207 | } |
||
| 208 | }; |
||
| 209 | |||
| 210 | /// The Loop pass manager. |
||
| 211 | /// |
||
| 212 | /// See the documentation for the PassManager template for details. It runs |
||
| 213 | /// a sequence of Loop passes over each Loop that the manager is run over. This |
||
| 214 | /// typedef serves as a convenient way to refer to this construct. |
||
| 215 | typedef PassManager<Loop, LoopAnalysisManager, LoopStandardAnalysisResults &, |
||
| 216 | LPMUpdater &> |
||
| 217 | LoopPassManager; |
||
| 218 | |||
| 219 | /// A partial specialization of the require analysis template pass to forward |
||
| 220 | /// the extra parameters from a transformation's run method to the |
||
| 221 | /// AnalysisManager's getResult. |
||
| 222 | template <typename AnalysisT> |
||
| 223 | struct RequireAnalysisPass<AnalysisT, Loop, LoopAnalysisManager, |
||
| 224 | LoopStandardAnalysisResults &, LPMUpdater &> |
||
| 225 | : PassInfoMixin< |
||
| 226 | RequireAnalysisPass<AnalysisT, Loop, LoopAnalysisManager, |
||
| 227 | LoopStandardAnalysisResults &, LPMUpdater &>> { |
||
| 228 | PreservedAnalyses run(Loop &L, LoopAnalysisManager &AM, |
||
| 229 | LoopStandardAnalysisResults &AR, LPMUpdater &) { |
||
| 230 | (void)AM.template getResult<AnalysisT>(L, AR); |
||
| 231 | return PreservedAnalyses::all(); |
||
| 232 | } |
||
| 233 | void printPipeline(raw_ostream &OS, |
||
| 234 | function_ref<StringRef(StringRef)> MapClassName2PassName) { |
||
| 235 | auto ClassName = AnalysisT::name(); |
||
| 236 | auto PassName = MapClassName2PassName(ClassName); |
||
| 237 | OS << "require<" << PassName << ">"; |
||
| 238 | } |
||
| 239 | }; |
||
| 240 | |||
| 241 | /// An alias template to easily name a require analysis loop pass. |
||
| 242 | template <typename AnalysisT> |
||
| 243 | using RequireAnalysisLoopPass = |
||
| 244 | RequireAnalysisPass<AnalysisT, Loop, LoopAnalysisManager, |
||
| 245 | LoopStandardAnalysisResults &, LPMUpdater &>; |
||
| 246 | |||
| 247 | class FunctionToLoopPassAdaptor; |
||
| 248 | |||
| 249 | /// This class provides an interface for updating the loop pass manager based |
||
| 250 | /// on mutations to the loop nest. |
||
| 251 | /// |
||
| 252 | /// A reference to an instance of this class is passed as an argument to each |
||
| 253 | /// Loop pass, and Loop passes should use it to update LPM infrastructure if |
||
| 254 | /// they modify the loop nest structure. |
||
| 255 | /// |
||
| 256 | /// \c LPMUpdater comes with two modes: the loop mode and the loop-nest mode. In |
||
| 257 | /// loop mode, all the loops in the function will be pushed into the worklist |
||
| 258 | /// and when new loops are added to the pipeline, their subloops are also |
||
| 259 | /// inserted recursively. On the other hand, in loop-nest mode, only top-level |
||
| 260 | /// loops are contained in the worklist and the addition of new (top-level) |
||
| 261 | /// loops will not trigger the addition of their subloops. |
||
| 262 | class LPMUpdater { |
||
| 263 | public: |
||
| 264 | /// This can be queried by loop passes which run other loop passes (like pass |
||
| 265 | /// managers) to know whether the loop needs to be skipped due to updates to |
||
| 266 | /// the loop nest. |
||
| 267 | /// |
||
| 268 | /// If this returns true, the loop object may have been deleted, so passes |
||
| 269 | /// should take care not to touch the object. |
||
| 270 | bool skipCurrentLoop() const { return SkipCurrentLoop; } |
||
| 271 | |||
| 272 | /// Loop passes should use this method to indicate they have deleted a loop |
||
| 273 | /// from the nest. |
||
| 274 | /// |
||
| 275 | /// Note that this loop must either be the current loop or a subloop of the |
||
| 276 | /// current loop. This routine must be called prior to removing the loop from |
||
| 277 | /// the loop nest. |
||
| 278 | /// |
||
| 279 | /// If this is called for the current loop, in addition to clearing any |
||
| 280 | /// state, this routine will mark that the current loop should be skipped by |
||
| 281 | /// the rest of the pass management infrastructure. |
||
| 282 | void markLoopAsDeleted(Loop &L, llvm::StringRef Name) { |
||
| 283 | LAM.clear(L, Name); |
||
| 284 | assert((&L == CurrentL || CurrentL->contains(&L)) && |
||
| 285 | "Cannot delete a loop outside of the " |
||
| 286 | "subloop tree currently being processed."); |
||
| 287 | if (&L == CurrentL) |
||
| 288 | SkipCurrentLoop = true; |
||
| 289 | } |
||
| 290 | |||
| 291 | void setParentLoop(Loop *L) { |
||
| 292 | #ifdef LLVM_ENABLE_ABI_BREAKING_CHECKS |
||
| 293 | ParentL = L; |
||
| 294 | #endif |
||
| 295 | } |
||
| 296 | |||
| 297 | /// Loop passes should use this method to indicate they have added new child |
||
| 298 | /// loops of the current loop. |
||
| 299 | /// |
||
| 300 | /// \p NewChildLoops must contain only the immediate children. Any nested |
||
| 301 | /// loops within them will be visited in postorder as usual for the loop pass |
||
| 302 | /// manager. |
||
| 303 | void addChildLoops(ArrayRef<Loop *> NewChildLoops) { |
||
| 304 | assert(!LoopNestMode && |
||
| 305 | "Child loops should not be pushed in loop-nest mode."); |
||
| 306 | // Insert ourselves back into the worklist first, as this loop should be |
||
| 307 | // revisited after all the children have been processed. |
||
| 308 | Worklist.insert(CurrentL); |
||
| 309 | |||
| 310 | #ifndef NDEBUG |
||
| 311 | for (Loop *NewL : NewChildLoops) |
||
| 312 | assert(NewL->getParentLoop() == CurrentL && "All of the new loops must " |
||
| 313 | "be immediate children of " |
||
| 314 | "the current loop!"); |
||
| 315 | #endif |
||
| 316 | |||
| 317 | appendLoopsToWorklist(NewChildLoops, Worklist); |
||
| 318 | |||
| 319 | // Also skip further processing of the current loop--it will be revisited |
||
| 320 | // after all of its newly added children are accounted for. |
||
| 321 | SkipCurrentLoop = true; |
||
| 322 | } |
||
| 323 | |||
| 324 | /// Loop passes should use this method to indicate they have added new |
||
| 325 | /// sibling loops to the current loop. |
||
| 326 | /// |
||
| 327 | /// \p NewSibLoops must only contain the immediate sibling loops. Any nested |
||
| 328 | /// loops within them will be visited in postorder as usual for the loop pass |
||
| 329 | /// manager. |
||
| 330 | void addSiblingLoops(ArrayRef<Loop *> NewSibLoops) { |
||
| 331 | #if defined(LLVM_ENABLE_ABI_BREAKING_CHECKS) && !defined(NDEBUG) |
||
| 332 | for (Loop *NewL : NewSibLoops) |
||
| 333 | assert(NewL->getParentLoop() == ParentL && |
||
| 334 | "All of the new loops must be siblings of the current loop!"); |
||
| 335 | #endif |
||
| 336 | |||
| 337 | if (LoopNestMode) |
||
| 338 | Worklist.insert(NewSibLoops); |
||
| 339 | else |
||
| 340 | appendLoopsToWorklist(NewSibLoops, Worklist); |
||
| 341 | |||
| 342 | // No need to skip the current loop or revisit it, as sibling loops |
||
| 343 | // shouldn't impact anything. |
||
| 344 | } |
||
| 345 | |||
| 346 | /// Restart the current loop. |
||
| 347 | /// |
||
| 348 | /// Loop passes should call this method to indicate the current loop has been |
||
| 349 | /// sufficiently changed that it should be re-visited from the begining of |
||
| 350 | /// the loop pass pipeline rather than continuing. |
||
| 351 | void revisitCurrentLoop() { |
||
| 352 | // Tell the currently in-flight pipeline to stop running. |
||
| 353 | SkipCurrentLoop = true; |
||
| 354 | |||
| 355 | // And insert ourselves back into the worklist. |
||
| 356 | Worklist.insert(CurrentL); |
||
| 357 | } |
||
| 358 | |||
| 359 | bool isLoopNestChanged() const { |
||
| 360 | return LoopNestChanged; |
||
| 361 | } |
||
| 362 | |||
| 363 | /// Loopnest passes should use this method to indicate if the |
||
| 364 | /// loopnest has been modified. |
||
| 365 | void markLoopNestChanged(bool Changed) { |
||
| 366 | LoopNestChanged = Changed; |
||
| 367 | } |
||
| 368 | |||
| 369 | private: |
||
| 370 | friend class llvm::FunctionToLoopPassAdaptor; |
||
| 371 | |||
| 372 | /// The \c FunctionToLoopPassAdaptor's worklist of loops to process. |
||
| 373 | SmallPriorityWorklist<Loop *, 4> &Worklist; |
||
| 374 | |||
| 375 | /// The analysis manager for use in the current loop nest. |
||
| 376 | LoopAnalysisManager &LAM; |
||
| 377 | |||
| 378 | Loop *CurrentL; |
||
| 379 | bool SkipCurrentLoop; |
||
| 380 | const bool LoopNestMode; |
||
| 381 | bool LoopNestChanged; |
||
| 382 | |||
| 383 | #ifdef LLVM_ENABLE_ABI_BREAKING_CHECKS |
||
| 384 | // In debug builds we also track the parent loop to implement asserts even in |
||
| 385 | // the face of loop deletion. |
||
| 386 | Loop *ParentL; |
||
| 387 | #endif |
||
| 388 | |||
| 389 | LPMUpdater(SmallPriorityWorklist<Loop *, 4> &Worklist, |
||
| 390 | LoopAnalysisManager &LAM, bool LoopNestMode = false, |
||
| 391 | bool LoopNestChanged = false) |
||
| 392 | : Worklist(Worklist), LAM(LAM), LoopNestMode(LoopNestMode), |
||
| 393 | LoopNestChanged(LoopNestChanged) {} |
||
| 394 | }; |
||
| 395 | |||
| 396 | template <typename IRUnitT, typename PassT> |
||
| 397 | std::optional<PreservedAnalyses> LoopPassManager::runSinglePass( |
||
| 398 | IRUnitT &IR, PassT &Pass, LoopAnalysisManager &AM, |
||
| 399 | LoopStandardAnalysisResults &AR, LPMUpdater &U, PassInstrumentation &PI) { |
||
| 400 | // Get the loop in case of Loop pass and outermost loop in case of LoopNest |
||
| 401 | // pass which is to be passed to BeforePass and AfterPass call backs. |
||
| 402 | const Loop &L = getLoopFromIR(IR); |
||
| 403 | // Check the PassInstrumentation's BeforePass callbacks before running the |
||
| 404 | // pass, skip its execution completely if asked to (callback returns false). |
||
| 405 | if (!PI.runBeforePass<Loop>(*Pass, L)) |
||
| 406 | return std::nullopt; |
||
| 407 | |||
| 408 | PreservedAnalyses PA = Pass->run(IR, AM, AR, U); |
||
| 409 | |||
| 410 | // do not pass deleted Loop into the instrumentation |
||
| 411 | if (U.skipCurrentLoop()) |
||
| 412 | PI.runAfterPassInvalidated<IRUnitT>(*Pass, PA); |
||
| 413 | else |
||
| 414 | PI.runAfterPass<Loop>(*Pass, L, PA); |
||
| 415 | return PA; |
||
| 416 | } |
||
| 417 | |||
| 418 | /// Adaptor that maps from a function to its loops. |
||
| 419 | /// |
||
| 420 | /// Designed to allow composition of a LoopPass(Manager) and a |
||
| 421 | /// FunctionPassManager. Note that if this pass is constructed with a \c |
||
| 422 | /// FunctionAnalysisManager it will run the \c LoopAnalysisManagerFunctionProxy |
||
| 423 | /// analysis prior to running the loop passes over the function to enable a \c |
||
| 424 | /// LoopAnalysisManager to be used within this run safely. |
||
| 425 | /// |
||
| 426 | /// The adaptor comes with two modes: the loop mode and the loop-nest mode, and |
||
| 427 | /// the worklist updater lived inside will be in the same mode as the adaptor |
||
| 428 | /// (refer to the documentation of \c LPMUpdater for more detailed explanation). |
||
| 429 | /// Specifically, in loop mode, all loops in the funciton will be pushed into |
||
| 430 | /// the worklist and processed by \p Pass, while only top-level loops are |
||
| 431 | /// processed in loop-nest mode. Please refer to the various specializations of |
||
| 432 | /// \fn createLoopFunctionToLoopPassAdaptor to see when loop mode and loop-nest |
||
| 433 | /// mode are used. |
||
| 434 | class FunctionToLoopPassAdaptor |
||
| 435 | : public PassInfoMixin<FunctionToLoopPassAdaptor> { |
||
| 436 | public: |
||
| 437 | using PassConceptT = |
||
| 438 | detail::PassConcept<Loop, LoopAnalysisManager, |
||
| 439 | LoopStandardAnalysisResults &, LPMUpdater &>; |
||
| 440 | |||
| 441 | explicit FunctionToLoopPassAdaptor(std::unique_ptr<PassConceptT> Pass, |
||
| 442 | bool UseMemorySSA = false, |
||
| 443 | bool UseBlockFrequencyInfo = false, |
||
| 444 | bool UseBranchProbabilityInfo = false, |
||
| 445 | bool LoopNestMode = false) |
||
| 446 | : Pass(std::move(Pass)), UseMemorySSA(UseMemorySSA), |
||
| 447 | UseBlockFrequencyInfo(UseBlockFrequencyInfo), |
||
| 448 | UseBranchProbabilityInfo(UseBranchProbabilityInfo), |
||
| 449 | LoopNestMode(LoopNestMode) { |
||
| 450 | LoopCanonicalizationFPM.addPass(LoopSimplifyPass()); |
||
| 451 | LoopCanonicalizationFPM.addPass(LCSSAPass()); |
||
| 452 | } |
||
| 453 | |||
| 454 | /// Runs the loop passes across every loop in the function. |
||
| 455 | PreservedAnalyses run(Function &F, FunctionAnalysisManager &AM); |
||
| 456 | void printPipeline(raw_ostream &OS, |
||
| 457 | function_ref<StringRef(StringRef)> MapClassName2PassName); |
||
| 458 | |||
| 459 | static bool isRequired() { return true; } |
||
| 460 | |||
| 461 | bool isLoopNestMode() const { return LoopNestMode; } |
||
| 462 | |||
| 463 | private: |
||
| 464 | std::unique_ptr<PassConceptT> Pass; |
||
| 465 | |||
| 466 | FunctionPassManager LoopCanonicalizationFPM; |
||
| 467 | |||
| 468 | bool UseMemorySSA = false; |
||
| 469 | bool UseBlockFrequencyInfo = false; |
||
| 470 | bool UseBranchProbabilityInfo = false; |
||
| 471 | const bool LoopNestMode; |
||
| 472 | }; |
||
| 473 | |||
| 474 | /// A function to deduce a loop pass type and wrap it in the templated |
||
| 475 | /// adaptor. |
||
| 476 | /// |
||
| 477 | /// If \p Pass is a loop pass, the returned adaptor will be in loop mode. |
||
| 478 | template <typename LoopPassT> |
||
| 479 | inline std::enable_if_t<is_detected<HasRunOnLoopT, LoopPassT>::value, |
||
| 480 | FunctionToLoopPassAdaptor> |
||
| 481 | createFunctionToLoopPassAdaptor(LoopPassT &&Pass, bool UseMemorySSA = false, |
||
| 482 | bool UseBlockFrequencyInfo = false, |
||
| 483 | bool UseBranchProbabilityInfo = false) { |
||
| 484 | using PassModelT = |
||
| 485 | detail::PassModel<Loop, LoopPassT, PreservedAnalyses, LoopAnalysisManager, |
||
| 486 | LoopStandardAnalysisResults &, LPMUpdater &>; |
||
| 487 | // Do not use make_unique, it causes too many template instantiations, |
||
| 488 | // causing terrible compile times. |
||
| 489 | return FunctionToLoopPassAdaptor( |
||
| 490 | std::unique_ptr<FunctionToLoopPassAdaptor::PassConceptT>( |
||
| 491 | new PassModelT(std::forward<LoopPassT>(Pass))), |
||
| 492 | UseMemorySSA, UseBlockFrequencyInfo, UseBranchProbabilityInfo, false); |
||
| 493 | } |
||
| 494 | |||
| 495 | /// If \p Pass is a loop-nest pass, \p Pass will first be wrapped into a |
||
| 496 | /// \c LoopPassManager and the returned adaptor will be in loop-nest mode. |
||
| 497 | template <typename LoopNestPassT> |
||
| 498 | inline std::enable_if_t<!is_detected<HasRunOnLoopT, LoopNestPassT>::value, |
||
| 499 | FunctionToLoopPassAdaptor> |
||
| 500 | createFunctionToLoopPassAdaptor(LoopNestPassT &&Pass, bool UseMemorySSA = false, |
||
| 501 | bool UseBlockFrequencyInfo = false, |
||
| 502 | bool UseBranchProbabilityInfo = false) { |
||
| 503 | LoopPassManager LPM; |
||
| 504 | LPM.addPass(std::forward<LoopNestPassT>(Pass)); |
||
| 505 | using PassModelT = |
||
| 506 | detail::PassModel<Loop, LoopPassManager, PreservedAnalyses, |
||
| 507 | LoopAnalysisManager, LoopStandardAnalysisResults &, |
||
| 508 | LPMUpdater &>; |
||
| 509 | // Do not use make_unique, it causes too many template instantiations, |
||
| 510 | // causing terrible compile times. |
||
| 511 | return FunctionToLoopPassAdaptor( |
||
| 512 | std::unique_ptr<FunctionToLoopPassAdaptor::PassConceptT>( |
||
| 513 | new PassModelT(std::move(LPM))), |
||
| 514 | UseMemorySSA, UseBlockFrequencyInfo, UseBranchProbabilityInfo, true); |
||
| 515 | } |
||
| 516 | |||
| 517 | /// If \p Pass is an instance of \c LoopPassManager, the returned adaptor will |
||
| 518 | /// be in loop-nest mode if the pass manager contains only loop-nest passes. |
||
| 519 | template <> |
||
| 520 | inline FunctionToLoopPassAdaptor |
||
| 521 | createFunctionToLoopPassAdaptor<LoopPassManager>( |
||
| 522 | LoopPassManager &&LPM, bool UseMemorySSA, bool UseBlockFrequencyInfo, |
||
| 523 | bool UseBranchProbabilityInfo) { |
||
| 524 | // Check if LPM contains any loop pass and if it does not, returns an adaptor |
||
| 525 | // in loop-nest mode. |
||
| 526 | using PassModelT = |
||
| 527 | detail::PassModel<Loop, LoopPassManager, PreservedAnalyses, |
||
| 528 | LoopAnalysisManager, LoopStandardAnalysisResults &, |
||
| 529 | LPMUpdater &>; |
||
| 530 | bool LoopNestMode = (LPM.getNumLoopPasses() == 0); |
||
| 531 | // Do not use make_unique, it causes too many template instantiations, |
||
| 532 | // causing terrible compile times. |
||
| 533 | return FunctionToLoopPassAdaptor( |
||
| 534 | std::unique_ptr<FunctionToLoopPassAdaptor::PassConceptT>( |
||
| 535 | new PassModelT(std::move(LPM))), |
||
| 536 | UseMemorySSA, UseBlockFrequencyInfo, UseBranchProbabilityInfo, |
||
| 537 | LoopNestMode); |
||
| 538 | } |
||
| 539 | |||
| 540 | /// Pass for printing a loop's contents as textual IR. |
||
| 541 | class PrintLoopPass : public PassInfoMixin<PrintLoopPass> { |
||
| 542 | raw_ostream &OS; |
||
| 543 | std::string Banner; |
||
| 544 | |||
| 545 | public: |
||
| 546 | PrintLoopPass(); |
||
| 547 | PrintLoopPass(raw_ostream &OS, const std::string &Banner = ""); |
||
| 548 | |||
| 549 | PreservedAnalyses run(Loop &L, LoopAnalysisManager &, |
||
| 550 | LoopStandardAnalysisResults &, LPMUpdater &); |
||
| 551 | }; |
||
| 552 | } |
||
| 553 | |||
| 554 | #endif // LLVM_TRANSFORMS_SCALAR_LOOPPASSMANAGER_H |