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 |