diff --git a/clang/docs/LibASTMatchersReference.html b/clang/docs/LibASTMatchersReference.html --- a/clang/docs/LibASTMatchersReference.html +++ b/clang/docs/LibASTMatchersReference.html @@ -502,6 +502,20 @@ +
Matches tag declarations. + +Example matches X, Z, U, S, E + class X; + template<class T> class Z {}; + struct S {}; + union U {}; + enum E { + A, B, C + }; +
Matches template type parameter declarations. @@ -3942,36 +3956,6 @@
Matches RecordDecl object that are spelled with "class." - -Example matches C, but not S or U. - struct S {}; - class C {}; - union U {}; -
Matches RecordDecl object that are spelled with "struct." - -Example matches S, but not C or U. - struct S {}; - class C {}; - union U {}; -
Matches RecordDecl object that are spelled with "union." - -Example matches U, but not C or S. - struct S {}; - class C {}; - union U {}; -
Matches if a node equals a previously bound node. @@ -4075,6 +4059,17 @@
Matches TagDecl object that are spelled with "class." + +Example matches C, but not S, U or E. + struct S {}; + class C {}; + union U {}; + enum E {}; +
Matches if a declaration has a body attached. @@ -4097,6 +4092,39 @@
Matches TagDecl object that are spelled with "enum." + +Example matches E, but not C, S or U. + struct S {}; + class C {}; + union U {}; + enum E {}; +
Matches TagDecl object that are spelled with "struct." + +Example matches S, but not C, U or E. + struct S {}; + class C {}; + union U {}; + enum E {}; +
Matches TagDecl object that are spelled with "union." + +Example matches U, but not C, S or E. + struct S {}; + class C {}; + union U {}; + enum E {}; +
Matches a TemplateArgument of integral type with a given value. 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 @@ -1194,6 +1194,20 @@ extern const internal::VariadicDynCastAllOfMatcherenumConstantDecl; +/// Matches tag declarations. +/// +/// Example matches X, Z, U, S, E +/// \code +/// class X; +/// template class Z {}; +/// struct S {}; +/// union U {}; +/// enum E { +/// A, B, C +/// }; +/// \endcode +extern const internal::VariadicDynCastAllOfMatcher tagDecl; + /// Matches method declarations. /// /// Example matches y @@ -4845,42 +4859,58 @@ return InnerMatcher.matches(Node.getType(), Finder, Builder); } -/// Matches RecordDecl object that are spelled with "struct." +/// Matches TagDecl object that are spelled with "struct." /// -/// Example matches S, but not C or U. +/// Example matches S, but not C, U or E. /// \code /// struct S {}; /// class C {}; /// union U {}; +/// enum E {}; /// \endcode -AST_MATCHER(RecordDecl, isStruct) { +AST_MATCHER(TagDecl, isStruct) { return Node.isStruct(); } -/// Matches RecordDecl object that are spelled with "union." +/// Matches TagDecl object that are spelled with "union." /// -/// Example matches U, but not C or S. +/// Example matches U, but not C, S or E. /// \code /// struct S {}; /// class C {}; /// union U {}; +/// enum E {}; /// \endcode -AST_MATCHER(RecordDecl, isUnion) { +AST_MATCHER(TagDecl, isUnion) { return Node.isUnion(); } -/// Matches RecordDecl object that are spelled with "class." +/// Matches TagDecl object that are spelled with "class." /// -/// Example matches C, but not S or U. +/// Example matches C, but not S, U or E. /// \code /// struct S {}; /// class C {}; /// union U {}; +/// enum E {}; /// \endcode -AST_MATCHER(RecordDecl, isClass) { +AST_MATCHER(TagDecl, isClass) { return Node.isClass(); } +/// Matches TagDecl object that are spelled with "enum." +/// +/// Example matches E, but not C, S or U. +/// \code +/// struct S {}; +/// class C {}; +/// union U {}; +/// enum E {}; +/// \endcode +AST_MATCHER(TagDecl, isEnum) { + return Node.isEnum(); +} + /// Matches the true branch expression of a conditional operator. /// /// Example 1 (conditional ternary operator): matches a 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 @@ -647,6 +647,7 @@ const internal::VariadicDynCastAllOfMatcher enumDecl; const internal::VariadicDynCastAllOfMatcher enumConstantDecl; +const internal::VariadicDynCastAllOfMatcher tagDecl; const internal::VariadicDynCastAllOfMatcher cxxMethodDecl; const internal::VariadicDynCastAllOfMatcher cxxConversionDecl; 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 @@ -492,6 +492,7 @@ REGISTER_MATCHER(substTemplateTypeParmType); REGISTER_MATCHER(switchCase); REGISTER_MATCHER(switchStmt); + REGISTER_MATCHER(tagDecl); REGISTER_MATCHER(tagType); REGISTER_MATCHER(templateArgument); REGISTER_MATCHER(templateArgumentCountIs); 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 @@ -2504,6 +2504,13 @@ EXPECT_TRUE(notMatches("enum X {};", enumDecl(isScoped()))); } +TEST(TagDeclKind, MatchesTagDeclKind) { + EXPECT_TRUE(matches("struct X {};", tagDecl(isStruct()))); + EXPECT_TRUE(matches("class C {};", tagDecl(isClass()))); + EXPECT_TRUE(matches("union U {};", tagDecl(isUnion()))); + EXPECT_TRUE(matches("enum E {};", tagDecl(isEnum()))); +} + TEST(HasTrailingReturn, MatchesTrailingReturn) { EXPECT_TRUE(matches("auto Y() -> int { return 0; }", functionDecl(hasTrailingReturn()))); diff --git a/clang/unittests/ASTMatchers/ASTMatchersNodeTest.cpp b/clang/unittests/ASTMatchers/ASTMatchersNodeTest.cpp --- a/clang/unittests/ASTMatchers/ASTMatchersNodeTest.cpp +++ b/clang/unittests/ASTMatchers/ASTMatchersNodeTest.cpp @@ -184,6 +184,13 @@ EXPECT_TRUE(notMatches("enum X {};", Matcher)); } +TEST(TagDecl, MatchesTagDecls) { + EXPECT_TRUE(matches("struct X {};", tagDecl(hasName("X")))); + EXPECT_TRUE(matches("class C {};", tagDecl(hasName("C")))); + EXPECT_TRUE(matches("union U {};", tagDecl(hasName("U")))); + EXPECT_TRUE(matches("enum E {};", tagDecl(hasName("E")))); +} + TEST(Matcher, UnresolvedLookupExpr) { // FIXME: The test is known to be broken on Windows with delayed template // parsing.