diff --git a/clang/docs/LibASTMatchersReference.html b/clang/docs/LibASTMatchersReference.html --- a/clang/docs/LibASTMatchersReference.html +++ b/clang/docs/LibASTMatchersReference.html @@ -5613,6 +5613,40 @@ +
Matches function calls and constructor calls + +Because CallExpr and CXXConstructExpr do not share a common +base class with API accessing arguments etc, AST Matchers for code +which should match both are typically duplicated. This matcher +removes the need for duplication. + +Given code +struct ConstructorTakesInt +{ + ConstructorTakesInt(int i) {} +}; + +void callTakesInt(int i) +{ +} + +void doCall() +{ + callTakesInt(42); +} + +void doConstruct() +{ + ConstructorTakesInt cti(42); +} + +The matcher +callOrConstruct(hasArgument(0, integerLiteral(equals(42)))) +matches the expression in both doCall and doConstruct +
Matches if any of the given matchers matches. diff --git a/clang/include/clang/ASTMatchers/ASTMatchers.h b/clang/include/clang/ASTMatchers/ASTMatchers.h --- a/clang/include/clang/ASTMatchers/ASTMatchers.h +++ b/clang/include/clang/ASTMatchers/ASTMatchers.h @@ -2829,6 +2829,44 @@ CXXRewrittenBinaryOperator> binaryOperation; +/// Matches function calls and constructor calls +/// +/// Because CallExpr and CXXConstructExpr do not share a common +/// base class with API accessing arguments etc, AST Matchers for code +/// which should match both are typically duplicated. This matcher +/// removes the need for duplication. +/// +/// Given code +/// \code +/// struct ConstructorTakesInt +/// { +/// ConstructorTakesInt(int i) {} +/// }; +/// +/// void callTakesInt(int i) +/// { +/// } +/// +/// void doCall() +/// { +/// callTakesInt(42); +/// } +/// +/// void doConstruct() +/// { +/// ConstructorTakesInt cti(42); +/// } +/// \endcode +/// +/// The matcher +/// \code +/// callOrConstruct(hasArgument(0, integerLiteral(equals(42)))) +/// \endcode +/// matches the expression in both doCall and doConstruct +extern const internal::MapAnyOfMatcher< + CallExpr, CXXConstructExpr, CXXUnresolvedConstructExpr, ObjCMessageExpr> + callOrConstruct; + /// Matches unary expressions that have a specific type of argument. /// /// Given diff --git a/clang/lib/ASTMatchers/ASTMatchersInternal.cpp b/clang/lib/ASTMatchers/ASTMatchersInternal.cpp --- a/clang/lib/ASTMatchers/ASTMatchersInternal.cpp +++ b/clang/lib/ASTMatchers/ASTMatchersInternal.cpp @@ -924,6 +924,9 @@ const internal::MapAnyOfMatcherbinaryOperation; +const internal::MapAnyOfMatcher + callOrConstruct; const internal::VariadicDynCastAllOfMatcher unaryOperator; const internal::VariadicDynCastAllOfMatcher conditionalOperator; diff --git a/clang/lib/ASTMatchers/Dynamic/Registry.cpp b/clang/lib/ASTMatchers/Dynamic/Registry.cpp --- a/clang/lib/ASTMatchers/Dynamic/Registry.cpp +++ b/clang/lib/ASTMatchers/Dynamic/Registry.cpp @@ -152,6 +152,7 @@ REGISTER_MATCHER(builtinType); REGISTER_MATCHER(cStyleCastExpr); REGISTER_MATCHER(callExpr); + REGISTER_MATCHER(callOrConstruct); REGISTER_MATCHER(caseStmt); REGISTER_MATCHER(castExpr); REGISTER_MATCHER(characterLiteral); diff --git a/clang/unittests/ASTMatchers/ASTMatchersNarrowingTest.cpp b/clang/unittests/ASTMatchers/ASTMatchersNarrowingTest.cpp --- a/clang/unittests/ASTMatchers/ASTMatchersNarrowingTest.cpp +++ b/clang/unittests/ASTMatchers/ASTMatchersNarrowingTest.cpp @@ -863,6 +863,48 @@ mapAnyOf(unaryOperator, cxxOperatorCallExpr) .with(hasAnyOperatorName("+", "!"), forFunction(functionDecl(hasName("opFree"))))))); + + Code = R"cpp( +struct ConstructorTakesInt +{ + ConstructorTakesInt(int i) {} +}; + +void callTakesInt(int i) +{ + +} + +void doCall() +{ + callTakesInt(42); +} + +void doConstruct() +{ + ConstructorTakesInt cti(42); +} +)cpp"; + + EXPECT_TRUE(matches( + Code, + traverse(TK_IgnoreUnlessSpelledInSource, + callOrConstruct( + forFunction(functionDecl(hasName("doCall"))), + hasArgument(0, integerLiteral(equals(42))), + hasAnyArgument(integerLiteral(equals(42))), + forEachArgumentWithParam(integerLiteral(equals(42)), + parmVarDecl(hasName("i"))))))); + + EXPECT_TRUE(matches( + Code, + traverse(TK_IgnoreUnlessSpelledInSource, + callOrConstruct( + forFunction(functionDecl(hasName("doConstruct"))), + hasArgument(0, integerLiteral(equals(42))), + hasAnyArgument(integerLiteral(equals(42))), + forEachArgumentWithParam(integerLiteral(equals(42)), + parmVarDecl(hasName("i"))))))); } TEST_P(ASTMatchersTest, IsDerivedFrom) {