Subversion Repositories QNX 8.QNX8 LLVM/Clang compiler suite

Rev

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