| 1 | // unittests/ASTMatchers/ASTMatchersInternalTest.cpp - AST matcher unit tests // |
| 2 | // |
| 3 | // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. |
| 4 | // See https://llvm.org/LICENSE.txt for license information. |
| 5 | // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception |
| 6 | // |
| 7 | //===----------------------------------------------------------------------===// |
| 8 | |
| 9 | #include "ASTMatchersTest.h" |
| 10 | #include "clang/AST/PrettyPrinter.h" |
| 11 | #include "clang/ASTMatchers/ASTMatchFinder.h" |
| 12 | #include "clang/ASTMatchers/ASTMatchers.h" |
| 13 | #include "clang/Tooling/Tooling.h" |
| 14 | #include "llvm/TargetParser/Host.h" |
| 15 | #include "llvm/TargetParser/Triple.h" |
| 16 | #include "llvm/Testing/Support/SupportHelpers.h" |
| 17 | #include "gtest/gtest.h" |
| 18 | |
| 19 | namespace clang { |
| 20 | namespace ast_matchers { |
| 21 | using internal::DynTypedMatcher; |
| 22 | |
| 23 | #if GTEST_HAS_DEATH_TEST |
| 24 | TEST(HasNameDeathTest, DiesOnEmptyName) { |
| 25 | ASSERT_DEBUG_DEATH({ |
| 26 | DeclarationMatcher HasEmptyName = recordDecl(hasName("" )); |
| 27 | EXPECT_TRUE(notMatches("class X {};" , HasEmptyName)); |
| 28 | }, "" ); |
| 29 | } |
| 30 | |
| 31 | TEST(HasNameDeathTest, DiesOnEmptyPattern) { |
| 32 | ASSERT_DEBUG_DEATH({ |
| 33 | DeclarationMatcher HasEmptyName = recordDecl(matchesName("" )); |
| 34 | EXPECT_TRUE(notMatches("class X {};" , HasEmptyName)); |
| 35 | }, "" ); |
| 36 | } |
| 37 | |
| 38 | // FIXME Re-enable these tests without breaking standalone builds. |
| 39 | #if 0 |
| 40 | // FIXME: Figure out why back traces aren't being generated on clang builds on |
| 41 | // windows. |
| 42 | #if ENABLE_BACKTRACES && (!defined(_MSC_VER) || !defined(__clang__)) |
| 43 | |
| 44 | AST_MATCHER(Decl, causeCrash) { |
| 45 | abort(); |
| 46 | return true; |
| 47 | } |
| 48 | |
| 49 | TEST(MatcherCrashDeathTest, CrashOnMatcherDump) { |
| 50 | llvm::EnablePrettyStackTrace(); |
| 51 | auto Matcher = testing::HasSubstr( |
| 52 | "ASTMatcher: Matching '<unknown>' against:\n\tFunctionDecl foo : " |
| 53 | "<input.cc:1:1, col:10>" ); |
| 54 | ASSERT_DEATH(matches("void foo();" , functionDecl(causeCrash())), Matcher); |
| 55 | } |
| 56 | |
| 57 | template <typename MatcherT> |
| 58 | static void crashTestNodeDump(MatcherT Matcher, |
| 59 | ArrayRef<StringRef> MatchedNodes, |
| 60 | StringRef Against, StringRef Code) { |
| 61 | llvm::EnablePrettyStackTrace(); |
| 62 | MatchFinder Finder; |
| 63 | |
| 64 | struct CrashCallback : public MatchFinder::MatchCallback { |
| 65 | void run(const MatchFinder::MatchResult &Result) override { abort(); } |
| 66 | std::optional<TraversalKind> getCheckTraversalKind() const override { |
| 67 | return TK_IgnoreUnlessSpelledInSource; |
| 68 | } |
| 69 | StringRef getID() const override { return "CrashTester" ; } |
| 70 | } Callback; |
| 71 | Finder.addMatcher(std::move(Matcher), &Callback); |
| 72 | if (MatchedNodes.empty()) { |
| 73 | ASSERT_DEATH(tooling::runToolOnCode( |
| 74 | newFrontendActionFactory(&Finder)->create(), Code), |
| 75 | testing::HasSubstr( |
| 76 | ("ASTMatcher: Processing 'CrashTester' against:\n\t" + |
| 77 | Against + "\nNo bound nodes" ) |
| 78 | .str())); |
| 79 | } else { |
| 80 | std::vector<testing::PolymorphicMatcher< |
| 81 | testing::internal::HasSubstrMatcher<std::string>>> |
| 82 | Matchers; |
| 83 | Matchers.reserve(MatchedNodes.size()); |
| 84 | for (auto Node : MatchedNodes) { |
| 85 | Matchers.push_back(testing::HasSubstr(Node.str())); |
| 86 | } |
| 87 | auto CrashMatcher = testing::AllOf( |
| 88 | testing::HasSubstr( |
| 89 | ("ASTMatcher: Processing 'CrashTester' against:\n\t" + Against + |
| 90 | "\n--- Bound Nodes Begin ---" ) |
| 91 | .str()), |
| 92 | testing::HasSubstr("--- Bound Nodes End ---" ), |
| 93 | testing::AllOfArray(Matchers)); |
| 94 | |
| 95 | ASSERT_DEATH(tooling::runToolOnCode( |
| 96 | newFrontendActionFactory(&Finder)->create(), Code), |
| 97 | CrashMatcher); |
| 98 | } |
| 99 | } |
| 100 | TEST(MatcherCrashDeathTest, CrashOnCallbackDump) { |
| 101 | crashTestNodeDump(forStmt(), {}, "ForStmt : <input.cc:1:14, col:21>" , |
| 102 | "void foo() { for(;;); }" ); |
| 103 | crashTestNodeDump( |
| 104 | forStmt(hasLoopInit(declStmt(hasSingleDecl( |
| 105 | varDecl(hasType(qualType().bind("QT" )), |
| 106 | hasType(type().bind("T" )), |
| 107 | hasInitializer( |
| 108 | integerLiteral().bind("IL" ))) |
| 109 | .bind("VD" ))) |
| 110 | .bind("DS" ))) |
| 111 | .bind("FS" ), |
| 112 | {"FS - { ForStmt : <input.cc:3:5, line:4:5> }" , |
| 113 | "DS - { DeclStmt : <input.cc:3:10, col:19> }" , |
| 114 | "IL - { IntegerLiteral : <input.cc:3:18> }" , "QT - { QualType : int }" , |
| 115 | "T - { BuiltinType : int }" , |
| 116 | "VD - { VarDecl I : <input.cc:3:10, col:18> }" }, |
| 117 | "ForStmt : <input.cc:3:5, line:4:5>" , |
| 118 | R"cpp( |
| 119 | void foo() { |
| 120 | for (int I = 0; I < 5; ++I) { |
| 121 | } |
| 122 | } |
| 123 | )cpp" ); |
| 124 | crashTestNodeDump( |
| 125 | cxxRecordDecl(hasMethod(cxxMethodDecl(hasName("operator+" )).bind("Op+" ))) |
| 126 | .bind("Unnamed" ), |
| 127 | {"Unnamed - { CXXRecordDecl (anonymous) : <input.cc:1:1, col:36> }" , |
| 128 | "Op+ - { CXXMethodDecl (anonymous struct)::operator+ : <input.cc:1:10, " |
| 129 | "col:29> }" }, |
| 130 | "CXXRecordDecl (anonymous) : <input.cc:1:1, col:36>" , |
| 131 | "struct { int operator+(int) const; } Unnamed;" ); |
| 132 | crashTestNodeDump( |
| 133 | cxxRecordDecl(hasMethod(cxxConstructorDecl(isDefaulted()).bind("Ctor" )), |
| 134 | hasMethod(cxxDestructorDecl(isDefaulted()).bind("Dtor" ))), |
| 135 | {"Ctor - { CXXConstructorDecl Foo::Foo : <input.cc:1:14, col:28> }" , |
| 136 | "Dtor - { CXXDestructorDecl Foo::~Foo : <input.cc:1:31, col:46> }" }, |
| 137 | "CXXRecordDecl Foo : <input.cc:1:1, col:49>" , |
| 138 | "struct Foo { Foo() = default; ~Foo() = default; };" ); |
| 139 | } |
| 140 | #endif // ENABLE_BACKTRACES |
| 141 | #endif |
| 142 | #endif |
| 143 | |
| 144 | TEST(ConstructVariadic, MismatchedTypes_Regression) { |
| 145 | EXPECT_TRUE( |
| 146 | matches("const int a = 0;" , internal::DynTypedMatcher::constructVariadic( |
| 147 | internal::DynTypedMatcher::VO_AnyOf, |
| 148 | ASTNodeKind::getFromNodeKind<QualType>(), |
| 149 | {isConstQualified(), arrayType()}) |
| 150 | .convertTo<QualType>())); |
| 151 | } |
| 152 | |
| 153 | // For testing AST_MATCHER_P(). |
| 154 | AST_MATCHER_P(Decl, just, internal::Matcher<Decl>, AMatcher) { |
| 155 | // Make sure all special variables are used: node, match_finder, |
| 156 | // bound_nodes_builder, and the parameter named 'AMatcher'. |
| 157 | return AMatcher.matches(Node, Finder, Builder); |
| 158 | } |
| 159 | |
| 160 | TEST(AstMatcherPMacro, Works) { |
| 161 | DeclarationMatcher HasClassB = just(AMatcher: has(recordDecl(hasName(Name: "B" )).bind(ID: "b" ))); |
| 162 | |
| 163 | EXPECT_TRUE(matchAndVerifyResultTrue("class A { class B {}; };" , |
| 164 | HasClassB, std::make_unique<VerifyIdIsBoundTo<Decl>>("b" ))); |
| 165 | |
| 166 | EXPECT_TRUE(matchAndVerifyResultFalse("class A { class B {}; };" , |
| 167 | HasClassB, std::make_unique<VerifyIdIsBoundTo<Decl>>("a" ))); |
| 168 | |
| 169 | EXPECT_TRUE(matchAndVerifyResultFalse("class A { class C {}; };" , |
| 170 | HasClassB, std::make_unique<VerifyIdIsBoundTo<Decl>>("b" ))); |
| 171 | } |
| 172 | |
| 173 | AST_POLYMORPHIC_MATCHER_P(polymorphicHas, |
| 174 | AST_POLYMORPHIC_SUPPORTED_TYPES(Decl, Stmt), |
| 175 | internal::Matcher<Decl>, AMatcher) { |
| 176 | return Finder->matchesChildOf( |
| 177 | Node, AMatcher, Builder, |
| 178 | ASTMatchFinder::BK_First); |
| 179 | } |
| 180 | |
| 181 | TEST(AstPolymorphicMatcherPMacro, Works) { |
| 182 | DeclarationMatcher HasClassB = |
| 183 | polymorphicHas(AMatcher: recordDecl(hasName(Name: "B" )).bind(ID: "b" )); |
| 184 | |
| 185 | EXPECT_TRUE(matchAndVerifyResultTrue("class A { class B {}; };" , |
| 186 | HasClassB, std::make_unique<VerifyIdIsBoundTo<Decl>>("b" ))); |
| 187 | |
| 188 | EXPECT_TRUE(matchAndVerifyResultFalse("class A { class B {}; };" , |
| 189 | HasClassB, std::make_unique<VerifyIdIsBoundTo<Decl>>("a" ))); |
| 190 | |
| 191 | EXPECT_TRUE(matchAndVerifyResultFalse("class A { class C {}; };" , |
| 192 | HasClassB, std::make_unique<VerifyIdIsBoundTo<Decl>>("b" ))); |
| 193 | |
| 194 | StatementMatcher StatementHasClassB = |
| 195 | polymorphicHas(AMatcher: recordDecl(hasName(Name: "B" ))); |
| 196 | |
| 197 | EXPECT_TRUE(matches("void x() { class B {}; }" , StatementHasClassB)); |
| 198 | } |
| 199 | |
| 200 | TEST(MatchFinder, CheckProfiling) { |
| 201 | MatchFinder::MatchFinderOptions Options; |
| 202 | llvm::StringMap<llvm::TimeRecord> Records; |
| 203 | Options.CheckProfiling.emplace(args&: Records); |
| 204 | MatchFinder Finder(std::move(Options)); |
| 205 | |
| 206 | struct NamedCallback : public MatchFinder::MatchCallback { |
| 207 | void run(const MatchFinder::MatchResult &Result) override {} |
| 208 | StringRef getID() const override { return "MyID" ; } |
| 209 | } Callback; |
| 210 | Finder.addMatcher(NodeMatch: decl(), Action: &Callback); |
| 211 | std::unique_ptr<FrontendActionFactory> Factory( |
| 212 | newFrontendActionFactory(ConsumerFactory: &Finder)); |
| 213 | ASSERT_TRUE(tooling::runToolOnCode(Factory->create(), "int x;" )); |
| 214 | |
| 215 | EXPECT_EQ(1u, Records.size()); |
| 216 | EXPECT_EQ("MyID" , Records.begin()->getKey()); |
| 217 | } |
| 218 | |
| 219 | class VerifyStartOfTranslationUnit : public MatchFinder::MatchCallback { |
| 220 | public: |
| 221 | VerifyStartOfTranslationUnit() : Called(false) {} |
| 222 | void run(const MatchFinder::MatchResult &Result) override { |
| 223 | EXPECT_TRUE(Called); |
| 224 | } |
| 225 | void onStartOfTranslationUnit() override { Called = true; } |
| 226 | bool Called; |
| 227 | }; |
| 228 | |
| 229 | TEST(MatchFinder, InterceptsStartOfTranslationUnit) { |
| 230 | MatchFinder Finder; |
| 231 | VerifyStartOfTranslationUnit VerifyCallback; |
| 232 | Finder.addMatcher(NodeMatch: decl(), Action: &VerifyCallback); |
| 233 | std::unique_ptr<FrontendActionFactory> Factory( |
| 234 | newFrontendActionFactory(ConsumerFactory: &Finder)); |
| 235 | ASSERT_TRUE(tooling::runToolOnCode(Factory->create(), "int x;" )); |
| 236 | EXPECT_TRUE(VerifyCallback.Called); |
| 237 | |
| 238 | VerifyCallback.Called = false; |
| 239 | std::unique_ptr<ASTUnit> AST(tooling::buildASTFromCode(Code: "int x;" )); |
| 240 | ASSERT_TRUE(AST.get()); |
| 241 | Finder.matchAST(Context&: AST->getASTContext()); |
| 242 | EXPECT_TRUE(VerifyCallback.Called); |
| 243 | } |
| 244 | |
| 245 | class VerifyEndOfTranslationUnit : public MatchFinder::MatchCallback { |
| 246 | public: |
| 247 | VerifyEndOfTranslationUnit() : Called(false) {} |
| 248 | void run(const MatchFinder::MatchResult &Result) override { |
| 249 | EXPECT_FALSE(Called); |
| 250 | } |
| 251 | void onEndOfTranslationUnit() override { Called = true; } |
| 252 | bool Called; |
| 253 | }; |
| 254 | |
| 255 | TEST(MatchFinder, InterceptsEndOfTranslationUnit) { |
| 256 | MatchFinder Finder; |
| 257 | VerifyEndOfTranslationUnit VerifyCallback; |
| 258 | Finder.addMatcher(NodeMatch: decl(), Action: &VerifyCallback); |
| 259 | std::unique_ptr<FrontendActionFactory> Factory( |
| 260 | newFrontendActionFactory(ConsumerFactory: &Finder)); |
| 261 | ASSERT_TRUE(tooling::runToolOnCode(Factory->create(), "int x;" )); |
| 262 | EXPECT_TRUE(VerifyCallback.Called); |
| 263 | |
| 264 | VerifyCallback.Called = false; |
| 265 | std::unique_ptr<ASTUnit> AST(tooling::buildASTFromCode(Code: "int x;" )); |
| 266 | ASSERT_TRUE(AST.get()); |
| 267 | Finder.matchAST(Context&: AST->getASTContext()); |
| 268 | EXPECT_TRUE(VerifyCallback.Called); |
| 269 | } |
| 270 | |
| 271 | TEST(Matcher, matchOverEntireASTContext) { |
| 272 | std::unique_ptr<ASTUnit> AST = |
| 273 | clang::tooling::buildASTFromCode(Code: "struct { int *foo; };" ); |
| 274 | ASSERT_TRUE(AST.get()); |
| 275 | auto PT = selectFirst<PointerType>( |
| 276 | BoundTo: "x" , Results: match(Matcher: pointerType().bind(ID: "x" ), Context&: AST->getASTContext())); |
| 277 | EXPECT_NE(nullptr, PT); |
| 278 | } |
| 279 | |
| 280 | TEST(DynTypedMatcherTest, TraversalKindForwardsToImpl) { |
| 281 | auto M = DynTypedMatcher(decl()); |
| 282 | EXPECT_FALSE(M.getTraversalKind()); |
| 283 | |
| 284 | M = DynTypedMatcher(traverse(TK: TK_AsIs, InnerMatcher: decl())); |
| 285 | EXPECT_THAT(M.getTraversalKind(), llvm::ValueIs(TK_AsIs)); |
| 286 | } |
| 287 | |
| 288 | TEST(DynTypedMatcherTest, ConstructWithTraversalKindSetsTK) { |
| 289 | auto M = DynTypedMatcher(decl()).withTraversalKind(TK: TK_AsIs); |
| 290 | EXPECT_THAT(M.getTraversalKind(), llvm::ValueIs(TK_AsIs)); |
| 291 | } |
| 292 | |
| 293 | TEST(DynTypedMatcherTest, ConstructWithTraversalKindOverridesNestedTK) { |
| 294 | auto M = DynTypedMatcher(decl()).withTraversalKind(TK: TK_AsIs).withTraversalKind( |
| 295 | TK: TK_IgnoreUnlessSpelledInSource); |
| 296 | EXPECT_THAT(M.getTraversalKind(), |
| 297 | llvm::ValueIs(TK_IgnoreUnlessSpelledInSource)); |
| 298 | } |
| 299 | |
| 300 | TEST(IsInlineMatcher, IsInline) { |
| 301 | EXPECT_TRUE(matches("void g(); inline void f();" , |
| 302 | functionDecl(isInline(), hasName("f" )))); |
| 303 | EXPECT_TRUE(matches("namespace n { inline namespace m {} }" , |
| 304 | namespaceDecl(isInline(), hasName("m" )))); |
| 305 | EXPECT_TRUE(matches("inline int Foo = 5;" , |
| 306 | varDecl(isInline(), hasName("Foo" )), {Lang_CXX17})); |
| 307 | } |
| 308 | |
| 309 | // FIXME: Figure out how to specify paths so the following tests pass on |
| 310 | // Windows. |
| 311 | #ifndef _WIN32 |
| 312 | |
| 313 | TEST(Matcher, IsExpansionInMainFileMatcher) { |
| 314 | EXPECT_TRUE(matches("class X {};" , |
| 315 | recordDecl(hasName("X" ), isExpansionInMainFile()))); |
| 316 | EXPECT_TRUE(notMatches("" , recordDecl(isExpansionInMainFile()))); |
| 317 | FileContentMappings M; |
| 318 | M.push_back(x: std::make_pair(x: "/other" , y: "class X {};" )); |
| 319 | EXPECT_TRUE(matchesConditionally("#include <other>\n" , |
| 320 | recordDecl(isExpansionInMainFile()), false, |
| 321 | {"-isystem/" }, M)); |
| 322 | } |
| 323 | |
| 324 | TEST(Matcher, IsExpansionInSystemHeader) { |
| 325 | FileContentMappings M; |
| 326 | M.push_back(x: std::make_pair(x: "/other" , y: "class X {};" )); |
| 327 | EXPECT_TRUE(matchesConditionally("#include \"other\"\n" , |
| 328 | recordDecl(isExpansionInSystemHeader()), |
| 329 | true, {"-isystem/" }, M)); |
| 330 | EXPECT_TRUE(matchesConditionally("#include \"other\"\n" , |
| 331 | recordDecl(isExpansionInSystemHeader()), |
| 332 | false, {"-I/" }, M)); |
| 333 | EXPECT_TRUE(notMatches("class X {};" , |
| 334 | recordDecl(isExpansionInSystemHeader()))); |
| 335 | EXPECT_TRUE(notMatches("" , recordDecl(isExpansionInSystemHeader()))); |
| 336 | } |
| 337 | |
| 338 | TEST(Matcher, IsExpansionInFileMatching) { |
| 339 | FileContentMappings M; |
| 340 | M.push_back(x: std::make_pair(x: "/foo" , y: "class A {};" )); |
| 341 | M.push_back(x: std::make_pair(x: "/bar" , y: "class B {};" )); |
| 342 | EXPECT_TRUE(matchesConditionally( |
| 343 | "#include <foo>\n" |
| 344 | "#include <bar>\n" |
| 345 | "class X {};" , |
| 346 | recordDecl(isExpansionInFileMatching("b.*" ), hasName("B" )), true, |
| 347 | {"-isystem/" }, M)); |
| 348 | EXPECT_TRUE(matchesConditionally( |
| 349 | "#include <foo>\n" |
| 350 | "#include <bar>\n" |
| 351 | "class X {};" , |
| 352 | recordDecl(isExpansionInFileMatching("f.*" ), hasName("X" )), false, |
| 353 | {"-isystem/" }, M)); |
| 354 | } |
| 355 | |
| 356 | #endif // _WIN32 |
| 357 | |
| 358 | } // end namespace ast_matchers |
| 359 | } // end namespace clang |
| 360 | |