15#include "llvm/ADT/STLExtras.h"
18using namespace ast_matchers;
27 const auto IgnoreDerivedToBase = [](
const Expr *
E,
auto Matcher) {
30 if (
const auto *Cast = dyn_cast<ImplicitCastExpr>(
E)) {
31 if ((Cast->getCastKind() == CK_DerivedToBase ||
32 Cast->getCastKind() == CK_UncheckedDerivedToBase) &&
33 Matcher(Cast->getSubExpr()))
39 const auto EvalCommaExpr = [](
const Expr *
E,
auto Matcher) {
41 while (
const auto *BOComma =
42 dyn_cast_or_null<BinaryOperator>(
Result->IgnoreParens())) {
43 if (!BOComma->isCommaOp())
45 Result = BOComma->getRHS();
58 const auto ConditionalOperatorM = [
Target](
const Expr *
E) {
59 if (
const auto *CO = dyn_cast<AbstractConditionalOperator>(
E)) {
60 const auto *TE = CO->getTrueExpr()->IgnoreParens();
63 const auto *FE = CO->getFalseExpr()->IgnoreParens();
71 return IgnoreDerivedToBase(SourceExprP,
73 return E ==
Target || ConditionalOperatorM(
E);
75 EvalCommaExpr(SourceExprP, [&](
const Expr *
E) {
76 return IgnoreDerivedToBase(
86 ast_matchers::internal::Matcher<Expr>, InnerMatcher) {
87 if (
Node.isTypeDependent()) {
88 return InnerMatcher.matches(*
Node.getLHS(), Finder, Builder) ||
89 InnerMatcher.matches(*
Node.getRHS(), Finder, Builder);
91 return InnerMatcher.matches(*
Node.getBase(), Finder, Builder);
97 return llvm::is_contained(
Node.capture_inits(),
E);
101 ast_matchers::internal::Matcher<DeclStmt>, InnerMatcher) {
102 const DeclStmt *
const Range =
Node.getRangeStmt();
103 return InnerMatcher.matches(*
Range, Finder, Builder);
107 auto *Exp = dyn_cast<Expr>(&
Node);
110 auto *
Target = dyn_cast<Expr>(Inner);
118class ExprPointeeResolve {
121 bool resolveExpr(
const Expr *
E) {
127 if (
const auto *BO = dyn_cast<BinaryOperator>(
E)) {
128 if (BO->isAdditiveOp())
129 return (resolveExpr(BO->getLHS()) || resolveExpr(BO->getRHS()));
131 return resolveExpr(BO->getRHS());
135 if (
const auto *PE = dyn_cast<ParenExpr>(
E))
136 return resolveExpr(PE->getSubExpr());
138 if (
const auto *ICE = dyn_cast<ImplicitCastExpr>(
E)) {
142 if (kind == CK_LValueToRValue || kind == CK_DerivedToBase ||
143 kind == CK_UncheckedDerivedToBase)
144 return resolveExpr(ICE->getSubExpr());
148 if (
const auto *ACE = dyn_cast<AbstractConditionalOperator>(
E))
149 return resolve(ACE->getTrueExpr()) || resolve(ACE->getFalseExpr());
155 ExprPointeeResolve(
const Expr *T) :
T(
T) {}
156 bool resolve(
const Expr *S) {
return resolveExpr(S); }
160 auto *Exp = dyn_cast<Expr>(&
Node);
163 auto *
Target = dyn_cast<Expr>(
T);
166 return ExprPointeeResolve{
Target}.resolve(Exp);
171AST_MATCHER_P(InitListExpr, hasAnyInit, ast_matchers::internal::Matcher<Expr>,
173 for (
const Expr *Arg :
Node.inits()) {
176 ast_matchers::internal::BoundNodesTreeBuilder
Result(*Builder);
177 if (InnerMatcher.matches(*Arg, Finder, &
Result)) {
178 *Builder = std::move(
Result);
185const ast_matchers::internal::VariadicDynCastAllOfMatcher<Stmt, CXXTypeidExpr>
188AST_MATCHER(CXXTypeidExpr, isPotentiallyEvaluated) {
189 return Node.isPotentiallyEvaluated();
193 const Decl *CalleeDecl =
Node.getCalleeDecl();
194 const auto *VD = dyn_cast_or_null<ValueDecl>(CalleeDecl);
197 const QualType
T = VD->getType().getCanonicalType();
198 const auto *MPT = dyn_cast<MemberPointerType>(
T);
199 const auto *FPT = MPT ? cast<FunctionProtoType>(MPT->getPointeeType())
200 : dyn_cast<FunctionProtoType>(
T);
203 return FPT->isConst();
207 ast_matchers::internal::Matcher<Expr>, InnerMatcher) {
208 if (
Node.isTypePredicate())
210 return InnerMatcher.matches(*
Node.getControllingExpr(), Finder, Builder);
214ast_matchers::internal::Matcher<T>
215findFirst(
const ast_matchers::internal::Matcher<T> &Matcher) {
219const auto nonConstReferenceType = [] {
220 return hasUnqualifiedDesugaredType(
224const auto nonConstPointerType = [] {
225 return hasUnqualifiedDesugaredType(
229const auto isMoveOnly = [] {
239template <
class T>
struct NodeID;
240template <>
struct NodeID<Expr> {
static constexpr StringRef value =
"expr"; };
241template <>
struct NodeID<
Decl> {
static constexpr StringRef value =
"decl"; };
242constexpr StringRef NodeID<Expr>::value;
243constexpr StringRef NodeID<Decl>::value;
246 class F =
const Stmt *(ExprMutationAnalyzer::Analyzer::*)(
const T *)>
247const Stmt *tryEachMatch(ArrayRef<ast_matchers::BoundNodes> Matches,
248 ExprMutationAnalyzer::Analyzer *Analyzer, F Finder) {
249 const StringRef
ID = NodeID<T>::value;
250 for (
const auto &
Nodes : Matches) {
251 if (
const Stmt *S = (Analyzer->*Finder)(
Nodes.getNodeAs<
T>(
ID)))
260 return findMutationMemoized(
262 {&ExprMutationAnalyzer::Analyzer::findDirectMutation,
263 &ExprMutationAnalyzer::Analyzer::findMemberMutation,
264 &ExprMutationAnalyzer::Analyzer::findArrayElementMutation,
265 &ExprMutationAnalyzer::Analyzer::findCastMutation,
266 &ExprMutationAnalyzer::Analyzer::findRangeLoopMutation,
267 &ExprMutationAnalyzer::Analyzer::findReferenceMutation,
268 &ExprMutationAnalyzer::Analyzer::findFunctionArgMutation},
278 return findMutationMemoized(
281 &ExprMutationAnalyzer::Analyzer::findPointeeValueMutation,
282 &ExprMutationAnalyzer::Analyzer::findPointeeMemberMutation,
283 &ExprMutationAnalyzer::Analyzer::findPointeeToNonConst,
290 return tryEachDeclRef(Dec,
294const Stmt *ExprMutationAnalyzer::Analyzer::findMutationMemoized(
298 auto [
Memoized, Inserted] = MemoizedResults.try_emplace(Exp);
305 for (
const auto &Finder : Finders) {
306 if (
const Stmt *S = (this->*Finder)(Exp))
307 return MemoizedResults[Exp] = S;
314ExprMutationAnalyzer::Analyzer::tryEachDeclRef(
const Decl *Dec,
315 MutationFinder Finder) {
316 const auto Refs =
match(
320 anyOf(equalsNode(Dec),
324 .bind(NodeID<Expr>::value)),
326 for (
const auto &RefNodes : Refs) {
327 const auto *
E = RefNodes.getNodeAs<Expr>(NodeID<Expr>::value);
328 if ((this->*Finder)(
E))
349 cxxTypeidExpr(
unless(isPotentiallyEvaluated())),
361 return tryEachMatch<Expr>(Matches,
this,
366ExprMutationAnalyzer::Analyzer::findDeclMutation(ArrayRef<BoundNodes> Matches) {
367 return tryEachMatch<Decl>(Matches,
this,
371const Stmt *ExprMutationAnalyzer::Analyzer::findExprPointeeMutation(
372 ArrayRef<ast_matchers::BoundNodes> Matches) {
373 return tryEachMatch<Expr>(
377const Stmt *ExprMutationAnalyzer::Analyzer::findDeclPointeeMutation(
378 ArrayRef<ast_matchers::BoundNodes> Matches) {
379 return tryEachMatch<Decl>(
384ExprMutationAnalyzer::Analyzer::findDirectMutation(
const Expr *Exp) {
386 const auto AsAssignmentLhs =
387 binaryOperator(isAssignmentOperator(), hasLHS(canResolveToExpr(Exp)));
390 const auto AsIncDecOperand =
392 hasUnaryOperand(canResolveToExpr(Exp)));
401 hasArgument(0, canResolveToExpr(Exp))),
407 hasEitherOperand(ignoringImpCasts(canResolveToExpr(Exp)))),
411 cxxFoldExpr(hasFoldInit(ignoringImpCasts(canResolveToExpr(Exp)))),
418 hasObjectExpression(canResolveToExpr(Exp))))))),
424 hasObjectExpression(canResolveToExpr(Exp))))))));
430 const auto AsAmpersandOperand =
434 hasUnaryOperand(canResolveToExpr(Exp)));
435 const auto AsPointerFromArrayDecay =
castExpr(
436 hasCastKind(CK_ArrayToPointerDecay),
444 cxxMethodDecl(ofClass(isMoveOnly()), returns(nonConstPointerType()))),
445 argumentCountIs(1), hasArgument(0, canResolveToExpr(Exp)));
452 const auto NonConstRefParam = forEachArgumentWithParamType(
453 anyOf(canResolveToExpr(Exp),
455 hasObjectExpression(ignoringImpCasts(canResolveToExpr(Exp))))),
456 nonConstReferenceType());
459 const auto AsNonConstRefArg =
465 callExpr(isTypeDependent(), hasAnyArgument(canResolveToExpr(Exp))),
481 const auto AsLambdaRefCaptureInit =
lambdaExpr(hasCaptureInit(Exp));
488 const auto AsNonConstRefReturn =
489 returnStmt(hasReturnValue(canResolveToExpr(Exp)));
493 allOf(canResolveToExpr(Exp), hasType(nonConstReferenceType())))));
495 const auto Matches =
match(
498 findFirst(
stmt(
anyOf(AsAssignmentLhs, AsIncDecOperand, AsNonConstThis,
499 AsAmpersandOperand, AsPointerFromArrayDecay,
500 AsOperatorArrowThis, AsNonConstRefArg,
501 AsLambdaRefCaptureInit, AsNonConstRefReturn,
502 AsNonConstRefRangeInit))
505 return selectFirst<Stmt>(
"stmt", Matches);
509ExprMutationAnalyzer::Analyzer::findMemberMutation(
const Expr *Exp) {
511 const auto MemberExprs =
match(
514 hasObjectExpression(canResolveToExpr(Exp))),
516 hasLHS(equalsNode(Exp)))))
517 .bind(NodeID<Expr>::value)),
519 return findExprMutation(MemberExprs);
523ExprMutationAnalyzer::Analyzer::findArrayElementMutation(
const Expr *Exp) {
525 const auto SubscriptExprs =
match(
527 anyOf(hasBaseConservative(canResolveToExpr(Exp)),
529 hasCastKind(CK_ArrayToPointerDecay),
530 hasSourceExpression(canResolveToExpr(Exp)))))))
531 .bind(NodeID<Expr>::value)),
533 return findExprMutation(SubscriptExprs);
536const Stmt *ExprMutationAnalyzer::Analyzer::findCastMutation(
const Expr *Exp) {
539 const auto ExplicitCast =
542 nonConstReferenceType()))))
546 if (
const auto *CastStmt = selectFirst<Stmt>(
"stmt", ExplicitCast))
550 const auto Casts =
match(
553 nonConstReferenceType())),
555 nonConstReferenceType())))))
556 .bind(NodeID<Expr>::value)),
559 if (
const Stmt *S = findExprMutation(Casts))
564 hasAnyName(
"::std::move",
"::std::forward"))),
565 hasArgument(0, canResolveToExpr(Exp)))
568 return findExprMutation(Calls);
572ExprMutationAnalyzer::Analyzer::findRangeLoopMutation(
const Expr *Exp) {
579 const auto DeclStmtToNonRefToArray =
declStmt(hasSingleDecl(
varDecl(hasType(
581 const auto RefToArrayRefToElements =
match(
585 hasType(nonConstPointerType())))
586 .bind(NodeID<Decl>::value)),
587 hasRangeStmt(DeclStmtToNonRefToArray),
588 hasRangeInit(canResolveToExpr(Exp))))
592 if (
const auto *BadRangeInitFromArray =
593 selectFirst<Stmt>(
"stmt", RefToArrayRefToElements))
594 return BadRangeInitFromArray;
603 const auto HasAnyNonConstIterator =
609 const auto DeclStmtToNonConstIteratorContainer =
declStmt(
613 const auto RefToContainerBadIterators =
match(
615 hasRangeStmt(DeclStmtToNonConstIteratorContainer),
616 hasRangeInit(canResolveToExpr(Exp)))))
620 if (
const auto *BadIteratorsContainer =
621 selectFirst<Stmt>(
"stmt", RefToContainerBadIterators))
622 return BadIteratorsContainer;
626 const auto LoopVars =
628 hasLoopVariable(
varDecl(hasType(nonConstReferenceType()))
629 .bind(NodeID<Decl>::value)),
630 hasRangeInit(canResolveToExpr(Exp)))),
632 return findDeclMutation(LoopVars);
636ExprMutationAnalyzer::Analyzer::findReferenceMutation(
const Expr *Exp) {
640 const auto Ref =
match(
644 returns(nonConstReferenceType()))),
645 argumentCountIs(1), hasArgument(0, canResolveToExpr(Exp)))
646 .bind(NodeID<Expr>::value)),
648 if (
const Stmt *S = findExprMutation(Ref))
652 const auto Refs =
match(
654 varDecl(hasType(nonConstReferenceType()),
655 hasInitializer(
anyOf(
656 canResolveToExpr(Exp),
657 memberExpr(hasObjectExpression(canResolveToExpr(Exp))))),
662 hasRangeStmt(equalsBoundNode(
"stmt"))))))))
663 .bind(NodeID<Decl>::value))),
665 return findDeclMutation(Refs);
669ExprMutationAnalyzer::Analyzer::findFunctionArgMutation(
const Expr *Exp) {
670 const auto NonConstRefParam = forEachArgumentWithParam(
671 canResolveToExpr(Exp),
672 parmVarDecl(hasType(nonConstReferenceType())).bind(
"parm"));
675 const auto Matches =
match(
681 "::std::move",
"::std::forward"))))),
684 .bind(NodeID<Expr>::value))),
686 for (
const auto &
Nodes : Matches) {
687 const auto *Exp =
Nodes.getNodeAs<Expr>(NodeID<Expr>::value);
688 const auto *
Func =
Nodes.getNodeAs<FunctionDecl>(
"func");
689 if (!
Func->getBody() || !
Func->getPrimaryTemplate())
692 const auto *Parm =
Nodes.getNodeAs<ParmVarDecl>(
"parm");
693 const ArrayRef<ParmVarDecl *> AllParams =
694 Func->getPrimaryTemplate()->getTemplatedDecl()->parameters();
696 AllParams[std::min<size_t>(Parm->getFunctionScopeIndex(),
697 AllParams.size() - 1)]
699 if (
const auto *
T = ParmType->
getAs<PackExpansionType>())
700 ParmType =
T->getPattern();
704 if (
const auto *RefType = ParmType->getAs<RValueReferenceType>()) {
705 if (!RefType->getPointeeType().getQualifiers() &&
706 isa<TemplateTypeParmType>(
707 RefType->getPointeeType().getCanonicalType())) {
710 *
Func, Context, Memorized);
711 if (Analyzer->findMutation(Parm))
723ExprMutationAnalyzer::Analyzer::findPointeeValueMutation(
const Expr *Exp) {
724 const auto Matches =
match(
729 hasUnaryOperand(canResolveToExprPointee(Exp))),
732 hasBaseConservative(canResolveToExprPointee(Exp)))))
733 .bind(NodeID<Expr>::value))),
735 return findExprMutation(Matches);
739ExprMutationAnalyzer::Analyzer::findPointeeMemberMutation(
const Expr *Exp) {
740 const Stmt *MemberCallExpr = selectFirst<Stmt>(
747 return MemberCallExpr;
750 memberExpr(hasObjectExpression(canResolveToExprPointee(Exp)))
751 .bind(NodeID<Expr>::value))),
753 return findExprMutation(Matches);
757ExprMutationAnalyzer::Analyzer::findPointeeToNonConst(
const Expr *Exp) {
758 const auto NonConstPointerOrDependentType =
759 type(
anyOf(nonConstPointerType(), isDependentType()));
762 const auto InitToNonConst =
763 varDecl(hasType(NonConstPointerOrDependentType),
764 hasInitializer(
expr(canResolveToExprPointee(Exp)).bind(
"stmt")));
765 const auto AssignToNonConst =
767 hasLHS(
expr(hasType(NonConstPointerOrDependentType))),
768 hasRHS(canResolveToExprPointee(Exp)));
770 const auto ArgOfInstantiationDependent =
allOf(
771 hasAnyArgument(canResolveToExprPointee(Exp)), isInstantiationDependent());
772 const auto ArgOfNonConstParameter = forEachArgumentWithParamType(
773 canResolveToExprPointee(Exp), NonConstPointerOrDependentType);
774 const auto CallLikeMatcher =
775 anyOf(ArgOfNonConstParameter, ArgOfInstantiationDependent);
776 const auto PassAsNonConstArg =
780 initListExpr(hasAnyInit(canResolveToExprPointee(Exp)))));
782 const auto CastToNonConst =
784 hasDestinationType(NonConstPointerOrDependentType));
788 const auto CaptureNoConst =
lambdaExpr(hasCaptureInit(Exp));
792 stmt(
anyOf(AssignToNonConst, PassAsNonConstArg,
793 CastToNonConst, CaptureNoConst))
797 return selectFirst<Stmt>(
"stmt", Matches);
800FunctionParmMutationAnalyzer::FunctionParmMutationAnalyzer(
801 const FunctionDecl &
Func, ASTContext &Context,
802 ExprMutationAnalyzer::Memoized &Memorized)
803 : BodyAnalyzer(*
Func.getBody(), Context, Memorized) {
804 if (
const auto *Ctor = dyn_cast<CXXConstructorDecl>(&
Func)) {
807 for (
const CXXCtorInitializer *
Init : Ctor->inits()) {
808 ExprMutationAnalyzer::Analyzer InitAnalyzer(*
Init->getInit(), Context,
810 for (
const ParmVarDecl *Parm : Ctor->parameters()) {
811 if (Results.contains(Parm))
813 if (
const Stmt *S = InitAnalyzer.findMutation(Parm))
822 auto [Place, Inserted] = Results.try_emplace(Parm);
824 return Place->second;
BoundNodesTreeBuilder Nodes
#define AST_MATCHER(Type, DefineMatcher)
AST_MATCHER(Type, DefineMatcher) { ... } defines a zero parameter function named DefineMatcher() that...
#define AST_MATCHER_P(Type, DefineMatcher, ParamType, Param)
AST_MATCHER_P(Type, DefineMatcher, ParamType, Param) { ... } defines a single-parameter function name...
llvm::MachO::Target Target
Holds long-lived AST nodes (such as types and decls) that can be referred to throughout the semantic ...
Decl - This represents one declaration (or definition), e.g.
static bool isUnevaluated(const Stmt *Stm, ASTContext &Context)
check whether stmt is unevaluated.
friend class FunctionParmMutationAnalyzer
This represents one expression.
Expr * IgnoreParens() LLVM_READONLY
Skip past any parentheses which might surround this expression until reaching a fixed point.
static FunctionParmMutationAnalyzer * getFunctionParmMutationAnalyzer(const FunctionDecl &Func, ASTContext &Context, ExprMutationAnalyzer::Memoized &Memorized)
const Stmt * findMutation(const ParmVarDecl *Parm)
Represents a parameter to a function.
Stmt - This represents one statement.
const T * getAs() const
Member-template getAs<specific type>'.
const internal::VariadicDynCastAllOfMatcher< Decl, VarDecl > varDecl
Matches variable declarations.
const internal::VariadicDynCastAllOfMatcher< Stmt, DeclRefExpr > declRefExpr
Matches expressions that refer to declarations.
const internal::VariadicOperatorMatcherFunc< 1, 1 > unless
Matches if the provided matcher does not match.
const internal::VariadicDynCastAllOfMatcher< Stmt, ImplicitCastExpr > implicitCastExpr
Matches the implicit cast nodes of Clang's AST.
const internal::ArgumentAdaptingMatcherFunc< internal::HasDescendantMatcher > hasDescendant
Matches AST nodes that have descendant AST nodes that match the provided matcher.
const internal::VariadicDynCastAllOfMatcher< Stmt, CXXDependentScopeMemberExpr > cxxDependentScopeMemberExpr
Matches member expressions where the actual member referenced could not be resolved because the base ...
const AstTypeMatcher< PointerType > pointerType
Matches pointer types, but does not match Objective-C object pointer types.
const internal::VariadicDynCastAllOfMatcher< Decl, BindingDecl > bindingDecl
Matches binding declarations Example matches foo and bar (matcher = bindingDecl()
const internal::VariadicDynCastAllOfMatcher< Decl, ParmVarDecl > parmVarDecl
Matches parameter variable declarations.
const internal::VariadicDynCastAllOfMatcher< Stmt, GenericSelectionExpr > genericSelectionExpr
Matches C11 _Generic expression.
const internal::VariadicDynCastAllOfMatcher< Stmt, ReturnStmt > returnStmt
Matches return statements.
internal::Matcher< NamedDecl > hasName(StringRef Name)
Matches NamedDecl nodes that have the specified name.
const internal::VariadicDynCastAllOfMatcher< Stmt, CallExpr > callExpr
Matches call expressions.
const internal::VariadicDynCastAllOfMatcher< Stmt, LambdaExpr > lambdaExpr
Matches lambda expressions.
const AstTypeMatcher< VariableArrayType > variableArrayType
Matches C arrays with a specified size that is not an integer-constant-expression.
const internal::VariadicDynCastAllOfMatcher< Stmt, UnaryExprOrTypeTraitExpr > unaryExprOrTypeTraitExpr
Matches sizeof (C99), alignof (C++11) and vec_step (OpenCL)
const internal::ArgumentAdaptingMatcherFunc< internal::ForEachDescendantMatcher > forEachDescendant
Matches AST nodes that have descendant AST nodes that match the provided matcher.
SmallVector< BoundNodes, 1 > match(MatcherT Matcher, const NodeT &Node, ASTContext &Context)
Returns the results of matching Matcher on Node.
const internal::VariadicDynCastAllOfMatcher< Decl, NamedDecl > namedDecl
Matches a declaration of anything that could have a name.
const internal::VariadicAllOfMatcher< Type > type
Matches Types in the clang AST.
const internal::VariadicAllOfMatcher< TypeLoc > typeLoc
Matches TypeLocs in the clang AST.
const internal::VariadicDynCastAllOfMatcher< Stmt, ParenListExpr > parenListExpr
Matches paren list expressions.
const AstTypeMatcher< ArrayType > arrayType
Matches all kinds of arrays.
const internal::VariadicDynCastAllOfMatcher< Stmt, UnaryOperator > unaryOperator
Matches unary operator expressions.
const internal::VariadicFunction< internal::Matcher< NamedDecl >, StringRef, internal::hasAnyNameFunc > hasAnyName
Matches NamedDecl nodes that have any of the specified names.
const internal::MapAnyOfMatcher< BinaryOperator, CXXOperatorCallExpr, CXXRewrittenBinaryOperator > binaryOperation
Matches nodes which can be used with binary operators.
const internal::VariadicDynCastAllOfMatcher< Stmt, ArraySubscriptExpr > arraySubscriptExpr
Matches array subscript expressions.
const internal::VariadicDynCastAllOfMatcher< Stmt, CXXForRangeStmt > cxxForRangeStmt
Matches range-based for statements.
const internal::VariadicDynCastAllOfMatcher< Stmt, CXXMemberCallExpr > cxxMemberCallExpr
Matches member call expressions.
const internal::VariadicDynCastAllOfMatcher< Decl, CXXConstructorDecl > cxxConstructorDecl
Matches C++ constructor declarations.
internal::BindableMatcher< Stmt > sizeOfExpr(const internal::Matcher< UnaryExprOrTypeTraitExpr > &InnerMatcher)
Same as unaryExprOrTypeTraitExpr, but only matching sizeof.
const internal::VariadicDynCastAllOfMatcher< Stmt, InitListExpr > initListExpr
Matches init list expressions.
const internal::VariadicDynCastAllOfMatcher< Stmt, CXXNoexceptExpr > cxxNoexceptExpr
Matches noexcept expressions.
const internal::VariadicDynCastAllOfMatcher< Stmt, BinaryOperator > binaryOperator
Matches binary operator expressions.
const internal::ArgumentAdaptingMatcherFunc< internal::HasMatcher > has
Matches AST nodes that have child AST nodes that match the provided matcher.
const internal::VariadicDynCastAllOfMatcher< Stmt, ExplicitCastExpr > explicitCastExpr
Matches explicit cast expressions.
const internal::VariadicDynCastAllOfMatcher< Stmt, CXXConstructExpr > cxxConstructExpr
Matches constructor call expressions (including implicit ones).
const internal::VariadicDynCastAllOfMatcher< Stmt, CXXOperatorCallExpr > cxxOperatorCallExpr
Matches overloaded operator calls.
internal::PolymorphicMatcher< internal::HasOverloadedOperatorNameMatcher, AST_POLYMORPHIC_SUPPORTED_TYPES(CXXOperatorCallExpr, FunctionDecl), std::vector< std::string > > hasOverloadedOperatorName(StringRef Name)
Matches overloaded operator names.
const internal::VariadicOperatorMatcherFunc< 2, std::numeric_limits< unsigned >::max()> allOf
Matches if all given matchers match.
const internal::VariadicDynCastAllOfMatcher< Decl, FunctionDecl > functionDecl
Matches function declarations.
const internal::VariadicDynCastAllOfMatcher< Stmt, UnresolvedMemberExpr > unresolvedMemberExpr
Matches unresolved member expressions.
const internal::VariadicDynCastAllOfMatcher< Stmt, MemberExpr > memberExpr
Matches member expressions.
const internal::VariadicDynCastAllOfMatcher< Decl, CXXRecordDecl > cxxRecordDecl
Matches C++ class declarations.
internal::Matcher< T > traverse(TraversalKind TK, const internal::Matcher< T > &InnerMatcher)
Causes all nested matchers to be matched with the specified traversal kind.
const AstTypeMatcher< ReferenceType > referenceType
Matches both lvalue and rvalue reference types.
const internal::VariadicDynCastAllOfMatcher< Stmt, CXXUnresolvedConstructExpr > cxxUnresolvedConstructExpr
Matches unresolved constructor call expressions.
internal::Matcher< T > findAll(const internal::Matcher< T > &Matcher)
Matches if the node or any descendant matches.
internal::PolymorphicMatcher< internal::HasDeclarationMatcher, void(internal::HasDeclarationSupportedTypes), internal::Matcher< Decl > > hasDeclaration(const internal::Matcher< Decl > &InnerMatcher)
Matches a node if the declaration associated with that node matches the given matcher.
const internal::VariadicDynCastAllOfMatcher< Stmt, DeclStmt > declStmt
Matches declaration statements.
const internal::VariadicAllOfMatcher< Stmt > stmt
Matches statements.
const internal::VariadicDynCastAllOfMatcher< Stmt, Expr > expr
Matches expressions.
const internal::VariadicDynCastAllOfMatcher< Stmt, CXXFoldExpr > cxxFoldExpr
Matches C++17 fold expressions.
const internal::VariadicOperatorMatcherFunc< 2, std::numeric_limits< unsigned >::max()> anyOf
Matches if any of the given matchers matches.
const internal::VariadicDynCastAllOfMatcher< Decl, CXXMethodDecl > cxxMethodDecl
Matches method declarations.
const internal::VariadicDynCastAllOfMatcher< Stmt, CastExpr > castExpr
Matches any cast nodes of Clang's AST.
const internal::ArgumentAdaptingMatcherFunc< internal::HasAncestorMatcher, internal::TypeList< Decl, NestedNameSpecifierLoc, Stmt, TypeLoc, Attr >, internal::TypeList< Decl, NestedNameSpecifierLoc, Stmt, TypeLoc, Attr > > hasAncestor
Matches AST nodes that have an ancestor that matches the provided matcher.
const internal::ArgumentAdaptingMatcherFunc< internal::HasParentMatcher, internal::TypeList< Decl, NestedNameSpecifierLoc, Stmt, TypeLoc, Attr >, internal::TypeList< Decl, NestedNameSpecifierLoc, Stmt, TypeLoc, Attr > > hasParent
Matches AST nodes that have a parent that matches the provided matcher.
unsigned kind
All of the diagnostics that can be emitted by the frontend.
std::variant< struct RequiresDecl, struct HeaderDecl, struct UmbrellaDirDecl, struct ModuleDecl, struct ExcludeDecl, struct ExportDecl, struct ExportAsDecl, struct ExternModuleDecl, struct UseDecl, struct LinkDecl, struct ConfigMacrosDecl, struct ConflictDecl > Decl
All declarations that can appear in a module declaration.
The JSON file list parser is used to communicate input to InstallAPI.
@ TK_AsIs
Will traverse all child nodes.
@ Result
The result type of a method or function.
CastKind
CastKind - The kind of operation required for a conversion.
const FunctionProtoType * T
static bool canExprResolveTo(const Expr *Source, const Expr *Target)
const Stmt * findPointeeMutation(const Expr *Exp)
const Stmt * findMutation(const Expr *Exp)
llvm::DenseMap< const Expr *, const Stmt * > ResultMap