172 lines
		
	
	
		
			5.0 KiB
		
	
	
	
		
			C++
		
	
	
	
			
		
		
	
	
			172 lines
		
	
	
		
			5.0 KiB
		
	
	
	
		
			C++
		
	
	
	
| //===- unittests/IR/TimePassesTest.cpp - TimePassesHandler tests ----------===//
 | |
| //
 | |
| // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
 | |
| // See https://llvm.org/LICENSE.txt for license information.
 | |
| // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
 | |
| //
 | |
| //===----------------------------------------------------------------------===//
 | |
| 
 | |
| #include "llvm/IR/LegacyPassManager.h"
 | |
| #include "llvm/Pass.h"
 | |
| #include "llvm/PassRegistry.h"
 | |
| #include <gtest/gtest.h>
 | |
| #include <llvm/ADT/SmallString.h>
 | |
| #include <llvm/IR/LLVMContext.h>
 | |
| #include <llvm/IR/Module.h>
 | |
| #include <llvm/IR/PassInstrumentation.h>
 | |
| #include <llvm/IR/PassManager.h>
 | |
| #include <llvm/IR/PassTimingInfo.h>
 | |
| #include <llvm/Support/raw_ostream.h>
 | |
| 
 | |
| using namespace llvm;
 | |
| 
 | |
| //===----------------------------------------------------------------------===//
 | |
| // Define dummy passes for legacy pass manager run.
 | |
| 
 | |
| namespace llvm {
 | |
| 
 | |
| void initializePass1Pass(PassRegistry &);
 | |
| void initializePass2Pass(PassRegistry &);
 | |
| 
 | |
| namespace {
 | |
| struct Pass1 : public ModulePass {
 | |
|   static char ID;
 | |
| 
 | |
| public:
 | |
|   Pass1() : ModulePass(ID) {}
 | |
|   bool runOnModule(Module &M) override { return false; }
 | |
|   void getAnalysisUsage(AnalysisUsage &AU) const override {
 | |
|     AU.setPreservesAll();
 | |
|   }
 | |
|   StringRef getPassName() const override { return "Pass1"; }
 | |
| };
 | |
| char Pass1::ID;
 | |
| 
 | |
| struct Pass2 : public ModulePass {
 | |
|   static char ID;
 | |
| 
 | |
| public:
 | |
|   Pass2() : ModulePass(ID) {}
 | |
|   bool runOnModule(Module &M) override { return false; }
 | |
|   void getAnalysisUsage(AnalysisUsage &AU) const override {
 | |
|     AU.setPreservesAll();
 | |
|   }
 | |
|   StringRef getPassName() const override { return "Pass2"; }
 | |
| };
 | |
| char Pass2::ID;
 | |
| } // namespace
 | |
| } // namespace llvm
 | |
| 
 | |
| INITIALIZE_PASS(Pass1, "Pass1", "Pass1", false, false)
 | |
| INITIALIZE_PASS(Pass2, "Pass2", "Pass2", false, false)
 | |
| 
 | |
