forked from OSchip/llvm-project
				
			
		
			
				
	
	
		
			389 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			C++
		
	
	
	
			
		
		
	
	
			389 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			C++
		
	
	
	
//===- CFGTest.cpp - CFG tests --------------------------------------------===//
 | 
						|
//
 | 
						|
//                     The LLVM Compiler Infrastructure
 | 
						|
//
 | 
						|
// This file is distributed under the University of Illinois Open Source
 | 
						|
// License. See LICENSE.TXT for details.
 | 
						|
//
 | 
						|
//===----------------------------------------------------------------------===//
 | 
						|
 | 
						|
#include "llvm/Analysis/CFG.h"
 | 
						|
#include "llvm/Analysis/LoopInfo.h"
 | 
						|
#include "llvm/AsmParser/Parser.h"
 | 
						|
#include "llvm/IR/Dominators.h"
 | 
						|
#include "llvm/IR/Function.h"
 | 
						|
#include "llvm/IR/InstIterator.h"
 | 
						|
#include "llvm/IR/LLVMContext.h"
 | 
						|
#include "llvm/IR/Module.h"
 | 
						|
#include "llvm/Pass.h"
 | 
						|
#include "llvm/IR/LegacyPassManager.h"
 | 
						|
#include "llvm/Support/ErrorHandling.h"
 | 
						|
#include "llvm/Support/SourceMgr.h"
 | 
						|
#include "gtest/gtest.h"
 | 
						|
 | 
						|
using namespace llvm;
 | 
						|
 | 
						|
