================ @@ -0,0 +1,117 @@ +//===--- CoroutineHostileRAII.cpp - clang-tidy ----------------------------===// +// +// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. +// See https://llvm.org/LICENSE.txt for license information. +// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception +// +//===----------------------------------------------------------------------===// + +#include "CoroutineHostileRAIICheck.h" +#include "../utils/OptionsUtils.h" +#include "clang/AST/Attr.h" +#include "clang/AST/Decl.h" +#include "clang/AST/ExprCXX.h" +#include "clang/AST/Stmt.h" +#include "clang/AST/Type.h" +#include "clang/ASTMatchers/ASTMatchFinder.h" +#include "clang/ASTMatchers/ASTMatchers.h" +#include "clang/ASTMatchers/ASTMatchersInternal.h" +#include "clang/Basic/AttrKinds.h" +#include "clang/Basic/DiagnosticIDs.h" + +using namespace clang::ast_matchers; +namespace clang::tidy::misc { +using ::clang::ast_matchers::internal::ASTMatchFinder; +using clang::ast_matchers::internal::BoundNodesTreeBuilder; + +// In case of a match, add the bindings as a separate match. Also don't clear +// the bindings if a match is not found (unlike Matcher::matches). +template <class Matcher, class Node> +bool match(Matcher M, Node &N, ASTMatchFinder *Finder, + BoundNodesTreeBuilder *Builder) { + BoundNodesTreeBuilder LocalBuilder; + if (M.matches(N, Finder, &LocalBuilder)) { + Builder->addMatch(LocalBuilder); + return true; + } + return false; +} + +CoroutineHostileRAIICheck::CoroutineHostileRAIICheck(StringRef Name, + ClangTidyContext *Context) + : ClangTidyCheck(Name, Context), + RAIITypesList(utils::options::parseStringList( + Options.get("RAIITypesList", "std::lock_guard;std::scoped_lock"))) {} + +AST_MATCHER_P(VarDecl, isHostileRAII, std::vector<StringRef>, RAIITypesList) { + return match(varDecl(hasType(hasCanonicalType(hasDeclaration( + hasAttr(attr::Kind::ScopedLockable))))) + .bind("scoped-lockable"), + Node, Finder, Builder) || + match(varDecl(hasType(hasCanonicalType(hasDeclaration( + namedDecl(hasAnyName(RAIITypesList)))))) + .bind("raii"), + Node, Finder, Builder); +} + +AST_MATCHER_P(DeclStmt, hasHostileRAII, std::vector<StringRef>, RAIITypesList) { + bool IsHostile = false; + for (Decl *D : Node.decls()) + IsHostile |= + match(varDecl(isHostileRAII(RAIITypesList)), *D, Finder, Builder); + return IsHostile; +} + +AST_MATCHER_P(Expr, isHostileSuspension, std::vector<StringRef>, ---------------- PiotrZSL wrote:
so final call would look like: ``` coawaitExpr(forEachPrevDeclStmt(forEach(varDecl(isHostileRAII)))) ``` Main thing is that this matcher should be re-usable, simply there shoudn't be a need for a match function. https://github.com/llvm/llvm-project/pull/68738 _______________________________________________ cfe-commits mailing list cfe-commits@lists.llvm.org https://lists.llvm.org/cgi-bin/mailman/listinfo/cfe-commits