| namespace {
 | |
| 
 | |
| TEST(TimePassesTest, LegacyCustomOut) {
 | |
|   PassInstrumentationCallbacks PIC;
 | |
|   PassInstrumentation PI(&PIC);
 | |
| 
 | |
|   LLVMContext Context;
 | |
|   Module M("TestModule", Context);
 | |
| 
 | |
|   SmallString<0> TimePassesStr;
 | |
|   raw_svector_ostream ReportStream(TimePassesStr);
 | |
| 
 | |
|   // Setup pass manager
 | |
|   legacy::PassManager PM1;
 | |
|   PM1.add(new llvm::Pass1());
 | |
|   PM1.add(new llvm::Pass2());
 | |
| 
 | |
|   // Enable time-passes and run passes.
 | |
|   TimePassesIsEnabled = true;
 | |
|   PM1.run(M);
 | |
| 
 | |
|   // Generating report.
 | |
|   reportAndResetTimings(&ReportStream);
 | |
| 
 | |
|   // There should be Pass1 and Pass2 in the report
 | |
|   EXPECT_FALSE(TimePassesStr.empty());
 | |
|   EXPECT_TRUE(TimePassesStr.str().contains("report"));
 | |
|   EXPECT_TRUE(TimePassesStr.str().contains("Pass1"));
 | |
|   EXPECT_TRUE(TimePassesStr.str().contains("Pass2"));
 | |
| 
 | |
|   // Clear and generate report again.
 | |
|   TimePassesStr.clear();
 | |
|   reportAndResetTimings(&ReportStream);
 | |
| 
 | |
|   // Since we did not run any passes since last print, report should be empty.
 | |
|   EXPECT_TRUE(TimePassesStr.empty());
 | |
| 
 | |
|   // Now run just a single pass to populate timers again.
 | |
|   legacy::PassManager PM2;
 | |
|   PM2.add(new llvm::Pass2());
 | |
|   PM2.run(M);
 | |
| 
 | |
|   // Generate report again.
 | |
|   reportAndResetTimings(&ReportStream);
 | |
| 
 | |
|   // There should be Pass2 in this report and no Pass1.
 | |
|   EXPECT_FALSE(TimePassesStr.str().empty());
 | |
|   EXPECT_TRUE(TimePassesStr.str().contains("report"));
 | |
|   EXPECT_FALSE(TimePassesStr.str().contains("Pass1"));
 | |
|   EXPECT_TRUE(TimePassesStr.str().contains("Pass2"));
 | |
| }
 | |
| 
 | |
| class MyPass1 : public PassInfoMixin<MyPass1> {};
 | |
| class MyPass2 : public PassInfoMixin<MyPass2> {};
 | |
| 
 | |
| TEST(TimePassesTest, CustomOut) {
 | |
|   PassInstrumentationCallbacks PIC;
 | |
|   PassInstrumentation PI(&PIC);
 | |
| 
 | |
|   LLVMContext Context;
 | |
|   Module M("TestModule", Context);
 | |
|   MyPass1 Pass1;
 | |
|   MyPass2 Pass2;
 | |
| 
 | |
|   SmallString<0> TimePassesStr;
 | |
|   raw_svector_ostream ReportStream(TimePassesStr);
 | |
| 
 | |
|   // Setup time-passes handler and redirect output to the stream.
 | |
|   std::unique_ptr<TimePassesHandler> TimePasses =
 | |
|       std::make_unique<TimePassesHandler>(true);
 | |
|   TimePasses->setOutStream(ReportStream);
 | |
|   TimePasses->registerCallbacks(PIC);
 | |
| 
 | |
|   // Pretending that passes are running to trigger the timers.
 | |
|   PI.runBeforePass(Pass1, M);
 | |
|   PI.runBeforePass(Pass2, M);
 | |
|   PI.runAfterPass(Pass2, M, PreservedAnalyses::all());
 | |
|   PI.runAfterPass(Pass1, M, PreservedAnalyses::all());
 | |
| 
 | |
|   // Generating report.
 | |
|   TimePasses->print();
 | |
| 
 | |
|   // There should be Pass1 and Pass2 in the report
 | |
|   EXPECT_FALSE(TimePassesStr.empty());
 | |
|   EXPECT_TRUE(TimePassesStr.str().contains("report"));
 | |
|   EXPECT_TRUE(TimePassesStr.str().contains("Pass1"));
 | |
|   EXPECT_TRUE(TimePassesStr.str().contains("Pass2"));
 | |
| 
 | |
|   // Clear and generate report again.
 | |
|   TimePassesStr.clear();
 | |
|   TimePasses->print();
 | |
|   // Since we did not run any passes since last print, report should be empty.
 | |
|   EXPECT_TRUE(TimePassesStr.empty());
 | |
| 
 | |
|   // Now trigger just a single pass to populate timers again.
 | |
|   PI.runBeforePass(Pass2, M);
 | |
|   PI.runAfterPass(Pass2, M, PreservedAnalyses::all());
 | |
| 
 | |
|   // Generate report by deleting the handler.
 | |
|   TimePasses.reset();
 | |
| 
 | |
|   // There should be Pass2 in this report and no Pass1.
 | |
|   EXPECT_FALSE(TimePassesStr.str().empty());
 | |
|   EXPECT_TRUE(TimePassesStr.str().contains("report"));
 | |
|   EXPECT_FALSE(TimePassesStr.str().contains("Pass1"));
 | |
|   EXPECT_TRUE(TimePassesStr.str().contains("Pass2"));
 | |
| }
 | |
| 
 | |
| } // end anonymous namespace
 |