From 2e8dc8590d8b412a131b127e7aa4aad0d7fc7fa0 Mon Sep 17 00:00:00 2001 From: Stephen Kelly Date: Fri, 6 Dec 2019 23:50:07 +0000 Subject: [PATCH] Add matchDynamic convenience functions Summary: These correspond to the existing match() free functions. Reviewers: aaron.ballman Subscribers: cfe-commits Differential Revision: https://reviews.llvm.org/D54406 --- clang/include/clang/ASTMatchers/ASTMatchFinder.h | 27 ++++++++++++++++++++++ .../unittests/ASTMatchers/ASTMatchersNodeTest.cpp | 24 +++++++++++++++++++ 2 files changed, 51 insertions(+) diff --git a/clang/include/clang/ASTMatchers/ASTMatchFinder.h b/clang/include/clang/ASTMatchers/ASTMatchFinder.h index 4c8dd87..f8160d5 100644 --- a/clang/include/clang/ASTMatchers/ASTMatchFinder.h +++ b/clang/include/clang/ASTMatchers/ASTMatchFinder.h @@ -309,6 +309,33 @@ match(MatcherT Matcher, ASTContext &Context) { return std::move(Callback.Nodes); } +inline SmallVector +matchDynamic(internal::DynTypedMatcher Matcher, + const ast_type_traits::DynTypedNode &Node, ASTContext &Context) { + internal::CollectMatchesCallback Callback; + MatchFinder Finder; + Finder.addDynamicMatcher(Matcher, &Callback); + Finder.match(Node, Context); + return std::move(Callback.Nodes); +} + +template +SmallVector matchDynamic(internal::DynTypedMatcher Matcher, + const NodeT &Node, + ASTContext &Context) { + return matchDynamic(Matcher, ast_type_traits::DynTypedNode::create(Node), + Context); +} + +inline SmallVector +matchDynamic(internal::DynTypedMatcher Matcher, ASTContext &Context) { + internal::CollectMatchesCallback Callback; + MatchFinder Finder; + Finder.addDynamicMatcher(Matcher, &Callback); + Finder.matchAST(Context); + return std::move(Callback.Nodes); +} + } // end namespace ast_matchers } // end namespace clang diff --git a/clang/unittests/ASTMatchers/ASTMatchersNodeTest.cpp b/clang/unittests/ASTMatchers/ASTMatchersNodeTest.cpp index 4c557cf..02514f7 100644 --- a/clang/unittests/ASTMatchers/ASTMatchersNodeTest.cpp +++ b/clang/unittests/ASTMatchers/ASTMatchersNodeTest.cpp @@ -1827,5 +1827,29 @@ void x(int x) { EXPECT_TRUE(notMatchesWithOpenMP(Source4, Matcher)); } +TEST(MatchFinderAPI, matchesDynamic) { + + std::string SourceCode = "struct A { void f() {} };"; + auto Matcher = functionDecl(isDefinition()).bind("method"); + + auto astUnit = tooling::buildASTFromCode(SourceCode); + + auto GlobalBoundNodes = matchDynamic(Matcher, astUnit->getASTContext()); + + EXPECT_EQ(GlobalBoundNodes.size(), 1u); + EXPECT_EQ(GlobalBoundNodes[0].getMap().size(), 1u); + + auto GlobalMethodNode = GlobalBoundNodes[0].getNodeAs("method"); + EXPECT_TRUE(GlobalMethodNode != nullptr); + + auto MethodBoundNodes = + matchDynamic(Matcher, *GlobalMethodNode, astUnit->getASTContext()); + EXPECT_EQ(MethodBoundNodes.size(), 1u); + EXPECT_EQ(MethodBoundNodes[0].getMap().size(), 1u); + + auto MethodNode = MethodBoundNodes[0].getNodeAs("method"); + EXPECT_EQ(MethodNode, GlobalMethodNode); +} + } // namespace ast_matchers } // namespace clang -- 2.7.4