namespace {
 | 
						|
 | 
						|
// This fixture assists in running the isPotentiallyReachable utility four ways
 | 
						|
// and ensuring it produces the correct answer each time.
 | 
						|
class IsPotentiallyReachableTest : public testing::Test {
 | 
						|
protected:
 | 
						|
  void ParseAssembly(const char *Assembly) {
 | 
						|
    SMDiagnostic Error;
 | 
						|
    M = parseAssemblyString(Assembly, Error, Context);
 | 
						|
 | 
						|
    std::string errMsg;
 | 
						|
    raw_string_ostream os(errMsg);
 | 
						|
    Error.print("", os);
 | 
						|
 | 
						|
    // A failure here means that the test itself is buggy.
 | 
						|
    if (!M)
 | 
						|
      report_fatal_error(os.str().c_str());
 | 
						|
 | 
						|
    Function *F = M->getFunction("test");
 | 
						|
    if (F == nullptr)
 | 
						|
      report_fatal_error("Test must have a function named @test");
 | 
						|
 | 
						|
    A = B = nullptr;
 | 
						|
    for (inst_iterator I = inst_begin(F), E = inst_end(F); I != E; ++I) {
 | 
						|
      if (I->hasName()) {
 | 
						|
        if (I->getName() == "A")
 | 
						|
          A = &*I;
 | 
						|
        else if (I->getName() == "B")
 | 
						|
          B = &*I;
 | 
						|
      }
 | 
						|
    }
 | 
						|
    if (A == nullptr)
 | 
						|
      report_fatal_error("@test must have an instruction %A");
 | 
						|
    if (B == nullptr)
 | 
						|
      report_fatal_error("@test must have an instruction %B");
 | 
						|
  }
 | 
						|
 | 
						|
  void ExpectPath(bool ExpectedResult) {
 | 
						|
    static char ID;
 | 
						|
    class IsPotentiallyReachableTestPass : public FunctionPass {
 | 
						|
     public:
 | 
						|
      IsPotentiallyReachableTestPass(bool ExpectedResult,
 | 
						|
                                     Instruction *A, Instruction *B)
 | 
						|
          : FunctionPass(ID), ExpectedResult(ExpectedResult), A(A), B(B) {}
 | 
						|
 | 
						|
      static int initialize() {
 | 
						|
        PassInfo *PI = new PassInfo("isPotentiallyReachable testing pass",
 | 
						|
                                    "", &ID, nullptr, true, true);
 | 
						|
        PassRegistry::getPassRegistry()->registerPass(*PI, false);
 | 
						|
        initializeLoopInfoWrapperPassPass(*PassRegistry::getPassRegistry());
 | 
						|
        initializeDominatorTreeWrapperPassPass(
 | 
						|
            *PassRegistry::getPassRegistry());
 | 
						|
        return 0;
 | 
						|
      }
 | 
						|
 | 
						|
      void getAnalysisUsage(AnalysisUsage &AU) const override {
 | 
						|
        AU.setPreservesAll();
 | 
						|
        AU.addRequired<LoopInfoWrapperPass>();
 | 
						|
        AU.addRequired<DominatorTreeWrapperPass>();
 | 
						|
      }
 | 
						|
 | 
						|
      bool runOnFunction(Function &F) override {
 | 
						|
        if (!F.hasName() || F.getName() != "test")
 | 
						|
          return false;
 | 
						|
 | 
						|
        LoopInfo *LI = &getAnalysis<LoopInfoWrapperPass>().getLoopInfo();
 | 
						|
        DominatorTree *DT =
 | 
						|
            &getAnalysis<DominatorTreeWrapperPass>().getDomTree();
 | 
						|
        EXPECT_EQ(isPotentiallyReachable(A, B, nullptr, nullptr),
 | 
						|
                  ExpectedResult);
 | 
						|
        EXPECT_EQ(isPotentiallyReachable(A, B, DT, nullptr), ExpectedResult);
 | 
						|
        EXPECT_EQ(isPotentiallyReachable(A, B, nullptr, LI), ExpectedResult);
 | 
						|
        EXPECT_EQ(isPotentiallyReachable(A, B, DT, LI), ExpectedResult);
 | 
						|
        return false;
 | 
						|
      }
 | 
						|
      bool ExpectedResult;
 | 
						|
      Instruction *A, *B;
 | 
						|
    };
 | 
						|
 | 
						|
    static int initialize = IsPotentiallyReachableTestPass::initialize();
 | 
						|
    (void)initialize;
 | 
						|
 | 
						|
    IsPotentiallyReachableTestPass *P =
 | 
						|
        new IsPotentiallyReachableTestPass(ExpectedResult, A, B);
 | 
						|
    legacy::PassManager PM;
 | 
						|
    PM.add(P);
 | 
						|
    PM.run(*M);
 | 
						|
  }
 | 
						|
 | 
						|
  LLVMContext Context;
 | 
						|
  std::unique_ptr<Module> M;
 | 
						|
  Instruction *A, *B;
 | 
						|
};
 | 
						|
 | 
						|
}
 | 
						|
 | 
						|
TEST_F(IsPotentiallyReachableTest, SameBlockNoPath) {
 | 
						|
  ParseAssembly(
 | 
						|
      "define void @test() {\n"
 | 
						|
      "entry:\n"
 | 
						|
      "  bitcast i8 undef to i8\n"
 | 
						|
      "  %B = bitcast i8 undef to i8\n"
 | 
						|
      "  bitcast i8 undef to i8\n"
 | 
						|
      "  bitcast i8 undef to i8\n"
 | 
						|
      "  %A = bitcast i8 undef to i8\n"
 | 
						|
      "  ret void\n"
 | 
						|
      "}\n");
 | 
						|
  ExpectPath(false);
 | 
						|
}
 | 
						|
 | 
						|
TEST_F(IsPotentiallyReachableTest, SameBlockPath) {
 | 
						|
  ParseAssembly(
 | 
						|
      "define void @test() {\n"
 | 
						|
      "entry:\n"
 | 
						|
      "  %A = bitcast i8 undef to i8\n"
 | 
						|
      "  bitcast i8 undef to i8\n"
 | 
						|
      "  bitcast i8 undef to i8\n"
 | 
						|
      "  %B = bitcast i8 undef to i8\n"
 | 
						|
      "  ret void\n"
 | 
						|
      "}\n");
 | 
						|
  ExpectPath(true);
 | 
						|
}
 | 
						|
 | 
						|
