Skip to content

Commit d7a6f3a

Browse files
committed
[LoopNest] Extend LPMUpdater and adaptor to handle loop-nest passes
This is a follow-up patch of D87045. The patch implements "loop-nest mode" for `LPMUpdater` and `FunctionToLoopPassAdaptor` in which only top-level loops are operated. `createFunctionToLoopPassAdaptor` decides whether the returned adaptor is in loop-nest mode or not based on the given pass. If the pass is a loop-nest pass or the pass is a `LoopPassManager` which contains only loop-nest passes, the loop-nest version of adaptor is returned; otherwise, the normal (loop) version of adaptor is returned. Reviewed By: Whitney Differential Revision: https://reviews.llvm.org/D87531
1 parent e18734f commit d7a6f3a

File tree

3 files changed

+159
-34
lines changed

3 files changed

+159
-34
lines changed

llvm/include/llvm/Transforms/Scalar/LoopPassManager.h

Lines changed: 91 additions & 14 deletions
Original file line numberDiff line numberDiff line change
@@ -53,6 +53,16 @@ namespace llvm {
5353
// Forward declarations of an update tracking API used in the pass manager.
5454
class LPMUpdater;
5555

56+
namespace {
57+
58+
template <typename PassT>
59+
using HasRunOnLoopT = decltype(std::declval<PassT>().run(
60+
std::declval<Loop &>(), std::declval<LoopAnalysisManager &>(),
61+
std::declval<LoopStandardAnalysisResults &>(),
62+
std::declval<LPMUpdater &>()));
63+
64+
} // namespace
65+
5666
// Explicit specialization and instantiation declarations for the pass manager.
5767
// See the comments on the definition of the specialization for details on how
5868
// it differs from the primary template.
@@ -62,13 +72,6 @@ class PassManager<Loop, LoopAnalysisManager, LoopStandardAnalysisResults &,
6272
: public PassInfoMixin<
6373
PassManager<Loop, LoopAnalysisManager, LoopStandardAnalysisResults &,
6474
LPMUpdater &>> {
65-
private:
66-
template <typename PassT>
67-
using HasRunOnLoopT = decltype(std::declval<PassT>().run(
68-
std::declval<Loop &>(), std::declval<LoopAnalysisManager &>(),
69-
std::declval<LoopStandardAnalysisResults &>(),
70-
std::declval<LPMUpdater &>()));
71-
7275
public:
7376
/// Construct a pass manager.
7477
///
@@ -154,6 +157,9 @@ class PassManager<Loop, LoopAnalysisManager, LoopStandardAnalysisResults &,
154157

155158
static bool isRequired() { return true; }
156159

160+
size_t getNumLoopPasses() const { return LoopPasses.size(); }
161+
size_t getNumLoopNestPasses() const { return LoopNestPasses.size(); }
162+
157163
protected:
158164
using LoopPassConceptT =
159165
detail::PassConcept<Loop, LoopAnalysisManager,
@@ -227,6 +233,13 @@ class FunctionToLoopPassAdaptor;
227233
/// A reference to an instance of this class is passed as an argument to each
228234
/// Loop pass, and Loop passes should use it to update LPM infrastructure if
229235
/// they modify the loop nest structure.
236+
///
237+
/// \c LPMUpdater comes with two modes: the loop mode and the loop-nest mode. In
238+
/// loop mode, all the loops in the function will be pushed into the worklist
239+
/// and when new loops are added to the pipeline, their subloops are also
240+
/// inserted recursively. On the other hand, in loop-nest mode, only top-level
241+
/// loops are contained in the worklist and the addition of new (top-level)
242+
/// loops will not trigger the addition of their subloops.
230243
class LPMUpdater {
231244
public:
232245
/// This can be queried by loop passes which run other loop passes (like pass
@@ -248,6 +261,8 @@ class LPMUpdater {
248261
/// state, this routine will mark that the current loop should be skipped by
249262
/// the rest of the pass management infrastructure.
250263
void markLoopAsDeleted(Loop &L, llvm::StringRef Name) {
264+
assert((!LoopNestMode || L.isOutermost()) &&
265+
"L should be a top-level loop in loop-nest mode.");
251266
LAM.clear(L, Name);
252267
assert((&L == CurrentL || CurrentL->contains(&L)) &&
253268
"Cannot delete a loop outside of the "
@@ -263,6 +278,8 @@ class LPMUpdater {
263278
/// loops within them will be visited in postorder as usual for the loop pass
264279
/// manager.
265280
void addChildLoops(ArrayRef<Loop *> NewChildLoops) {
281+
assert(!LoopNestMode &&
282+
"Child loops should not be pushed in loop-nest mode.");
266283
// Insert ourselves back into the worklist first, as this loop should be
267284
// revisited after all the children have been processed.
268285
Worklist.insert(CurrentL);
@@ -294,7 +311,10 @@ class LPMUpdater {
294311
"All of the new loops must be siblings of the current loop!");
295312
#endif
296313

297-
appendLoopsToWorklist(NewSibLoops, Worklist);
314+
if (LoopNestMode)
315+
Worklist.insert(NewSibLoops);
316+
else
317+
appendLoopsToWorklist(NewSibLoops, Worklist);
298318

299319
// No need to skip the current loop or revisit it, as sibling loops
300320
// shouldn't impact anything.
@@ -324,6 +344,7 @@ class LPMUpdater {
324344

325345
Loop *CurrentL;
326346
bool SkipCurrentLoop;
347+
const bool LoopNestMode;
327348

328349
#ifndef NDEBUG
329350
// In debug builds we also track the parent loop to implement asserts even in
@@ -332,8 +353,8 @@ class LPMUpdater {
332353
#endif
333354

334355
LPMUpdater(SmallPriorityWorklist<Loop *, 4> &Worklist,
335-
LoopAnalysisManager &LAM)
336-
: Worklist(Worklist), LAM(LAM) {}
356+
LoopAnalysisManager &LAM, bool LoopNestMode = false)
357+
: Worklist(Worklist), LAM(LAM), LoopNestMode(LoopNestMode) {}
337358
};
338359

339360
template <typename IRUnitT, typename PassT>
@@ -366,6 +387,15 @@ Optional<PreservedAnalyses> LoopPassManager::runSinglePass(
366387
/// FunctionAnalysisManager it will run the \c LoopAnalysisManagerFunctionProxy
367388
/// analysis prior to running the loop passes over the function to enable a \c
368389
/// LoopAnalysisManager to be used within this run safely.
390+
///
391+
/// The adaptor comes with two modes: the loop mode and the loop-nest mode, and
392+
/// the worklist updater lived inside will be in the same mode as the adaptor
393+
/// (refer to the documentation of \c LPMUpdater for more detailed explanation).
394+
/// Specifically, in loop mode, all loops in the funciton will be pushed into
395+
/// the worklist and processed by \p Pass, while only top-level loops are
396+
/// processed in loop-nest mode. Please refer to the various specializations of
397+
/// \fn createLoopFunctionToLoopPassAdaptor to see when loop mode and loop-nest
398+
/// mode are used.
369399
class FunctionToLoopPassAdaptor
370400
: public PassInfoMixin<FunctionToLoopPassAdaptor> {
371401
public:
@@ -376,10 +406,12 @@ class FunctionToLoopPassAdaptor
376406
explicit FunctionToLoopPassAdaptor(std::unique_ptr<PassConceptT> Pass,
377407
bool UseMemorySSA = false,
378408
bool UseBlockFrequencyInfo = false,
379-
bool DebugLogging = false)
409+
bool DebugLogging = false,
410+
bool LoopNestMode = false)
380411
: Pass(std::move(Pass)), LoopCanonicalizationFPM(DebugLogging),
381412
UseMemorySSA(UseMemorySSA),
382-
UseBlockFrequencyInfo(UseBlockFrequencyInfo) {
413+
UseBlockFrequencyInfo(UseBlockFrequencyInfo),
414+
LoopNestMode(LoopNestMode) {
383415
LoopCanonicalizationFPM.addPass(LoopSimplifyPass());
384416
LoopCanonicalizationFPM.addPass(LCSSAPass());
385417
}
@@ -389,19 +421,25 @@ class FunctionToLoopPassAdaptor
389421

390422
static bool isRequired() { return true; }
391423

424+
bool isLoopNestMode() const { return LoopNestMode; }
425+
392426
private:
393427
std::unique_ptr<PassConceptT> Pass;
394428

395429
FunctionPassManager LoopCanonicalizationFPM;
396430

397431
bool UseMemorySSA = false;
398432
bool UseBlockFrequencyInfo = false;
433+
const bool LoopNestMode;
399434
};
400435

401436
/// A function to deduce a loop pass type and wrap it in the templated
402437
/// adaptor.
438+
///
439+
/// If \p Pass is a loop pass, the returned adaptor will be in loop mode.
403440
template <typename LoopPassT>
404-
FunctionToLoopPassAdaptor
441+
inline std::enable_if_t<is_detected<HasRunOnLoopT, LoopPassT>::value,
442+
FunctionToLoopPassAdaptor>
405443
createFunctionToLoopPassAdaptor(LoopPassT Pass, bool UseMemorySSA = false,
406444
bool UseBlockFrequencyInfo = false,
407445
bool DebugLogging = false) {
@@ -410,7 +448,46 @@ createFunctionToLoopPassAdaptor(LoopPassT Pass, bool UseMemorySSA = false,
410448
LoopStandardAnalysisResults &, LPMUpdater &>;
411449
return FunctionToLoopPassAdaptor(
412450
std::make_unique<PassModelT>(std::move(Pass)), UseMemorySSA,
413-
UseBlockFrequencyInfo, DebugLogging);
451+
UseBlockFrequencyInfo, DebugLogging, false);
452+
}
453+
454+
/// If \p Pass is a loop-nest pass, \p Pass will first be wrapped into a
455+
/// \c LoopPassManager and the returned adaptor will be in loop-nest mode.
456+
template <typename LoopNestPassT>
457+
inline std::enable_if_t<!is_detected<HasRunOnLoopT, LoopNestPassT>::value,
458+
FunctionToLoopPassAdaptor>
459+
createFunctionToLoopPassAdaptor(LoopNestPassT Pass, bool UseMemorySSA = false,
460+
bool UseBlockFrequencyInfo = false,
461+
bool DebugLogging = false) {
462+
LoopPassManager LPM(DebugLogging);
463+
LPM.addPass(std::move(Pass));
464+
using PassModelT =
465+
detail::PassModel<Loop, LoopPassManager, PreservedAnalyses,
466+
LoopAnalysisManager, LoopStandardAnalysisResults &,
467+
LPMUpdater &>;
468+
return FunctionToLoopPassAdaptor(std::make_unique<PassModelT>(std::move(LPM)),
469+
UseMemorySSA, UseBlockFrequencyInfo,
470+
DebugLogging, true);
471+
}
472+
473+
/// If \p Pass is an instance of \c LoopPassManager, the returned adaptor will
474+
/// be in loop-nest mode if the pass manager contains only loop-nest passes.
475+
template <>
476+
inline FunctionToLoopPassAdaptor
477+
createFunctionToLoopPassAdaptor<LoopPassManager>(LoopPassManager LPM,
478+
bool UseMemorySSA,
479+
bool UseBlockFrequencyInfo,
480+
bool DebugLogging) {
481+
// Check if LPM contains any loop pass and if it does not, returns an adaptor
482+
// in loop-nest mode.
483+
using PassModelT =
484+
detail::PassModel<Loop, LoopPassManager, PreservedAnalyses,
485+
LoopAnalysisManager, LoopStandardAnalysisResults &,
486+
LPMUpdater &>;
487+
bool LoopNestMode = (LPM.getNumLoopPasses() == 0);
488+
return FunctionToLoopPassAdaptor(std::make_unique<PassModelT>(std::move(LPM)),
489+
UseMemorySSA, UseBlockFrequencyInfo,
490+
DebugLogging, LoopNestMode);
414491
}
415492

416493
/// Pass for printing a loop's contents as textual IR.

llvm/lib/Transforms/Scalar/LoopPassManager.cpp

Lines changed: 9 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -222,11 +222,16 @@ PreservedAnalyses FunctionToLoopPassAdaptor::run(Function &F,
222222

223223
// Register the worklist and loop analysis manager so that loop passes can
224224
// update them when they mutate the loop nest structure.
225-
LPMUpdater Updater(Worklist, LAM);
225+
LPMUpdater Updater(Worklist, LAM, LoopNestMode);
226226

227227
// Add the loop nests in the reverse order of LoopInfo. See method
228228
// declaration.
229-
appendLoopsToWorklist(LI, Worklist);
229+
if (!LoopNestMode) {
230+
appendLoopsToWorklist(LI, Worklist);
231+
} else {
232+
for (Loop *L : LI)
233+
Worklist.insert(L);
234+
}
230235

231236
#ifndef NDEBUG
232237
PI.pushBeforeNonSkippedPassCallback([&LAR, &LI](StringRef PassID, Any IR) {
@@ -247,6 +252,8 @@ PreservedAnalyses FunctionToLoopPassAdaptor::run(Function &F,
247252

248253
do {
249254
Loop *L = Worklist.pop_back_val();
255+
assert(!(LoopNestMode && L->getParentLoop()) &&
256+
"L should be a top-level loop in loop-nest mode.");
250257

251258
// Reset the update structure for this loop.
252259
Updater.CurrentL = L;

llvm/unittests/Transforms/Scalar/LoopPassManagerTest.cpp

Lines changed: 59 additions & 18 deletions
Original file line numberDiff line numberDiff line change
@@ -1603,28 +1603,69 @@ TEST_F(LoopPassManagerTest, LoopDeletion) {
16031603
}
16041604

16051605
TEST_F(LoopPassManagerTest, HandleLoopNestPass) {
1606-
::testing::InSequence MakeExpectationsSequenced;
1606+
::testing::Sequence FSequence, GSequence;
16071607

1608-
EXPECT_CALL(MLPHandle, run(HasName("loop.0.0"), _, _, _)).Times(2);
1609-
EXPECT_CALL(MLPHandle, run(HasName("loop.0.1"), _, _, _)).Times(2);
1610-
EXPECT_CALL(MLPHandle, run(HasName("loop.0"), _, _, _));
1611-
EXPECT_CALL(MLNPHandle, run(HasName("loop.0"), _, _, _));
1612-
EXPECT_CALL(MLPHandle, run(HasName("loop.0"), _, _, _));
1613-
EXPECT_CALL(MLNPHandle, run(HasName("loop.0"), _, _, _));
1614-
EXPECT_CALL(MLPHandle, run(HasName("loop.g.0"), _, _, _));
1615-
EXPECT_CALL(MLNPHandle, run(HasName("loop.g.0"), _, _, _));
1616-
EXPECT_CALL(MLPHandle, run(HasName("loop.g.0"), _, _, _));
1617-
EXPECT_CALL(MLNPHandle, run(HasName("loop.g.0"), _, _, _));
1608+
EXPECT_CALL(MLPHandle, run(HasName("loop.0.0"), _, _, _))
1609+
.Times(2)
1610+
.InSequence(FSequence);
1611+
EXPECT_CALL(MLPHandle, run(HasName("loop.0.1"), _, _, _))
1612+
.Times(2)
1613+
.InSequence(FSequence);
1614+
EXPECT_CALL(MLPHandle, run(HasName("loop.0"), _, _, _)).InSequence(FSequence);
1615+
EXPECT_CALL(MLNPHandle, run(HasName("loop.0"), _, _, _))
1616+
.InSequence(FSequence);
1617+
EXPECT_CALL(MLPHandle, run(HasName("loop.0"), _, _, _)).InSequence(FSequence);
1618+
EXPECT_CALL(MLNPHandle, run(HasName("loop.0"), _, _, _))
1619+
.InSequence(FSequence);
1620+
EXPECT_CALL(MLPHandle, run(HasName("loop.g.0"), _, _, _))
1621+
.InSequence(GSequence);
1622+
EXPECT_CALL(MLNPHandle, run(HasName("loop.g.0"), _, _, _))
1623+
.InSequence(GSequence);
1624+
EXPECT_CALL(MLPHandle, run(HasName("loop.g.0"), _, _, _))
1625+
.InSequence(GSequence);
1626+
EXPECT_CALL(MLNPHandle, run(HasName("loop.g.0"), _, _, _))
1627+
.InSequence(GSequence);
16181628

1619-
LoopPassManager LPM(true);
1620-
LPM.addPass(MLPHandle.getPass());
1621-
LPM.addPass(MLNPHandle.getPass());
1622-
LPM.addPass(MLPHandle.getPass());
1623-
LPM.addPass(MLNPHandle.getPass());
1629+
EXPECT_CALL(MLNPHandle, run(HasName("loop.0"), _, _, _))
1630+
.InSequence(FSequence);
1631+
EXPECT_CALL(MLNPHandle, run(HasName("loop.g.0"), _, _, _))
1632+
.InSequence(GSequence);
1633+
1634+
EXPECT_CALL(MLNPHandle, run(HasName("loop.0"), _, _, _))
1635+
.InSequence(FSequence);
1636+
EXPECT_CALL(MLNPHandle, run(HasName("loop.g.0"), _, _, _))
1637+
.InSequence(GSequence);
16241638

16251639
ModulePassManager MPM(true);
1626-
MPM.addPass(createModuleToFunctionPassAdaptor(
1627-
createFunctionToLoopPassAdaptor(std::move(LPM))));
1640+
FunctionPassManager FPM(true);
1641+
1642+
{
1643+
LoopPassManager LPM(true);
1644+
LPM.addPass(MLPHandle.getPass());
1645+
LPM.addPass(MLNPHandle.getPass());
1646+
LPM.addPass(MLPHandle.getPass());
1647+
LPM.addPass(MLNPHandle.getPass());
1648+
1649+
auto Adaptor = createFunctionToLoopPassAdaptor(std::move(LPM));
1650+
ASSERT_FALSE(Adaptor.isLoopNestMode());
1651+
FPM.addPass(std::move(Adaptor));
1652+
}
1653+
1654+
{
1655+
auto Adaptor = createFunctionToLoopPassAdaptor(MLNPHandle.getPass());
1656+
ASSERT_TRUE(Adaptor.isLoopNestMode());
1657+
FPM.addPass(std::move(Adaptor));
1658+
}
1659+
1660+
{
1661+
LoopPassManager LPM(true);
1662+
LPM.addPass(MLNPHandle.getPass());
1663+
auto Adaptor = createFunctionToLoopPassAdaptor(MLNPHandle.getPass());
1664+
ASSERT_TRUE(Adaptor.isLoopNestMode());
1665+
FPM.addPass(std::move(Adaptor));
1666+
}
1667+
1668+
MPM.addPass(createModuleToFunctionPassAdaptor(std::move(FPM)));
16281669
MPM.run(*M, MAM);
16291670
}
16301671

0 commit comments

Comments
 (0)