Skip to content

Commit bf968e2

Browse files
committed
[ASTMatchers] Add support for dynamic matching of ofKind narrowing matcher
Summary: Adds support for using the ofKind in clang-query and other dynamic matcher use cases Reviewers: klimek, aaron.ballman, jdoerfert Reviewed By: aaron.ballman Subscribers: cfe-commits Tags: #clang Differential Revision: https://reviews.llvm.org/D77791
1 parent 2477cec commit bf968e2

File tree

4 files changed

+53
-3
lines changed

4 files changed

+53
-3
lines changed

clang/lib/ASTMatchers/Dynamic/Marshallers.cpp

Lines changed: 14 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -95,3 +95,17 @@ clang::ast_matchers::dynamic::internal::ArgTypeTraits<
9595
"OMPC_");
9696
return llvm::None;
9797
}
98+
99+
llvm::Optional<std::string>
100+
clang::ast_matchers::dynamic::internal::ArgTypeTraits<
101+
clang::UnaryExprOrTypeTrait>::getBestGuess(const VariantValue &Value) {
102+
static constexpr llvm::StringRef Allowed[] = {
103+
"UETT_SizeOf", "UETT_AlignOf",
104+
"UETT_VecStep", "UETT_OpenMPRequiredSimdAlign",
105+
"UETT_PreferredAlignOf",
106+
};
107+
if (Value.isString())
108+
return ::getBestGuess(Value.getString(), llvm::makeArrayRef(Allowed),
109+
"UETT_");
110+
return llvm::None;
111+
}

clang/lib/ASTMatchers/Dynamic/Marshallers.h

Lines changed: 30 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -27,6 +27,7 @@
2727
#include "clang/Basic/AttrKinds.h"
2828
#include "clang/Basic/LLVM.h"
2929
#include "clang/Basic/OpenMPKinds.h"
30+
#include "clang/Basic/TypeTraits.h"
3031
#include "llvm/ADT/ArrayRef.h"
3132
#include "llvm/ADT/None.h"
3233
#include "llvm/ADT/Optional.h"
@@ -214,6 +215,35 @@ template <> struct ArgTypeTraits<OpenMPClauseKind> {
214215
static llvm::Optional<std::string> getBestGuess(const VariantValue &Value);
215216
};
216217

218+
template <> struct ArgTypeTraits<UnaryExprOrTypeTrait> {
219+
private:
220+
static Optional<UnaryExprOrTypeTrait>
221+
getUnaryOrTypeTraitKind(llvm::StringRef ClauseKind) {
222+
// FIXME: Type traits should probably be in a `.def` to make less error
223+
// prone.
224+
return llvm::StringSwitch<Optional<UnaryExprOrTypeTrait>>(ClauseKind)
225+
.Case("UETT_SizeOf", UETT_SizeOf)
226+
.Case("UETT_AlignOf", UETT_AlignOf)
227+
.Case("UETT_VecStep", UETT_VecStep)
228+
.Case("UETT_OpenMPRequiredSimdAlign", UETT_OpenMPRequiredSimdAlign)
229+
.Case("UETT_PreferredAlignOf", UETT_PreferredAlignOf)
230+
.Default(llvm::None);
231+
}
232+
233+
public:
234+
static bool is(const VariantValue &Value) {
235+
return Value.isString() && getUnaryOrTypeTraitKind(Value.getString());
236+
}
237+
238+
static UnaryExprOrTypeTrait get(const VariantValue &Value) {
239+
return *getUnaryOrTypeTraitKind(Value.getString());
240+
}
241+
242+
static ArgKind getKind() { return ArgKind(ArgKind::AK_String); }
243+
244+
static llvm::Optional<std::string> getBestGuess(const VariantValue &Value);
245+
};
246+
217247
/// Matcher descriptor interface.
218248
///
219249
/// Provides a \c create() method that constructs the matcher from the provided

clang/lib/ASTMatchers/Dynamic/Registry.cpp

Lines changed: 1 addition & 3 deletions
Original file line numberDiff line numberDiff line change
@@ -95,9 +95,6 @@ void RegistryMaps::registerMatcher(
9595
RegistryMaps::RegistryMaps() {
9696
// TODO: Here is the list of the missing matchers, grouped by reason.
9797
//
98-
// Need Variant/Parser fixes:
99-
// ofKind
100-
//
10198
// Polymorphic + argument overload:
10299
// findAll
103100
//
@@ -458,6 +455,7 @@ RegistryMaps::RegistryMaps() {
458455
REGISTER_MATCHER(objcThrowStmt);
459456
REGISTER_MATCHER(objcTryStmt);
460457
REGISTER_MATCHER(ofClass);
458+
REGISTER_MATCHER(ofKind);
461459
REGISTER_MATCHER(ompDefaultClause);
462460
REGISTER_MATCHER(ompExecutableDirective);
463461
REGISTER_MATCHER(on);

clang/unittests/ASTMatchers/Dynamic/ParserTest.cpp

Lines changed: 8 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -242,6 +242,14 @@ TEST(ParserTest, FullParserTest) {
242242
EXPECT_TRUE(matches("void f(int a, int x);", M));
243243
EXPECT_FALSE(matches("void f(int x, int a);", M));
244244

245+
Code = "unaryExprOrTypeTraitExpr(ofKind(\"UETT_SizeOf\"))";
246+
llvm::Optional<DynTypedMatcher> UnaryExprSizeOf(
247+
Parser::parseMatcherExpression(Code, nullptr, nullptr, &Error));
248+
EXPECT_EQ("", Error.toStringFull());
249+
Matcher<Stmt> MStmt = UnaryExprSizeOf->unconditionalConvertTo<Stmt>();
250+
EXPECT_TRUE(matches("unsigned X = sizeof(int);", MStmt));
251+
EXPECT_FALSE(matches("unsigned X = alignof(int);", MStmt));
252+
245253
Code = "hasInitializer(\n binaryOperator(hasLHS(\"A\")))";
246254
EXPECT_TRUE(!Parser::parseMatcherExpression(Code, &Error).hasValue());
247255
EXPECT_EQ("1:1: Error parsing argument 1 for matcher hasInitializer.\n"

0 commit comments

Comments
 (0)