xref: /aosp_15_r20/external/clang/unittests/Analysis/CFGTest.cpp (revision 67e74705e28f6214e480b399dd47ea732279e315)
1*67e74705SXin Li //===- unittests/Analysis/CFGTest.cpp - CFG tests -------------------------===//
2*67e74705SXin Li //
3*67e74705SXin Li //                     The LLVM Compiler Infrastructure
4*67e74705SXin Li //
5*67e74705SXin Li // This file is distributed under the University of Illinois Open Source
6*67e74705SXin Li // License. See LICENSE.TXT for details.
7*67e74705SXin Li //
8*67e74705SXin Li //===----------------------------------------------------------------------===//
9*67e74705SXin Li 
10*67e74705SXin Li #include "clang/ASTMatchers/ASTMatchFinder.h"
11*67e74705SXin Li #include "clang/Analysis/CFG.h"
12*67e74705SXin Li #include "clang/Tooling/Tooling.h"
13*67e74705SXin Li #include "gtest/gtest.h"
14*67e74705SXin Li #include <string>
15*67e74705SXin Li #include <vector>
16*67e74705SXin Li 
17*67e74705SXin Li namespace clang {
18*67e74705SXin Li namespace analysis {
19*67e74705SXin Li namespace {
20*67e74705SXin Li 
21*67e74705SXin Li // Constructing a CFG for a range-based for over a dependent type fails (but
22*67e74705SXin Li // should not crash).
TEST(CFG,RangeBasedForOverDependentType)23*67e74705SXin Li TEST(CFG, RangeBasedForOverDependentType) {
24*67e74705SXin Li   const char *Code = "class Foo;\n"
25*67e74705SXin Li                      "template <typename T>\n"
26*67e74705SXin Li                      "void f(const T &Range) {\n"
27*67e74705SXin Li                      "  for (const Foo *TheFoo : Range) {\n"
28*67e74705SXin Li                      "  }\n"
29*67e74705SXin Li                      "}\n";
30*67e74705SXin Li 
31*67e74705SXin Li   class CFGCallback : public ast_matchers::MatchFinder::MatchCallback {
32*67e74705SXin Li   public:
33*67e74705SXin Li     bool SawFunctionBody = false;
34*67e74705SXin Li 
35*67e74705SXin Li     void run(const ast_matchers::MatchFinder::MatchResult &Result) override {
36*67e74705SXin Li       const auto *Func = Result.Nodes.getNodeAs<FunctionDecl>("func");
37*67e74705SXin Li       Stmt *Body = Func->getBody();
38*67e74705SXin Li       if (!Body)
39*67e74705SXin Li         return;
40*67e74705SXin Li       SawFunctionBody = true;
41*67e74705SXin Li       std::unique_ptr<CFG> cfg =
42*67e74705SXin Li           CFG::buildCFG(nullptr, Body, Result.Context, CFG::BuildOptions());
43*67e74705SXin Li       EXPECT_EQ(nullptr, cfg);
44*67e74705SXin Li     }
45*67e74705SXin Li   } Callback;
46*67e74705SXin Li 
47*67e74705SXin Li   ast_matchers::MatchFinder Finder;
48*67e74705SXin Li   Finder.addMatcher(ast_matchers::functionDecl().bind("func"), &Callback);
49*67e74705SXin Li   std::unique_ptr<tooling::FrontendActionFactory> Factory(
50*67e74705SXin Li       tooling::newFrontendActionFactory(&Finder));
51*67e74705SXin Li   std::vector<std::string> Args = {"-std=c++11", "-fno-delayed-template-parsing"};
52*67e74705SXin Li   ASSERT_TRUE(tooling::runToolOnCodeWithArgs(Factory->create(), Code, Args));
53*67e74705SXin Li   EXPECT_TRUE(Callback.SawFunctionBody);
54*67e74705SXin Li }
55*67e74705SXin Li 
56*67e74705SXin Li } // namespace
57*67e74705SXin Li } // namespace analysis
58*67e74705SXin Li } // namespace clang
59