//===- CGSCCPassManagerTest.cpp -------------------------------------------===// // // The LLVM Compiler Infrastructure // // This file is distributed under the University of Illinois Open Source // License. See LICENSE.TXT for details. // //===----------------------------------------------------------------------===// #include "llvm/Analysis/CGSCCPassManager.h" #include "llvm/Analysis/LazyCallGraph.h" #include "llvm/AsmParser/Parser.h" #include "llvm/IR/Function.h" #include "llvm/IR/InstIterator.h" #include "llvm/IR/LLVMContext.h" #include "llvm/IR/Module.h" #include "llvm/IR/PassManager.h" #include "llvm/Support/SourceMgr.h" #include "gtest/gtest.h" using namespace llvm; namespace { class TestModuleAnalysis : public AnalysisInfoMixin { public: struct Result { Result(int Count) : FunctionCount(Count) {} int FunctionCount; }; TestModuleAnalysis(int &Runs) : Runs(Runs) {} Result run(Module &M, ModuleAnalysisManager &AM) { ++Runs; return Result(M.size()); } private: friend AnalysisInfoMixin; static AnalysisKey Key; int &Runs; }; AnalysisKey TestModuleAnalysis::Key; class TestSCCAnalysis : public AnalysisInfoMixin { public: struct Result { Result(int Count) : FunctionCount(Count) {} int FunctionCount; }; TestSCCAnalysis(int &Runs) : Runs(Runs) {} Result run(LazyCallGraph::SCC &C, CGSCCAnalysisManager &AM, LazyCallGraph &) { ++Runs; return Result(C.size()); } private: friend AnalysisInfoMixin; static AnalysisKey Key; int &Runs; }; AnalysisKey TestSCCAnalysis::Key; class TestFunctionAnalysis : public AnalysisInfoMixin { public: struct Result { Result(int Count) : InstructionCount(Count) {} int InstructionCount; }; TestFunctionAnalysis(int &Runs) : Runs(Runs) {} Result run(Function &F, FunctionAnalysisManager &AM) { ++Runs; int Count = 0; for (Instruction &I : instructions(F)) { (void)I; ++Count; } return Result(Count); } private: friend AnalysisInfoMixin; static AnalysisKey Key; int &Runs; }; AnalysisKey TestFunctionAnalysis::Key; class TestImmutableFunctionAnalysis : public AnalysisInfoMixin { public: struct Result { bool invalidate(Function &, const PreservedAnalyses &) { return false; } }; TestImmutableFunctionAnalysis(int &Runs) : Runs(Runs) {} Result run(Function &F, FunctionAnalysisManager &AM) { ++Runs; return Result(); } private: friend AnalysisInfoMixin; static AnalysisKey Key; int &Runs; }; AnalysisKey TestImmutableFunctionAnalysis::Key; struct LambdaSCCPass : public PassInfoMixin { template LambdaSCCPass(T &&Arg) : Func(std::forward(Arg)) {} // We have to explicitly define all the special member functions because MSVC // refuses to generate them. LambdaSCCPass(LambdaSCCPass &&Arg) : Func(std::move(Arg.Func)) {} LambdaSCCPass &operator=(LambdaSCCPass &&RHS) { Func = std::move(RHS.Func); return *this; } PreservedAnalyses run(LazyCallGraph::SCC &C, CGSCCAnalysisManager &AM, LazyCallGraph &CG, CGSCCUpdateResult &UR) { return Func(C, AM, CG, UR); } std::function Func; }; struct LambdaFunctionPass : public PassInfoMixin { template LambdaFunctionPass(T &&Arg) : Func(std::forward(Arg)) {} // We have to explicitly define all the special member functions because MSVC // refuses to generate them. LambdaFunctionPass(LambdaFunctionPass &&Arg) : Func(std::move(Arg.Func)) {} LambdaFunctionPass &operator=(LambdaFunctionPass &&RHS) { Func = std::move(RHS.Func); return *this; } PreservedAnalyses run(Function &F, FunctionAnalysisManager &AM) { return Func(F, AM); } std::function Func; }; std::unique_ptr parseIR(const char *IR) { // We just use a static context here. This is never called from multiple // threads so it is harmless no matter how it is implemented. We just need // the context to outlive the module which it does. static LLVMContext C; SMDiagnostic Err; return parseAssemblyString(IR, Err, C); } class CGSCCPassManagerTest : public ::testing::Test { protected: LLVMContext Context; FunctionAnalysisManager FAM; CGSCCAnalysisManager CGAM; ModuleAnalysisManager MAM; std::unique_ptr M; public: CGSCCPassManagerTest() : FAM(/*DebugLogging*/ true), CGAM(/*DebugLogging*/ true), MAM(/*DebugLogging*/ true), M(parseIR("define void @f() {\n" "entry:\n" " call void @g()\n" " call void @h1()\n" " ret void\n" "}\n" "define void @g() {\n" "entry:\n" " call void @g()\n" " call void @x()\n" " ret void\n" "}\n" "define void @h1() {\n" "entry:\n" " call void @h2()\n" " ret void\n" "}\n" "define void @h2() {\n" "entry:\n" " call void @h3()\n" " call void @x()\n" " ret void\n" "}\n" "define void @h3() {\n" "entry:\n" " call void @h1()\n" " ret void\n" "}\n" "define void @x() {\n" "entry:\n" " ret void\n" "}\n")) { MAM.registerPass([&] { return LazyCallGraphAnalysis(); }); MAM.registerPass([&] { return FunctionAnalysisManagerModuleProxy(FAM); }); MAM.registerPass([&] { return CGSCCAnalysisManagerModuleProxy(CGAM); }); CGAM.registerPass([&] { return FunctionAnalysisManagerCGSCCProxy(FAM); }); CGAM.registerPass([&] { return ModuleAnalysisManagerCGSCCProxy(MAM); }); FAM.registerPass([&] { return CGSCCAnalysisManagerFunctionProxy(CGAM); }); FAM.registerPass([&] { return ModuleAnalysisManagerFunctionProxy(MAM); }); } }; TEST_F(CGSCCPassManagerTest, Basic) { int FunctionAnalysisRuns = 0; FAM.registerPass([&] { return TestFunctionAnalysis(FunctionAnalysisRuns); }); int ImmutableFunctionAnalysisRuns = 0; FAM.registerPass([&] { return TestImmutableFunctionAnalysis(ImmutableFunctionAnalysisRuns); }); int SCCAnalysisRuns = 0; CGAM.registerPass([&] { return TestSCCAnalysis(SCCAnalysisRuns); }); int ModuleAnalysisRuns = 0; MAM.registerPass([&] { return TestModuleAnalysis(ModuleAnalysisRuns); }); ModulePassManager MPM(/*DebugLogging*/ true); MPM.addPass(RequireAnalysisPass()); CGSCCPassManager CGPM1(/*DebugLogging*/ true); int SCCPassRunCount1 = 0; int AnalyzedInstrCount1 = 0; int AnalyzedSCCFunctionCount1 = 0; int AnalyzedModuleFunctionCount1 = 0; CGPM1.addPass( LambdaSCCPass([&](LazyCallGraph::SCC &C, CGSCCAnalysisManager &AM, LazyCallGraph &CG, CGSCCUpdateResult &UR) { ++SCCPassRunCount1; const ModuleAnalysisManager &MAM = AM.getResult(C, CG).getManager(); FunctionAnalysisManager &FAM = AM.getResult(C, CG).getManager(); if (TestModuleAnalysis::Result *TMA = MAM.getCachedResult( *C.begin()->getFunction().getParent())) AnalyzedModuleFunctionCount1 += TMA->FunctionCount; TestSCCAnalysis::Result &AR = AM.getResult(C, CG); AnalyzedSCCFunctionCount1 += AR.FunctionCount; for (LazyCallGraph::Node &N : C) { TestFunctionAnalysis::Result &FAR = FAM.getResult(N.getFunction()); AnalyzedInstrCount1 += FAR.InstructionCount; // Just ensure we get the immutable results. (void)FAM.getResult(N.getFunction()); } return PreservedAnalyses::all(); })); FunctionPassManager FPM1(/*DebugLogging*/ true); int FunctionPassRunCount1 = 0; FPM1.addPass(LambdaFunctionPass([&](Function &, FunctionAnalysisManager &) { ++FunctionPassRunCount1; return PreservedAnalyses::all(); })); CGPM1.addPass(createCGSCCToFunctionPassAdaptor(std::move(FPM1))); MPM.addPass(createModuleToPostOrderCGSCCPassAdaptor(std::move(CGPM1))); MPM.run(*M, MAM); EXPECT_EQ(1, ModuleAnalysisRuns); EXPECT_EQ(4, SCCAnalysisRuns); EXPECT_EQ(6, FunctionAnalysisRuns); EXPECT_EQ(6, ImmutableFunctionAnalysisRuns); EXPECT_EQ(4, SCCPassRunCount1); EXPECT_EQ(14, AnalyzedInstrCount1); EXPECT_EQ(6, AnalyzedSCCFunctionCount1); EXPECT_EQ(4 * 6, AnalyzedModuleFunctionCount1); } // Test that an SCC pass which fails to preserve a module analysis does in fact // invalidate that module analysis. TEST_F(CGSCCPassManagerTest, TestSCCPassInvalidatesModuleAnalysis) { int ModuleAnalysisRuns = 0; MAM.registerPass([&] { return TestModuleAnalysis(ModuleAnalysisRuns); }); ModulePassManager MPM(/*DebugLogging*/ true); MPM.addPass(RequireAnalysisPass()); // The first CGSCC run we preserve everything and make sure that works and // the module analysis is available in the second CGSCC run from the one // required module pass above. CGSCCPassManager CGPM1(/*DebugLogging*/ true); int CountFoundModuleAnalysis1 = 0; CGPM1.addPass( LambdaSCCPass([&](LazyCallGraph::SCC &C, CGSCCAnalysisManager &AM, LazyCallGraph &CG, CGSCCUpdateResult &UR) { const auto &MAM = AM.getResult(C, CG).getManager(); auto *TMA = MAM.getCachedResult( *C.begin()->getFunction().getParent()); if (TMA) ++CountFoundModuleAnalysis1; return PreservedAnalyses::all(); })); MPM.addPass(createModuleToPostOrderCGSCCPassAdaptor(std::move(CGPM1))); // The second CGSCC run checks that the module analysis got preserved the // previous time and in one SCC fails to preserve it. CGSCCPassManager CGPM2(/*DebugLogging*/ true); int CountFoundModuleAnalysis2 = 0; CGPM2.addPass( LambdaSCCPass([&](LazyCallGraph::SCC &C, CGSCCAnalysisManager &AM, LazyCallGraph &CG, CGSCCUpdateResult &UR) { const auto &MAM = AM.getResult(C, CG).getManager(); auto *TMA = MAM.getCachedResult( *C.begin()->getFunction().getParent()); if (TMA) ++CountFoundModuleAnalysis2; // Only fail to preserve analyses on one SCC and make sure that gets // propagated. return C.getName() == "(g)" ? PreservedAnalyses::none() : PreservedAnalyses::all(); })); MPM.addPass(createModuleToPostOrderCGSCCPassAdaptor(std::move(CGPM2))); // The third CGSCC run should fail to find a cached module analysis as it // should have been invalidated by the above CGSCC run. CGSCCPassManager CGPM3(/*DebugLogging*/ true); int CountFoundModuleAnalysis3 = 0; CGPM3.addPass( LambdaSCCPass([&](LazyCallGraph::SCC &C, CGSCCAnalysisManager &AM, LazyCallGraph &CG, CGSCCUpdateResult &UR) { const auto &MAM = AM.getResult(C, CG).getManager(); auto *TMA = MAM.getCachedResult( *C.begin()->getFunction().getParent()); if (TMA) ++CountFoundModuleAnalysis3; return PreservedAnalyses::none(); })); MPM.addPass(createModuleToPostOrderCGSCCPassAdaptor(std::move(CGPM3))); MPM.run(*M, MAM); EXPECT_EQ(1, ModuleAnalysisRuns); EXPECT_EQ(4, CountFoundModuleAnalysis1); EXPECT_EQ(4, CountFoundModuleAnalysis2); EXPECT_EQ(0, CountFoundModuleAnalysis3); } // Similar to the above, but test that this works for function passes embedded // *within* a CGSCC layer. TEST_F(CGSCCPassManagerTest, TestFunctionPassInsideCGSCCInvalidatesModuleAnalysis) { int ModuleAnalysisRuns = 0; MAM.registerPass([&] { return TestModuleAnalysis(ModuleAnalysisRuns); }); ModulePassManager MPM(/*DebugLogging*/ true); MPM.addPass(RequireAnalysisPass()); // The first run we preserve everything and make sure that works and the // module analysis is available in the second run from the one required // module pass above. FunctionPassManager FPM1(/*DebugLogging*/ true); // Start true and mark false if we ever failed to find a module analysis // because we expect this to succeed for each SCC. bool FoundModuleAnalysis1 = true; FPM1.addPass( LambdaFunctionPass([&](Function &F, FunctionAnalysisManager &AM) { const auto &MAM = AM.getResult(F).getManager(); auto *TMA = MAM.getCachedResult(*F.getParent()); if (!TMA) FoundModuleAnalysis1 = false; return PreservedAnalyses::all(); })); CGSCCPassManager CGPM1(/*DebugLogging*/ true); CGPM1.addPass(createCGSCCToFunctionPassAdaptor(std::move(FPM1))); MPM.addPass(createModuleToPostOrderCGSCCPassAdaptor(std::move(CGPM1))); // The second run checks that the module analysis got preserved the previous // time and in one function fails to preserve it. FunctionPassManager FPM2(/*DebugLogging*/ true); // Again, start true and mark false if we ever failed to find a module analysis // because we expect this to succeed for each SCC. bool FoundModuleAnalysis2 = true; FPM2.addPass( LambdaFunctionPass([&](Function &F, FunctionAnalysisManager &AM) { const auto &MAM = AM.getResult(F).getManager(); auto *TMA = MAM.getCachedResult(*F.getParent()); if (!TMA) FoundModuleAnalysis2 = false; // Only fail to preserve analyses on one SCC and make sure that gets // propagated. return F.getName() == "h2" ? PreservedAnalyses::none() : PreservedAnalyses::all(); })); CGSCCPassManager CGPM2(/*DebugLogging*/ true); CGPM2.addPass(createCGSCCToFunctionPassAdaptor(std::move(FPM2))); MPM.addPass(createModuleToPostOrderCGSCCPassAdaptor(std::move(CGPM2))); // The third run should fail to find a cached module analysis as it should // have been invalidated by the above run. FunctionPassManager FPM3(/*DebugLogging*/ true); // Start false and mark true if we ever *succeeded* to find a module // analysis, as we expect this to fail for every function. bool FoundModuleAnalysis3 = false; FPM3.addPass( LambdaFunctionPass([&](Function &F, FunctionAnalysisManager &AM) { const auto &MAM = AM.getResult(F).getManager(); auto *TMA = MAM.getCachedResult(*F.getParent()); if (TMA) FoundModuleAnalysis3 = true; return PreservedAnalyses::none(); })); CGSCCPassManager CGPM3(/*DebugLogging*/ true); CGPM3.addPass(createCGSCCToFunctionPassAdaptor(std::move(FPM3))); MPM.addPass(createModuleToPostOrderCGSCCPassAdaptor(std::move(CGPM3))); MPM.run(*M, MAM); EXPECT_EQ(1, ModuleAnalysisRuns); EXPECT_TRUE(FoundModuleAnalysis1); EXPECT_TRUE(FoundModuleAnalysis2); EXPECT_FALSE(FoundModuleAnalysis3); } }