10#include "clang/AST/ASTContext.h"
11#include "clang/ASTMatchers/ASTMatchFinder.h"
25 StrictMode(Options.getLocalOrGlobal(
"StrictMode", false)) {}
28 MatchFinder *Finder) {
29 const auto BasicStringType =
30 hasType(qualType(hasUnqualifiedDesugaredType(recordType(
31 hasDeclaration(cxxRecordDecl(hasName(
"::std::basic_string")))))));
33 const auto BasicStringPlusOperator = cxxOperatorCallExpr(
34 hasOverloadedOperatorName(
"+"),
35 hasAnyArgument(ignoringImpCasts(declRefExpr(BasicStringType))));
37 const auto PlusOperator =
39 hasOverloadedOperatorName(
"+"),
40 hasAnyArgument(ignoringImpCasts(declRefExpr(BasicStringType))),
41 hasDescendant(BasicStringPlusOperator))
42 .bind(
"plusOperator");
44 const auto AssignOperator = cxxOperatorCallExpr(
45 hasOverloadedOperatorName(
"="),
46 hasArgument(0, declRefExpr(BasicStringType,
47 hasDeclaration(decl().bind(
"lhsStrT")))
49 hasArgument(1, stmt(hasDescendant(declRefExpr(
50 hasDeclaration(decl(equalsBoundNode(
"lhsStrT"))))))),
51 hasDescendant(BasicStringPlusOperator));
54 Finder->addMatcher(cxxOperatorCallExpr(anyOf(AssignOperator, PlusOperator)),
58 cxxOperatorCallExpr(anyOf(AssignOperator, PlusOperator),
59 hasAncestor(stmt(anyOf(cxxForRangeStmt(),
60 whileStmt(), forStmt())))),
66 const MatchFinder::MatchResult &Result) {
67 const auto *LhsStr = Result.Nodes.getNodeAs<DeclRefExpr>(
"lhsStr");
68 const auto *PlusOperator =
69 Result.Nodes.getNodeAs<CXXOperatorCallExpr>(
"plusOperator");
71 "string concatenation results in allocation of unnecessary temporary "
72 "strings; consider using 'operator+=' or 'string::append()' instead";
75 diag(LhsStr->getExprLoc(), DiagMsg);
76 else if (PlusOperator)
77 diag(PlusOperator->getExprLoc(), DiagMsg);
llvm::SmallString< 256U > Name
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.
llvm::StringMap< ClangTidyValue > OptionMap