TEST_F(IsPotentiallyReachableTest, SameBlockNoLoop) {
 | 
						|
  ParseAssembly(
 | 
						|
      "define void @test() {\n"
 | 
						|
      "entry:\n"
 | 
						|
      "  br label %middle\n"
 | 
						|
      "middle:\n"
 | 
						|
      "  %B = bitcast i8 undef to i8\n"
 | 
						|
      "  bitcast i8 undef to i8\n"
 | 
						|
      "  bitcast i8 undef to i8\n"
 | 
						|
      "  %A = bitcast i8 undef to i8\n"
 | 
						|
      "  br label %nextblock\n"
 | 
						|
      "nextblock:\n"
 | 
						|
      "  ret void\n"
 | 
						|
      "}\n");
 | 
						|
  ExpectPath(false);
 | 
						|
}
 | 
						|
 | 
						|
TEST_F(IsPotentiallyReachableTest, StraightNoPath) {
 | 
						|
  ParseAssembly(
 | 
						|
      "define void @test() {\n"
 | 
						|
      "entry:\n"
 | 
						|
      "  %B = bitcast i8 undef to i8\n"
 | 
						|
      "  br label %exit\n"
 | 
						|
      "exit:\n"
 | 
						|
      "  %A = bitcast i8 undef to i8\n"
 | 
						|
      "  ret void\n"
 | 
						|
      "}");
 | 
						|
  ExpectPath(false);
 | 
						|
}
 | 
						|
 | 
						|
TEST_F(IsPotentiallyReachableTest, StraightPath) {
 | 
						|
  ParseAssembly(
 | 
						|
      "define void @test() {\n"
 | 
						|
      "entry:\n"
 | 
						|
      "  %A = bitcast i8 undef to i8\n"
 | 
						|
      "  br label %exit\n"
 | 
						|
      "exit:\n"
 | 
						|
      "  %B = bitcast i8 undef to i8\n"
 | 
						|
      "  ret void\n"
 | 
						|
      "}");
 | 
						|
  ExpectPath(true);
 | 
						|
}
 | 
						|
 | 
						|
TEST_F(IsPotentiallyReachableTest, DestUnreachable) {
 | 
						|
  ParseAssembly(
 | 
						|
      "define void @test() {\n"
 | 
						|
      "entry:\n"
 | 
						|
      "  br label %midblock\n"
 | 
						|
      "midblock:\n"
 | 
						|
      "  %A = bitcast i8 undef to i8\n"
 | 
						|
      "  ret void\n"
 | 
						|
      "unreachable:\n"
 | 
						|
      "  %B = bitcast i8 undef to i8\n"
 | 
						|
      "  br label %midblock\n"
 | 
						|
      "}");
 | 
						|
  ExpectPath(false);
 | 
						|
}
 | 
						|
 | 
						|
TEST_F(IsPotentiallyReachableTest, BranchToReturn) {
 | 
						|
  ParseAssembly(
 | 
						|
      "define void @test(i1 %x) {\n"
 | 
						|
      "entry:\n"
 | 
						|
      "  %A = bitcast i8 undef to i8\n"
 | 
						|
      "  br i1 %x, label %block1, label %block2\n"
 | 
						|
      "block1:\n"
 | 
						|
      "  ret void\n"
 | 
						|
      "block2:\n"
 | 
						|
      "  %B = bitcast i8 undef to i8\n"
 | 
						|
      "  ret void\n"
 | 
						|
      "}");
 | 
						|
  ExpectPath(true);
 | 
						|
}
 | 
						|
 | 
						|
TEST_F(IsPotentiallyReachableTest, SimpleLoop1) {
 | 
						|
  ParseAssembly(
 | 
						|
      "declare i1 @switch()\n"
 | 
						|
      "\n"
 | 
						|
      "define void @test() {\n"
 | 
						|
      "entry:\n"
 | 
						|
      "  br label %loop\n"
 | 
						|
      "loop:\n"
 | 
						|
      "  %B = bitcast i8 undef to i8\n"
 | 
						|
      "  %A = bitcast i8 undef to i8\n"
 | 
						|
      "  %x = call i1 @switch()\n"
 | 
						|
      "  br i1 %x, label %loop, label %exit\n"
 | 
						|
      "exit:\n"
 | 
						|
      "  ret void\n"
 | 
						|
      "}");
 | 
						|
  ExpectPath(true);
 | 
						|
}
 | 
						|
 | 
						|
