10#include "clang/AST/ASTContext.h"
11#include "clang/ASTMatchers/ASTMatchFinder.h"
12#include "clang/ASTMatchers/ASTMatchers.h"
21 return Node.getBeginLoc().isValid();
25 clang::ast_matchers::internal::Matcher<clang::Type>,
27 const clang::Type *TypeNode =
Node.getTypePtr();
28 return TypeNode !=
nullptr &&
29 InnerMatcher.matches(*TypeNode, Finder,
Builder);
33 return Node.isExternCContext();
37 const clang::DeclContext *DC =
Node.getDeclContext();
38 const auto *FD = llvm::dyn_cast<clang::FunctionDecl>(DC);
39 return FD ? FD->isMain() :
false;
46 AllowStringArrays(Options.get(
"AllowStringArrays", false)) {}
49 Options.
store(Opts,
"AllowStringArrays", AllowStringArrays);
53 ast_matchers::internal::Matcher<TypeLoc> IgnoreStringArrayIfNeededMatcher =
55 if (AllowStringArrays)
56 IgnoreStringArrayIfNeededMatcher =
57 unless(typeLoc(loc(hasCanonicalType(incompleteArrayType(
58 hasElementType(isAnyCharacter())))),
59 hasParent(varDecl(hasInitializer(stringLiteral()),
60 unless(parmVarDecl())))));
63 typeLoc(hasValidBeginLoc(), hasType(arrayType()),
64 optionally(hasParent(parmVarDecl().bind(
"param_decl"))),
65 unless(anyOf(hasParent(parmVarDecl(isArgvOfMain())),
66 hasParent(varDecl(isExternC())),
68 hasParent(recordDecl(isExternCContext())))),
69 hasAncestor(functionDecl(isExternC())))),
70 std::move(IgnoreStringArrayIfNeededMatcher))
76 const auto *ArrayType = Result.Nodes.getNodeAs<TypeLoc>(
"typeloc");
77 const bool IsInParam =
78 Result.Nodes.getNodeAs<ParmVarDecl>(
"param_decl") !=
nullptr;
79 const bool IsVLA = ArrayType->getTypePtr()->isVariableArrayType();
80 enum class RecommendType { Array, Vector, Span };
81 llvm::SmallVector<const char *> RecommendTypes{};
83 RecommendTypes.push_back(
"'std::vector'");
84 }
else if (ArrayType->getTypePtr()->isIncompleteArrayType() && IsInParam) {
87 if (Result.Context->getLangOpts().CPlusPlus20)
88 RecommendTypes.push_back(
"'std::span'");
90 RecommendTypes.push_back(
"'std::array'");
91 RecommendTypes.push_back(
"'std::vector'");
94 RecommendTypes.push_back(
"'std::array'");
96 diag(ArrayType->getBeginLoc(),
97 "do not declare %select{C-style|C VLA}0 arrays, use %1 instead")
98 << IsVLA << llvm::join(RecommendTypes,
" or ");
llvm::SmallString< 256U > Name
CodeCompletionBuilder Builder
::clang::DynTypedNode Node
void store(ClangTidyOptions::OptionMap &Options, StringRef LocalName, StringRef Value) const
Stores an option with the check-local name LocalName with string value Value to Options.
Base class for all clang-tidy checks.
DiagnosticBuilder diag(SourceLocation Loc, StringRef Description, DiagnosticIDs::Level Level=DiagnosticIDs::Warning)
Add a diagnostic with the check's name.
Every ClangTidyCheck reports errors through a DiagnosticsEngine provided by this context.
void registerMatchers(ast_matchers::MatchFinder *Finder) override
Override this to register AST matchers with Finder.
AvoidCArraysCheck(StringRef Name, ClangTidyContext *Context)
void check(const ast_matchers::MatchFinder::MatchResult &Result) override
ClangTidyChecks that register ASTMatchers should do the actual work in here.
void storeOptions(ClangTidyOptions::OptionMap &Opts) override
Should store all options supported by this check with their current values or default values for opti...
AST_MATCHER(Expr, isMacroID)
AST_MATCHER_P(Expr, lengthExprForStringNode, std::string, ID)
llvm::StringMap< ClangTidyValue > OptionMap