1 //===- unittests/Analysis/CFGTest.cpp - CFG 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 "clang/ASTMatchers/ASTMatchFinder.h" 11 #include "clang/Analysis/CFG.h" 12 #include "clang/Tooling/Tooling.h" 13 #include "gtest/gtest.h" 14 #include <string> 15 #include <vector> 16 17 namespace clang { 18 namespace analysis { 19 namespace { 20 21 enum BuildResult { 22 ToolFailed, 23 ToolRan, 24 SawFunctionBody, 25 BuiltCFG, 26 }; 27 28 class CFGCallback : public ast_matchers::MatchFinder::MatchCallback { 29 public: 30 BuildResult TheBuildResult = ToolRan; 31 32 void run(const ast_matchers::MatchFinder::MatchResult &Result) override { 33 const auto *Func = Result.Nodes.getNodeAs<FunctionDecl>("func"); 34 Stmt *Body = Func->getBody(); 35 if (!Body) 36 return; 37 TheBuildResult = SawFunctionBody; 38 if (CFG::buildCFG(nullptr, Body, Result.Context, CFG::BuildOptions())) 39 TheBuildResult = BuiltCFG; 40 } 41 }; 42 43 BuildResult BuildCFG(const char *Code) { 44 CFGCallback Callback; 45 46 ast_matchers::MatchFinder Finder; 47 Finder.addMatcher(ast_matchers::functionDecl().bind("func"), &Callback); 48 std::unique_ptr<tooling::FrontendActionFactory> Factory( 49 tooling::newFrontendActionFactory(&Finder)); 50 std::vector<std::string> Args = {"-std=c++11", "-fno-delayed-template-parsing"}; 51 if (!tooling::runToolOnCodeWithArgs(Factory->create(), Code, Args)) 52 return ToolFailed; 53 return Callback.TheBuildResult; 54 } 55 56 // Constructing a CFG for a range-based for over a dependent type fails (but 57 // should not crash). 58 TEST(CFG, RangeBasedForOverDependentType) { 59 const char *Code = "class Foo;\n" 60 "template <typename T>\n" 61 "void f(const T &Range) {\n" 62 " for (const Foo *TheFoo : Range) {\n" 63 " }\n" 64 "}\n"; 65 EXPECT_EQ(SawFunctionBody, BuildCFG(Code)); 66 } 67 68 // Constructing a CFG containing a delete expression on a dependent type should 69 // not crash. 70 TEST(CFG, DeleteExpressionOnDependentType) { 71 const char *Code = "template<class T>\n" 72 "void f(T t) {\n" 73 " delete t;\n" 74 "}\n"; 75 EXPECT_EQ(BuiltCFG, BuildCFG(Code)); 76 } 77 78 } // namespace 79 } // namespace analysis 80 } // namespace clang 81