TEST_F(IsPotentiallyReachableTest, SimpleLoop2) {
 | 
						|
  ParseAssembly(
 | 
						|
      "declare i1 @switch()\n"
 | 
						|
      "\n"
 | 
						|
      "define void @test() {\n"
 | 
						|
      "entry:\n"
 | 
						|
      "  %B = bitcast i8 undef to i8\n"
 | 
						|
      "  br label %loop\n"
 | 
						|
      "loop:\n"
 | 
						|
      "  %A = bitcast i8 undef to i8\n"
 | 
						|
      "  %x = call i1 @switch()\n"
 | 
						|
      "  br i1 %x, label %loop, label %exit\n"
 | 
						|
      "exit:\n"
 | 
						|
      "  ret void\n"
 | 
						|
      "}");
 | 
						|
  ExpectPath(false);
 | 
						|
}
 | 
						|
 | 
						|
TEST_F(IsPotentiallyReachableTest, SimpleLoop3) {
 | 
						|
  ParseAssembly(
 | 
						|
      "declare i1 @switch()\n"
 | 
						|
      "\n"
 | 
						|
      "define void @test() {\n"
 | 
						|
      "entry:\n"
 | 
						|
      "  br label %loop\n"
 | 
						|
      "loop:\n"
 | 
						|
      "  %B = bitcast i8 undef to i8\n"
 | 
						|
      "  %x = call i1 @switch()\n"
 | 
						|
      "  br i1 %x, label %loop, label %exit\n"
 | 
						|
      "exit:\n"
 | 
						|
      "  %A = bitcast i8 undef to i8\n"
 | 
						|
      "  ret void\n"
 | 
						|
      "}");
 | 
						|
  ExpectPath(false);
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
TEST_F(IsPotentiallyReachableTest, OneLoopAfterTheOther1) {
 | 
						|
  ParseAssembly(
 | 
						|
      "declare i1 @switch()\n"
 | 
						|
      "\n"
 | 
						|
      "define void @test() {\n"
 | 
						|
      "entry:\n"
 | 
						|
      "  br label %loop1\n"
 | 
						|
      "loop1:\n"
 | 
						|
      "  %A = bitcast i8 undef to i8\n"
 | 
						|
      "  %x = call i1 @switch()\n"
 | 
						|
      "  br i1 %x, label %loop1, label %loop1exit\n"
 | 
						|
      "loop1exit:\n"
 | 
						|
      "  br label %loop2\n"
 | 
						|
      "loop2:\n"
 | 
						|
      "  %B = bitcast i8 undef to i8\n"
 | 
						|
      "  %y = call i1 @switch()\n"
 | 
						|
      "  br i1 %x, label %loop2, label %loop2exit\n"
 | 
						|
      "loop2exit:"
 | 
						|
      "  ret void\n"
 | 
						|
      "}");
 | 
						|
  ExpectPath(true);
 | 
						|
}
 | 
						|
 | 
						|
TEST_F(IsPotentiallyReachableTest, OneLoopAfterTheOther2) {
 | 
						|
  ParseAssembly(
 | 
						|
      "declare i1 @switch()\n"
 | 
						|
      "\n"
 | 
						|
      "define void @test() {\n"
 | 
						|
      "entry:\n"
 | 
						|
      "  br label %loop1\n"
 | 
						|
      "loop1:\n"
 | 
						|
      "  %B = bitcast i8 undef to i8\n"
 | 
						|
      "  %x = call i1 @switch()\n"
 | 
						|
      "  br i1 %x, label %loop1, label %loop1exit\n"
 | 
						|
      "loop1exit:\n"
 | 
						|
      "  br label %loop2\n"
 | 
						|
      "loop2:\n"
 | 
						|
      "  %A = bitcast i8 undef to i8\n"
 | 
						|
      "  %y = call i1 @switch()\n"
 | 
						|
      "  br i1 %x, label %loop2, label %loop2exit\n"
 | 
						|
      "loop2exit:"
 | 
						|
      "  ret void\n"
 | 
						|
      "}");
 | 
						|
  ExpectPath(false);
 | 
						|
}
 | 
						|
 | 
						|
TEST_F(IsPotentiallyReachableTest, OneLoopAfterTheOtherInsideAThirdLoop) {
 | 
						|
  ParseAssembly(
 | 
						|
      "declare i1 @switch()\n"
 | 
						|
      "\n"
 | 
						|
      "define void @test() {\n"
 | 
						|
      "entry:\n"
 | 
						|
      "  br label %outerloop3\n"
 | 
						|
      "outerloop3:\n"
 | 
						|
      "  br label %innerloop1\n"
 | 
						|
      "innerloop1:\n"
 | 
						|
      "  %B = bitcast i8 undef to i8\n"
 | 
						|
      "  %x = call i1 @switch()\n"
 | 
						|
      "  br i1 %x, label %innerloop1, label %innerloop1exit\n"
 | 
						|
      "innerloop1exit:\n"
 | 
						|
      "  br label %innerloop2\n"
 | 
						|
      "innerloop2:\n"
 | 
						|
      "  %A = bitcast i8 undef to i8\n"
 | 
						|
      "  %y = call i1 @switch()\n"
 | 
						|
      "  br i1 %x, label %innerloop2, label %innerloop2exit\n"
 | 
						|
      "innerloop2exit:"
 | 
						|
      "  ;; In outer loop3 now.\n"
 | 
						|
      "  %z = call i1 @switch()\n"
 | 
						|
      "  br i1 %z, label %outerloop3, label %exit\n"
 | 
						|
      "exit:\n"
 | 
						|
      "  ret void\n"
 | 
						|
      "}");
 | 
						|
  ExpectPath(true);
 | 
						|
}
 | 
						|
 | 
						|
static const char *BranchInsideLoopIR =
 | 
						|
    "declare i1 @switch()\n"
 | 
						|
    "\n"
 | 
						|
    "define void @test() {\n"
 | 
						|
    "entry:\n"
 | 
						|
    "  br label %loop\n"
 | 
						|
    "loop:\n"
 | 
						|
    "  %x = call i1 @switch()\n"
 | 
						|
    "  br i1 %x, label %nextloopblock, label %exit\n"
 | 
						|
    "nextloopblock:\n"
 | 
						|
    "  %y = call i1 @switch()\n"
 | 
						|
    "  br i1 %y, label %left, label %right\n"
 | 
						|
    "left:\n"
 | 
						|
    "  %A = bitcast i8 undef to i8\n"
 | 
						|
    "  br label %loop\n"
 | 
						|
    "right:\n"
 | 
						|
    "  %B = bitcast i8 undef to i8\n"
 | 
						|
    "  br label %loop\n"
 | 
						|
    "exit:\n"
 | 
						|
    "  ret void\n"
 | 
						|
    "}";
 | 
						|
 | 
						|
TEST_F(IsPotentiallyReachableTest, BranchInsideLoop) {
 | 
						|
  ParseAssembly(BranchInsideLoopIR);
 | 
						|
  ExpectPath(true);
 | 
						|
}
 | 
						|
 | 
						|
TEST_F(IsPotentiallyReachableTest, ModifyTest) {
 | 
						|
  ParseAssembly(BranchInsideLoopIR);
 | 
						|
 | 
						|
  succ_iterator S = succ_begin(&*++M->getFunction("test")->begin());
 | 
						|
  BasicBlock *OldBB = S[0];
 | 
						|
  S[0] = S[1];
 | 
						|
  ExpectPath(false);
 | 
						|
  S[0] = OldBB;
 | 
						|
  ExpectPath(true);
 | 
						|
}
 |