//===- unittests/Analysis/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 "clang/ASTMatchers/ASTMatchFinder.h" #include "clang/Analysis/CFG.h" #include "clang/Tooling/Tooling.h" #include "gtest/gtest.h" #include <string> #include <vector> namespace clang { namespace analysis { namespace { // Constructing a CFG for a range-based for over a dependent type fails (but // should not crash). TEST(CFG, RangeBasedForOverDependentType) { const char *Code = "class Foo;\n" "template <typename T>\n" "void f(const T &Range) {\n" " for (const Foo *TheFoo : Range) {\n" " }\n" "}\n"; class CFGCallback : public ast_matchers::MatchFinder::MatchCallback { public: bool SawFunctionBody = false; void run(const ast_matchers::MatchFinder::MatchResult &Result) override { const auto *Func = Result.Nodes.getNodeAs<FunctionDecl>("func"); Stmt *Body = Func->getBody(); if (!Body) return; SawFunctionBody = true; std::unique_ptr<CFG> cfg = CFG::buildCFG(nullptr, Body, Result.Context, CFG::BuildOptions()); EXPECT_EQ(nullptr, cfg); } } Callback; ast_matchers::MatchFinder Finder; Finder.addMatcher(ast_matchers::functionDecl().bind("func"), &Callback); std::unique_ptr<tooling::FrontendActionFactory> Factory( tooling::newFrontendActionFactory(&Finder)); std::vector<std::string> Args = {"-std=c++11", "-fno-delayed-template-parsing"}; ASSERT_TRUE(tooling::runToolOnCodeWithArgs(Factory->create(), Code, Args)); EXPECT_TRUE(Callback.SawFunctionBody); } } // namespace } // namespace analysis } // namespace clang