1 //===- llvm/unittest/IR/PassManager.cpp - PassManager tests ---------------===//
2 //
3 //                     The LLVM Compiler Infrastructure
4 //
5 // This file is distributed under the University of Illinois Open Source
6 // License. See LICENSE.TXT for details.
7 //
8 //===----------------------------------------------------------------------===//
9 
10 #include "llvm/IR/PassManager.h"
11 #include "llvm/AsmParser/Parser.h"
12 #include "llvm/IR/Function.h"
13 #include "llvm/IR/LLVMContext.h"
14 #include "llvm/IR/Module.h"
15 #include "llvm/Support/SourceMgr.h"
16 #include "gtest/gtest.h"
17 
18 using namespace llvm;
19 
20 namespace {
21 
22 class TestFunctionAnalysis : public AnalysisInfoMixin<TestFunctionAnalysis> {
23 public:
24   struct Result {
Result__anon2a8083420111::TestFunctionAnalysis::Result25     Result(int Count) : InstructionCount(Count) {}
26     int InstructionCount;
27   };
28 
TestFunctionAnalysis(int & Runs)29   TestFunctionAnalysis(int &Runs) : Runs(Runs) {}
30 
31   /// Run the analysis pass over the function and return a result.
run(Function & F,FunctionAnalysisManager & AM)32   Result run(Function &F, FunctionAnalysisManager &AM) {
33     ++Runs;
34     int Count = 0;
35     for (Function::iterator BBI = F.begin(), BBE = F.end(); BBI != BBE; ++BBI)
36       for (BasicBlock::iterator II = BBI->begin(), IE = BBI->end(); II != IE;
37            ++II)
38         ++Count;
39     return Result(Count);
40   }
41 
42 private:
43   friend AnalysisInfoMixin<TestFunctionAnalysis>;
44   static AnalysisKey Key;
45 
46   int &Runs;
47 };
48 
49 AnalysisKey TestFunctionAnalysis::Key;
50 
51 class TestModuleAnalysis : public AnalysisInfoMixin<TestModuleAnalysis> {
52 public:
53   struct Result {
Result__anon2a8083420111::TestModuleAnalysis::Result54     Result(int Count) : FunctionCount(Count) {}
55     int FunctionCount;
56   };
57 
TestModuleAnalysis(int & Runs)58   TestModuleAnalysis(int &Runs) : Runs(Runs) {}
59 
run(Module & M,ModuleAnalysisManager & AM)60   Result run(Module &M, ModuleAnalysisManager &AM) {
61     ++Runs;
62     int Count = 0;
63     for (Module::iterator I = M.begin(), E = M.end(); I != E; ++I)
64       ++Count;
65     return Result(Count);
66   }
67 
68 private:
69   friend AnalysisInfoMixin<TestModuleAnalysis>;
70   static AnalysisKey Key;
71 
72   int &Runs;
73 };
74 
75 AnalysisKey TestModuleAnalysis::Key;
76 
77 struct TestModulePass : PassInfoMixin<TestModulePass> {
TestModulePass__anon2a8083420111::TestModulePass78   TestModulePass(int &RunCount) : RunCount(RunCount) {}
79 
run__anon2a8083420111::TestModulePass80   PreservedAnalyses run(Module &M, ModuleAnalysisManager &) {
81     ++RunCount;
82     return PreservedAnalyses::none();
83   }
84 
85   int &RunCount;
86 };
87 
88 struct TestPreservingModulePass : PassInfoMixin<TestPreservingModulePass> {
run__anon2a8083420111::TestPreservingModulePass89   PreservedAnalyses run(Module &M, ModuleAnalysisManager &) {
90     return PreservedAnalyses::all();
91   }
92 };
93 
94 struct TestFunctionPass : PassInfoMixin<TestFunctionPass> {
TestFunctionPass__anon2a8083420111::TestFunctionPass95   TestFunctionPass(int &RunCount, int &AnalyzedInstrCount,
96                    int &AnalyzedFunctionCount,
97                    bool OnlyUseCachedResults = false)
98       : RunCount(RunCount), AnalyzedInstrCount(AnalyzedInstrCount),
99         AnalyzedFunctionCount(AnalyzedFunctionCount),
100         OnlyUseCachedResults(OnlyUseCachedResults) {}
101 
run__anon2a8083420111::TestFunctionPass102   PreservedAnalyses run(Function &F, FunctionAnalysisManager &AM) {
103     ++RunCount;
104 
105     const ModuleAnalysisManager &MAM =
106         AM.getResult<ModuleAnalysisManagerFunctionProxy>(F).getManager();
107     if (TestModuleAnalysis::Result *TMA =
108             MAM.getCachedResult<TestModuleAnalysis>(*F.getParent()))
109       AnalyzedFunctionCount += TMA->FunctionCount;
110 
111     if (OnlyUseCachedResults) {
112       // Hack to force the use of the cached interface.
113       if (TestFunctionAnalysis::Result *AR =
114               AM.getCachedResult<TestFunctionAnalysis>(F))
115         AnalyzedInstrCount += AR->InstructionCount;
116     } else {
117       // Typical path just runs the analysis as needed.
118       TestFunctionAnalysis::Result &AR = AM.getResult<TestFunctionAnalysis>(F);
119       AnalyzedInstrCount += AR.InstructionCount;
120     }
121 
122     return PreservedAnalyses::all();
123   }
124 
125   int &RunCount;
126   int &AnalyzedInstrCount;
127   int &AnalyzedFunctionCount;
128   bool OnlyUseCachedResults;
129 };
130 
131 // A test function pass that invalidates all function analyses for a function
132 // with a specific name.
133 struct TestInvalidationFunctionPass
134     : PassInfoMixin<TestInvalidationFunctionPass> {
TestInvalidationFunctionPass__anon2a8083420111::TestInvalidationFunctionPass135   TestInvalidationFunctionPass(StringRef FunctionName) : Name(FunctionName) {}
136 
run__anon2a8083420111::TestInvalidationFunctionPass137   PreservedAnalyses run(Function &F, FunctionAnalysisManager &AM) {
138     return F.getName() == Name ? PreservedAnalyses::none()
139                                : PreservedAnalyses::all();
140   }
141 
142   StringRef Name;
143 };
144 
parseIR(LLVMContext & Context,const char * IR)145 std::unique_ptr<Module> parseIR(LLVMContext &Context, const char *IR) {
146   SMDiagnostic Err;
147   return parseAssemblyString(IR, Err, Context);
148 }
149 
150 class PassManagerTest : public ::testing::Test {
151 protected:
152   LLVMContext Context;
153   std::unique_ptr<Module> M;
154 
155 public:
PassManagerTest()156   PassManagerTest()
157       : M(parseIR(Context, "define void @f() {\n"
158                            "entry:\n"
159                            "  call void @g()\n"
160                            "  call void @h()\n"
161                            "  ret void\n"
162                            "}\n"
163                            "define void @g() {\n"
164                            "  ret void\n"
165                            "}\n"
166                            "define void @h() {\n"
167                            "  ret void\n"
168                            "}\n")) {}
169 };
170 
TEST(PreservedAnalysesTest,Basic)171 TEST(PreservedAnalysesTest, Basic) {
172   PreservedAnalyses PA1 = PreservedAnalyses();
173   {
174     auto PAC = PA1.getChecker<TestFunctionAnalysis>();
175     EXPECT_FALSE(PAC.preserved());
176     EXPECT_FALSE(PAC.preservedSet<AllAnalysesOn<Function>>());
177   }
178   {
179     auto PAC = PA1.getChecker<TestModuleAnalysis>();
180     EXPECT_FALSE(PAC.preserved());
181     EXPECT_FALSE(PAC.preservedSet<AllAnalysesOn<Module>>());
182   }
183   auto PA2 = PreservedAnalyses::none();
184   {
185     auto PAC = PA2.getChecker<TestFunctionAnalysis>();
186     EXPECT_FALSE(PAC.preserved());
187     EXPECT_FALSE(PAC.preservedSet<AllAnalysesOn<Function>>());
188   }
189   auto PA3 = PreservedAnalyses::all();
190   {
191     auto PAC = PA3.getChecker<TestFunctionAnalysis>();
192     EXPECT_TRUE(PAC.preserved());
193     EXPECT_TRUE(PAC.preservedSet<AllAnalysesOn<Function>>());
194   }
195   PreservedAnalyses PA4 = PA1;
196   {
197     auto PAC = PA4.getChecker<TestFunctionAnalysis>();
198     EXPECT_FALSE(PAC.preserved());
199     EXPECT_FALSE(PAC.preservedSet<AllAnalysesOn<Function>>());
200   }
201   PA4 = PA3;
202   {
203     auto PAC = PA4.getChecker<TestFunctionAnalysis>();
204     EXPECT_TRUE(PAC.preserved());
205     EXPECT_TRUE(PAC.preservedSet<AllAnalysesOn<Function>>());
206   }
207   PA4 = std::move(PA2);
208   {
209     auto PAC = PA4.getChecker<TestFunctionAnalysis>();
210     EXPECT_FALSE(PAC.preserved());
211     EXPECT_FALSE(PAC.preservedSet<AllAnalysesOn<Function>>());
212   }
213   auto PA5 = PreservedAnalyses::allInSet<AllAnalysesOn<Function>>();
214   {
215     auto PAC = PA5.getChecker<TestFunctionAnalysis>();
216     EXPECT_FALSE(PAC.preserved());
217     EXPECT_TRUE(PAC.preservedSet<AllAnalysesOn<Function>>());
218     EXPECT_FALSE(PAC.preservedSet<AllAnalysesOn<Module>>());
219   }
220 }
221 
TEST(PreservedAnalysesTest,Preserve)222 TEST(PreservedAnalysesTest, Preserve) {
223   auto PA = PreservedAnalyses::none();
224   PA.preserve<TestFunctionAnalysis>();
225   EXPECT_TRUE(PA.getChecker<TestFunctionAnalysis>().preserved());
226   EXPECT_FALSE(PA.getChecker<TestModuleAnalysis>().preserved());
227   PA.preserve<TestModuleAnalysis>();
228   EXPECT_TRUE(PA.getChecker<TestFunctionAnalysis>().preserved());
229   EXPECT_TRUE(PA.getChecker<TestModuleAnalysis>().preserved());
230 
231   // Redundant calls are fine.
232   PA.preserve<TestFunctionAnalysis>();
233   EXPECT_TRUE(PA.getChecker<TestFunctionAnalysis>().preserved());
234   EXPECT_TRUE(PA.getChecker<TestModuleAnalysis>().preserved());
235 }
236 
TEST(PreservedAnalysesTest,PreserveSets)237 TEST(PreservedAnalysesTest, PreserveSets) {
238   auto PA = PreservedAnalyses::none();
239   PA.preserveSet<AllAnalysesOn<Function>>();
240   EXPECT_TRUE(PA.getChecker<TestFunctionAnalysis>()
241                   .preservedSet<AllAnalysesOn<Function>>());
242   EXPECT_FALSE(PA.getChecker<TestModuleAnalysis>()
243                    .preservedSet<AllAnalysesOn<Module>>());
244   PA.preserveSet<AllAnalysesOn<Module>>();
245   EXPECT_TRUE(PA.getChecker<TestFunctionAnalysis>()
246                   .preservedSet<AllAnalysesOn<Function>>());
247   EXPECT_TRUE(PA.getChecker<TestModuleAnalysis>()
248                   .preservedSet<AllAnalysesOn<Module>>());
249 
250   // Mixing is fine.
251   PA.preserve<TestFunctionAnalysis>();
252   EXPECT_TRUE(PA.getChecker<TestFunctionAnalysis>()
253                   .preservedSet<AllAnalysesOn<Function>>());
254   EXPECT_TRUE(PA.getChecker<TestModuleAnalysis>()
255                   .preservedSet<AllAnalysesOn<Module>>());
256 
257   // Redundant calls are fine.
258   PA.preserveSet<AllAnalysesOn<Module>>();
259   EXPECT_TRUE(PA.getChecker<TestFunctionAnalysis>()
260                   .preservedSet<AllAnalysesOn<Function>>());
261   EXPECT_TRUE(PA.getChecker<TestModuleAnalysis>()
262                   .preservedSet<AllAnalysesOn<Module>>());
263 }
264 
TEST(PreservedAnalysisTest,Intersect)265 TEST(PreservedAnalysisTest, Intersect) {
266   // Setup the initial sets.
267   auto PA1 = PreservedAnalyses::none();
268   PA1.preserve<TestFunctionAnalysis>();
269   PA1.preserveSet<AllAnalysesOn<Module>>();
270   auto PA2 = PreservedAnalyses::none();
271   PA2.preserve<TestFunctionAnalysis>();
272   PA2.preserveSet<AllAnalysesOn<Function>>();
273   PA2.preserve<TestModuleAnalysis>();
274   PA2.preserveSet<AllAnalysesOn<Module>>();
275   auto PA3 = PreservedAnalyses::none();
276   PA3.preserve<TestModuleAnalysis>();
277   PA3.preserveSet<AllAnalysesOn<Function>>();
278 
279   // Self intersection is a no-op.
280   auto Intersected = PA1;
281   Intersected.intersect(PA1);
282   EXPECT_TRUE(Intersected.getChecker<TestFunctionAnalysis>().preserved());
283   EXPECT_FALSE(Intersected.getChecker<TestFunctionAnalysis>()
284                    .preservedSet<AllAnalysesOn<Function>>());
285   EXPECT_FALSE(Intersected.getChecker<TestModuleAnalysis>().preserved());
286   EXPECT_TRUE(Intersected.getChecker<TestModuleAnalysis>()
287                   .preservedSet<AllAnalysesOn<Module>>());
288 
289   // Intersecting with all is a no-op.
290   Intersected.intersect(PreservedAnalyses::all());
291   EXPECT_TRUE(Intersected.getChecker<TestFunctionAnalysis>().preserved());
292   EXPECT_FALSE(Intersected.getChecker<TestFunctionAnalysis>()
293                    .preservedSet<AllAnalysesOn<Function>>());
294   EXPECT_FALSE(Intersected.getChecker<TestModuleAnalysis>().preserved());
295   EXPECT_TRUE(Intersected.getChecker<TestModuleAnalysis>()
296                   .preservedSet<AllAnalysesOn<Module>>());
297 
298   // Intersecting a narrow set with a more broad set is the narrow set.
299   Intersected.intersect(PA2);
300   EXPECT_TRUE(Intersected.getChecker<TestFunctionAnalysis>().preserved());
301   EXPECT_FALSE(Intersected.getChecker<TestFunctionAnalysis>()
302                    .preservedSet<AllAnalysesOn<Function>>());
303   EXPECT_FALSE(Intersected.getChecker<TestModuleAnalysis>().preserved());
304   EXPECT_TRUE(Intersected.getChecker<TestModuleAnalysis>()
305                   .preservedSet<AllAnalysesOn<Module>>());
306 
307   // Intersecting a broad set with a more narrow set is the narrow set.
308   Intersected = PA2;
309   Intersected.intersect(PA1);
310   EXPECT_TRUE(Intersected.getChecker<TestFunctionAnalysis>().preserved());
311   EXPECT_FALSE(Intersected.getChecker<TestFunctionAnalysis>()
312                    .preservedSet<AllAnalysesOn<Function>>());
313   EXPECT_FALSE(Intersected.getChecker<TestModuleAnalysis>().preserved());
314   EXPECT_TRUE(Intersected.getChecker<TestModuleAnalysis>()
315                   .preservedSet<AllAnalysesOn<Module>>());
316 
317   // Intersecting with empty clears.
318   Intersected.intersect(PreservedAnalyses::none());
319   EXPECT_FALSE(Intersected.getChecker<TestFunctionAnalysis>().preserved());
320   EXPECT_FALSE(Intersected.getChecker<TestFunctionAnalysis>()
321                    .preservedSet<AllAnalysesOn<Function>>());
322   EXPECT_FALSE(Intersected.getChecker<TestModuleAnalysis>().preserved());
323   EXPECT_FALSE(Intersected.getChecker<TestModuleAnalysis>()
324                    .preservedSet<AllAnalysesOn<Module>>());
325 
326   // Intersecting non-overlapping clears.
327   Intersected = PA1;
328   Intersected.intersect(PA3);
329   EXPECT_FALSE(Intersected.getChecker<TestFunctionAnalysis>().preserved());
330   EXPECT_FALSE(Intersected.getChecker<TestFunctionAnalysis>()
331                    .preservedSet<AllAnalysesOn<Function>>());
332   EXPECT_FALSE(Intersected.getChecker<TestModuleAnalysis>().preserved());
333   EXPECT_FALSE(Intersected.getChecker<TestModuleAnalysis>()
334                    .preservedSet<AllAnalysesOn<Module>>());
335 
336   // Intersecting with moves works in when there is storage on both sides.
337   Intersected = PA1;
338   auto Tmp = PA2;
339   Intersected.intersect(std::move(Tmp));
340   EXPECT_TRUE(Intersected.getChecker<TestFunctionAnalysis>().preserved());
341   EXPECT_FALSE(Intersected.getChecker<TestFunctionAnalysis>()
342                    .preservedSet<AllAnalysesOn<Function>>());
343   EXPECT_FALSE(Intersected.getChecker<TestModuleAnalysis>().preserved());
344   EXPECT_TRUE(Intersected.getChecker<TestModuleAnalysis>()
345                   .preservedSet<AllAnalysesOn<Module>>());
346 
347   // Intersecting with move works for incoming all and existing all.
348   auto Tmp2 = PreservedAnalyses::all();
349   Intersected.intersect(std::move(Tmp2));
350   EXPECT_TRUE(Intersected.getChecker<TestFunctionAnalysis>().preserved());
351   EXPECT_FALSE(Intersected.getChecker<TestFunctionAnalysis>()
352                    .preservedSet<AllAnalysesOn<Function>>());
353   EXPECT_FALSE(Intersected.getChecker<TestModuleAnalysis>().preserved());
354   EXPECT_TRUE(Intersected.getChecker<TestModuleAnalysis>()
355                   .preservedSet<AllAnalysesOn<Module>>());
356   Intersected = PreservedAnalyses::all();
357   auto Tmp3 = PA1;
358   Intersected.intersect(std::move(Tmp3));
359   EXPECT_TRUE(Intersected.getChecker<TestFunctionAnalysis>().preserved());
360   EXPECT_FALSE(Intersected.getChecker<TestFunctionAnalysis>()
361                    .preservedSet<AllAnalysesOn<Function>>());
362   EXPECT_FALSE(Intersected.getChecker<TestModuleAnalysis>().preserved());
363   EXPECT_TRUE(Intersected.getChecker<TestModuleAnalysis>()
364                   .preservedSet<AllAnalysesOn<Module>>());
365 }
366 
TEST(PreservedAnalysisTest,Abandon)367 TEST(PreservedAnalysisTest, Abandon) {
368   auto PA = PreservedAnalyses::none();
369 
370   // We can abandon things after they are preserved.
371   PA.preserve<TestFunctionAnalysis>();
372   PA.abandon<TestFunctionAnalysis>();
373   EXPECT_FALSE(PA.getChecker<TestFunctionAnalysis>().preserved());
374 
375   // Repeated is fine, and abandoning if they were never preserved is fine.
376   PA.abandon<TestFunctionAnalysis>();
377   EXPECT_FALSE(PA.getChecker<TestFunctionAnalysis>().preserved());
378   PA.abandon<TestModuleAnalysis>();
379   EXPECT_FALSE(PA.getChecker<TestModuleAnalysis>().preserved());
380 
381   // Even if the sets are preserved, the abandoned analyses' checker won't
382   // return true for those sets.
383   PA.preserveSet<AllAnalysesOn<Function>>();
384   PA.preserveSet<AllAnalysesOn<Module>>();
385   EXPECT_FALSE(PA.getChecker<TestFunctionAnalysis>()
386                    .preservedSet<AllAnalysesOn<Function>>());
387   EXPECT_FALSE(PA.getChecker<TestModuleAnalysis>()
388                    .preservedSet<AllAnalysesOn<Module>>());
389 
390   // But an arbitrary (opaque) analysis will still observe the sets as
391   // preserved. This also checks that we can use an explicit ID rather than
392   // a type.
393   AnalysisKey FakeKey, *FakeID = &FakeKey;
394   EXPECT_TRUE(PA.getChecker(FakeID).preservedSet<AllAnalysesOn<Function>>());
395   EXPECT_TRUE(PA.getChecker(FakeID).preservedSet<AllAnalysesOn<Module>>());
396 }
397 
TEST_F(PassManagerTest,Basic)398 TEST_F(PassManagerTest, Basic) {
399   FunctionAnalysisManager FAM(/*DebugLogging*/ true);
400   int FunctionAnalysisRuns = 0;
401   FAM.registerPass([&] { return TestFunctionAnalysis(FunctionAnalysisRuns); });
402 
403   ModuleAnalysisManager MAM(/*DebugLogging*/ true);
404   int ModuleAnalysisRuns = 0;
405   MAM.registerPass([&] { return TestModuleAnalysis(ModuleAnalysisRuns); });
406   MAM.registerPass([&] { return FunctionAnalysisManagerModuleProxy(FAM); });
407   FAM.registerPass([&] { return ModuleAnalysisManagerFunctionProxy(MAM); });
408 
409   ModulePassManager MPM;
410 
411   // Count the runs over a Function.
412   int FunctionPassRunCount1 = 0;
413   int AnalyzedInstrCount1 = 0;
414   int AnalyzedFunctionCount1 = 0;
415   {
416     // Pointless scoped copy to test move assignment.
417     ModulePassManager NestedMPM(/*DebugLogging*/ true);
418     FunctionPassManager FPM;
419     {
420       // Pointless scope to test move assignment.
421       FunctionPassManager NestedFPM(/*DebugLogging*/ true);
422       NestedFPM.addPass(TestFunctionPass(
423           FunctionPassRunCount1, AnalyzedInstrCount1, AnalyzedFunctionCount1));
424       FPM = std::move(NestedFPM);
425     }
426     NestedMPM.addPass(createModuleToFunctionPassAdaptor(std::move(FPM)));
427     MPM = std::move(NestedMPM);
428   }
429 
430   // Count the runs over a module.
431   int ModulePassRunCount = 0;
432   MPM.addPass(TestModulePass(ModulePassRunCount));
433 
434   // Count the runs over a Function in a separate manager.
435   int FunctionPassRunCount2 = 0;
436   int AnalyzedInstrCount2 = 0;
437   int AnalyzedFunctionCount2 = 0;
438   {
439     FunctionPassManager FPM(/*DebugLogging*/ true);
440     FPM.addPass(TestFunctionPass(FunctionPassRunCount2, AnalyzedInstrCount2,
441                                  AnalyzedFunctionCount2));
442     MPM.addPass(createModuleToFunctionPassAdaptor(std::move(FPM)));
443   }
444 
445   // A third function pass manager but with only preserving intervening passes
446   // and with a function pass that invalidates exactly one analysis.
447   MPM.addPass(TestPreservingModulePass());
448   int FunctionPassRunCount3 = 0;
449   int AnalyzedInstrCount3 = 0;
450   int AnalyzedFunctionCount3 = 0;
451   {
452     FunctionPassManager FPM(/*DebugLogging*/ true);
453     FPM.addPass(TestFunctionPass(FunctionPassRunCount3, AnalyzedInstrCount3,
454                                  AnalyzedFunctionCount3));
455     FPM.addPass(TestInvalidationFunctionPass("f"));
456     MPM.addPass(createModuleToFunctionPassAdaptor(std::move(FPM)));
457   }
458 
459   // A fourth function pass manager but with only preserving intervening
460   // passes but triggering the module analysis.
461   MPM.addPass(RequireAnalysisPass<TestModuleAnalysis, Module>());
462   int FunctionPassRunCount4 = 0;
463   int AnalyzedInstrCount4 = 0;
464   int AnalyzedFunctionCount4 = 0;
465   {
466     FunctionPassManager FPM;
467     FPM.addPass(TestFunctionPass(FunctionPassRunCount4, AnalyzedInstrCount4,
468                                  AnalyzedFunctionCount4));
469     MPM.addPass(createModuleToFunctionPassAdaptor(std::move(FPM)));
470   }
471 
472   // A fifth function pass manager which invalidates one function first but
473   // uses only cached results.
474   int FunctionPassRunCount5 = 0;
475   int AnalyzedInstrCount5 = 0;
476   int AnalyzedFunctionCount5 = 0;
477   {
478     FunctionPassManager FPM(/*DebugLogging*/ true);
479     FPM.addPass(TestInvalidationFunctionPass("f"));
480     FPM.addPass(TestFunctionPass(FunctionPassRunCount5, AnalyzedInstrCount5,
481                                  AnalyzedFunctionCount5,
482                                  /*OnlyUseCachedResults=*/true));
483     MPM.addPass(createModuleToFunctionPassAdaptor(std::move(FPM)));
484   }
485 
486   MPM.run(*M, MAM);
487 
488   // Validate module pass counters.
489   EXPECT_EQ(1, ModulePassRunCount);
490 
491   // Validate all function pass counter sets are the same.
492   EXPECT_EQ(3, FunctionPassRunCount1);
493   EXPECT_EQ(5, AnalyzedInstrCount1);
494   EXPECT_EQ(0, AnalyzedFunctionCount1);
495   EXPECT_EQ(3, FunctionPassRunCount2);
496   EXPECT_EQ(5, AnalyzedInstrCount2);
497   EXPECT_EQ(0, AnalyzedFunctionCount2);
498   EXPECT_EQ(3, FunctionPassRunCount3);
499   EXPECT_EQ(5, AnalyzedInstrCount3);
500   EXPECT_EQ(0, AnalyzedFunctionCount3);
501   EXPECT_EQ(3, FunctionPassRunCount4);
502   EXPECT_EQ(5, AnalyzedInstrCount4);
503   EXPECT_EQ(9, AnalyzedFunctionCount4);
504   EXPECT_EQ(3, FunctionPassRunCount5);
505   EXPECT_EQ(2, AnalyzedInstrCount5); // Only 'g' and 'h' were cached.
506   EXPECT_EQ(9, AnalyzedFunctionCount5);
507 
508   // Validate the analysis counters:
509   //   first run over 3 functions, then module pass invalidates
510   //   second run over 3 functions, nothing invalidates
511   //   third run over 0 functions, but 1 function invalidated
512   //   fourth run over 1 function
513   //   fifth run invalidates 1 function first, but runs over 0 functions
514   EXPECT_EQ(7, FunctionAnalysisRuns);
515 
516   EXPECT_EQ(1, ModuleAnalysisRuns);
517 }
518 
519 // A customized pass manager that passes extra arguments through the
520 // infrastructure.
521 typedef AnalysisManager<Function, int> CustomizedAnalysisManager;
522 typedef PassManager<Function, CustomizedAnalysisManager, int, int &>
523     CustomizedPassManager;
524 
525 class CustomizedAnalysis : public AnalysisInfoMixin<CustomizedAnalysis> {
526 public:
527   struct Result {
Result__anon2a8083420111::CustomizedAnalysis::Result528     Result(int I) : I(I) {}
529     int I;
530   };
531 
run(Function & F,CustomizedAnalysisManager & AM,int I)532   Result run(Function &F, CustomizedAnalysisManager &AM, int I) {
533     return Result(I);
534   }
535 
536 private:
537   friend AnalysisInfoMixin<CustomizedAnalysis>;
538   static AnalysisKey Key;
539 };
540 
541 AnalysisKey CustomizedAnalysis::Key;
542 
543 struct CustomizedPass : PassInfoMixin<CustomizedPass> {
544   std::function<void(CustomizedAnalysis::Result &, int &)> Callback;
545 
546   template <typename CallbackT>
CustomizedPass__anon2a8083420111::CustomizedPass547   CustomizedPass(CallbackT Callback) : Callback(Callback) {}
548 
run__anon2a8083420111::CustomizedPass549   PreservedAnalyses run(Function &F, CustomizedAnalysisManager &AM, int I,
550                         int &O) {
551     Callback(AM.getResult<CustomizedAnalysis>(F, I), O);
552     return PreservedAnalyses::none();
553   }
554 };
555 
TEST_F(PassManagerTest,CustomizedPassManagerArgs)556 TEST_F(PassManagerTest, CustomizedPassManagerArgs) {
557   CustomizedAnalysisManager AM;
558   AM.registerPass([&] { return CustomizedAnalysis(); });
559 
560   CustomizedPassManager PM;
561 
562   // Add an instance of the customized pass that just accumulates the input
563   // after it is round-tripped through the analysis.
564   int Result = 0;
565   PM.addPass(
566       CustomizedPass([](CustomizedAnalysis::Result &R, int &O) { O += R.I; }));
567 
568   // Run this over every function with the input of 42.
569   for (Function &F : *M)
570     PM.run(F, AM, 42, Result);
571 
572   // And ensure that we accumulated the correct result.
573   EXPECT_EQ(42 * (int)M->size(), Result);
574 }
575 
576 /// A test analysis pass which caches in its result another analysis pass and
577 /// uses it to serve queries. This requires the result to invalidate itself
578 /// when its dependency is invalidated.
579 struct TestIndirectFunctionAnalysis
580     : public AnalysisInfoMixin<TestIndirectFunctionAnalysis> {
581   struct Result {
Result__anon2a8083420111::TestIndirectFunctionAnalysis::Result582     Result(TestFunctionAnalysis::Result &FDep, TestModuleAnalysis::Result &MDep)
583         : FDep(FDep), MDep(MDep) {}
584     TestFunctionAnalysis::Result &FDep;
585     TestModuleAnalysis::Result &MDep;
586 
invalidate__anon2a8083420111::TestIndirectFunctionAnalysis::Result587     bool invalidate(Function &F, const PreservedAnalyses &PA,
588                     FunctionAnalysisManager::Invalidator &Inv) {
589       auto PAC = PA.getChecker<TestIndirectFunctionAnalysis>();
590       return !(PAC.preserved() ||
591                PAC.preservedSet<AllAnalysesOn<Function>>()) ||
592              Inv.invalidate<TestFunctionAnalysis>(F, PA);
593     }
594   };
595 
TestIndirectFunctionAnalysis__anon2a8083420111::TestIndirectFunctionAnalysis596   TestIndirectFunctionAnalysis(int &Runs) : Runs(Runs) {}
597 
598   /// Run the analysis pass over the function and return a result.
run__anon2a8083420111::TestIndirectFunctionAnalysis599   Result run(Function &F, FunctionAnalysisManager &AM) {
600     ++Runs;
601     auto &FDep = AM.getResult<TestFunctionAnalysis>(F);
602     auto &Proxy = AM.getResult<ModuleAnalysisManagerFunctionProxy>(F);
603     const ModuleAnalysisManager &MAM = Proxy.getManager();
604     // For the test, we insist that the module analysis starts off in the
605     // cache.
606     auto &MDep = *MAM.getCachedResult<TestModuleAnalysis>(*F.getParent());
607     // And register the dependency as module analysis dependencies have to be
608     // pre-registered on the proxy.
609     Proxy.registerOuterAnalysisInvalidation<TestModuleAnalysis,
610                                             TestIndirectFunctionAnalysis>();
611     return Result(FDep, MDep);
612   }
613 
614 private:
615   friend AnalysisInfoMixin<TestIndirectFunctionAnalysis>;
616   static AnalysisKey Key;
617 
618   int &Runs;
619 };
620 
621 AnalysisKey TestIndirectFunctionAnalysis::Key;
622 
623 /// A test analysis pass which chaches in its result the result from the above
624 /// indirect analysis pass.
625 ///
626 /// This allows us to ensure that whenever an analysis pass is invalidated due
627 /// to dependencies (especially dependencies across IR units that trigger
628 /// asynchronous invalidation) we correctly detect that this may in turn cause
629 /// other analysis to be invalidated.
630 struct TestDoublyIndirectFunctionAnalysis
631     : public AnalysisInfoMixin<TestDoublyIndirectFunctionAnalysis> {
632   struct Result {
Result__anon2a8083420111::TestDoublyIndirectFunctionAnalysis::Result633     Result(TestIndirectFunctionAnalysis::Result &IDep) : IDep(IDep) {}
634     TestIndirectFunctionAnalysis::Result &IDep;
635 
invalidate__anon2a8083420111::TestDoublyIndirectFunctionAnalysis::Result636     bool invalidate(Function &F, const PreservedAnalyses &PA,
637                     FunctionAnalysisManager::Invalidator &Inv) {
638       auto PAC = PA.getChecker<TestDoublyIndirectFunctionAnalysis>();
639       return !(PAC.preserved() ||
640                PAC.preservedSet<AllAnalysesOn<Function>>()) ||
641              Inv.invalidate<TestIndirectFunctionAnalysis>(F, PA);
642     }
643   };
644 
TestDoublyIndirectFunctionAnalysis__anon2a8083420111::TestDoublyIndirectFunctionAnalysis645   TestDoublyIndirectFunctionAnalysis(int &Runs) : Runs(Runs) {}
646 
647   /// Run the analysis pass over the function and return a result.
run__anon2a8083420111::TestDoublyIndirectFunctionAnalysis648   Result run(Function &F, FunctionAnalysisManager &AM) {
649     ++Runs;
650     auto &IDep = AM.getResult<TestIndirectFunctionAnalysis>(F);
651     return Result(IDep);
652   }
653 
654 private:
655   friend AnalysisInfoMixin<TestDoublyIndirectFunctionAnalysis>;
656   static AnalysisKey Key;
657 
658   int &Runs;
659 };
660 
661 AnalysisKey TestDoublyIndirectFunctionAnalysis::Key;
662 
663 struct LambdaPass : public PassInfoMixin<LambdaPass> {
664   using FuncT = std::function<PreservedAnalyses(Function &, FunctionAnalysisManager &)>;
665 
LambdaPass__anon2a8083420111::LambdaPass666   LambdaPass(FuncT Func) : Func(std::move(Func)) {}
667 
run__anon2a8083420111::LambdaPass668   PreservedAnalyses run(Function &F, FunctionAnalysisManager &AM) {
669     return Func(F, AM);
670   }
671 
672   FuncT Func;
673 };
674 
TEST_F(PassManagerTest,IndirectAnalysisInvalidation)675 TEST_F(PassManagerTest, IndirectAnalysisInvalidation) {
676   FunctionAnalysisManager FAM(/*DebugLogging*/ true);
677   int FunctionAnalysisRuns = 0, ModuleAnalysisRuns = 0,
678       IndirectAnalysisRuns = 0, DoublyIndirectAnalysisRuns = 0;
679   FAM.registerPass([&] { return TestFunctionAnalysis(FunctionAnalysisRuns); });
680   FAM.registerPass(
681       [&] { return TestIndirectFunctionAnalysis(IndirectAnalysisRuns); });
682   FAM.registerPass([&] {
683     return TestDoublyIndirectFunctionAnalysis(DoublyIndirectAnalysisRuns);
684   });
685 
686   ModuleAnalysisManager MAM(/*DebugLogging*/ true);
687   MAM.registerPass([&] { return TestModuleAnalysis(ModuleAnalysisRuns); });
688   MAM.registerPass([&] { return FunctionAnalysisManagerModuleProxy(FAM); });
689   FAM.registerPass([&] { return ModuleAnalysisManagerFunctionProxy(MAM); });
690 
691   int InstrCount = 0, FunctionCount = 0;
692   ModulePassManager MPM(/*DebugLogging*/ true);
693   FunctionPassManager FPM(/*DebugLogging*/ true);
694   // First just use the analysis to get the instruction count, and preserve
695   // everything.
696   FPM.addPass(LambdaPass([&](Function &F, FunctionAnalysisManager &AM) {
697     auto &DoublyIndirectResult =
698         AM.getResult<TestDoublyIndirectFunctionAnalysis>(F);
699     auto &IndirectResult = DoublyIndirectResult.IDep;
700     InstrCount += IndirectResult.FDep.InstructionCount;
701     FunctionCount += IndirectResult.MDep.FunctionCount;
702     return PreservedAnalyses::all();
703   }));
704   // Next, invalidate
705   //   - both analyses for "f",
706   //   - just the underlying (indirect) analysis for "g", and
707   //   - just the direct analysis for "h".
708   FPM.addPass(LambdaPass([&](Function &F, FunctionAnalysisManager &AM) {
709     auto &DoublyIndirectResult =
710         AM.getResult<TestDoublyIndirectFunctionAnalysis>(F);
711     auto &IndirectResult = DoublyIndirectResult.IDep;
712     InstrCount += IndirectResult.FDep.InstructionCount;
713     FunctionCount += IndirectResult.MDep.FunctionCount;
714     auto PA = PreservedAnalyses::none();
715     if (F.getName() == "g")
716       PA.preserve<TestFunctionAnalysis>();
717     else if (F.getName() == "h")
718       PA.preserve<TestIndirectFunctionAnalysis>();
719     return PA;
720   }));
721   // Finally, use the analysis again on each function, forcing re-computation
722   // for all of them.
723   FPM.addPass(LambdaPass([&](Function &F, FunctionAnalysisManager &AM) {
724     auto &DoublyIndirectResult =
725         AM.getResult<TestDoublyIndirectFunctionAnalysis>(F);
726     auto &IndirectResult = DoublyIndirectResult.IDep;
727     InstrCount += IndirectResult.FDep.InstructionCount;
728     FunctionCount += IndirectResult.MDep.FunctionCount;
729     return PreservedAnalyses::all();
730   }));
731 
732   // Create a second function pass manager. This will cause the module-level
733   // invalidation to occur, which will force yet another invalidation of the
734   // indirect function-level analysis as the module analysis it depends on gets
735   // invalidated.
736   FunctionPassManager FPM2(/*DebugLogging*/ true);
737   FPM2.addPass(LambdaPass([&](Function &F, FunctionAnalysisManager &AM) {
738     auto &DoublyIndirectResult =
739         AM.getResult<TestDoublyIndirectFunctionAnalysis>(F);
740     auto &IndirectResult = DoublyIndirectResult.IDep;
741     InstrCount += IndirectResult.FDep.InstructionCount;
742     FunctionCount += IndirectResult.MDep.FunctionCount;
743     return PreservedAnalyses::all();
744   }));
745 
746   // Add a requires pass to populate the module analysis and then our function
747   // pass pipeline.
748   MPM.addPass(RequireAnalysisPass<TestModuleAnalysis, Module>());
749   MPM.addPass(createModuleToFunctionPassAdaptor(std::move(FPM)));
750   // Now require the module analysis again (it will have been invalidated once)
751   // and then use it again from a function pass manager.
752   MPM.addPass(RequireAnalysisPass<TestModuleAnalysis, Module>());
753   MPM.addPass(createModuleToFunctionPassAdaptor(std::move(FPM2)));
754   MPM.run(*M, MAM);
755 
756   // There are generally two possible runs for each of the three functions. But
757   // for one function, we only invalidate the indirect analysis so the base one
758   // only gets run five times.
759   EXPECT_EQ(5, FunctionAnalysisRuns);
760   // The module analysis pass should be run twice here.
761   EXPECT_EQ(2, ModuleAnalysisRuns);
762   // The indirect analysis is invalidated for each function (either directly or
763   // indirectly) and run twice for each.
764   EXPECT_EQ(9, IndirectAnalysisRuns);
765   EXPECT_EQ(9, DoublyIndirectAnalysisRuns);
766 
767   // There are five instructions in the module and we add the count four
768   // times.
769   EXPECT_EQ(5 * 4, InstrCount);
770 
771   // There are three functions and we count them four times for each of the
772   // three functions.
773   EXPECT_EQ(3 * 4 * 3, FunctionCount);
774 }
775 }
776