10#include "../utils/LexerUtils.h"
11#include "../utils/Matchers.h"
12#include "clang/AST/ASTContext.h"
13#include "clang/ASTMatchers/ASTMatchFinder.h"
14#include "clang/Lex/Lexer.h"
24 const LangOptions &LangOpts) {
25 const Token PrevToken =
27 if (PrevToken.is(tok::unknown))
30 if (PrevToken.isNot(tok::equal))
31 return {PrevToken.getEndLoc(),
Range.getEnd()};
34 {PrevToken.getLocation(),
Range.getEnd()}, SM, LangOpts);
39 IgnoreBaseInCopyConstructors);
43 auto ConstructorMatcher =
46 hasDeclaration(cxxConstructorDecl(
47 ofClass(cxxRecordDecl(unless(isTriviallyDefaultConstructible()))
51 auto HasUnionAsParent = hasParent(recordDecl(isUnion()));
53 auto HasTypeEqualToConstructorClass = hasType(qualType(
54 hasCanonicalType(qualType(hasDeclaration(equalsBoundNode(
"class"))))));
58 unless(isDelegatingConstructor()), ofClass(unless(isUnion())),
59 forEachConstructorInitializer(
61 withInitializer(ConstructorMatcher),
62 anyOf(isBaseInitializer(),
63 forField(fieldDecl(unless(hasType(isConstQualified())),
64 unless(HasUnionAsParent),
65 HasTypeEqualToConstructorClass))))
70 Finder->addMatcher(fieldDecl(hasInClassInitializer(ConstructorMatcher),
71 HasTypeEqualToConstructorClass,
72 unless(HasUnionAsParent))
78 const auto *Construct = Result.Nodes.getNodeAs<CXXConstructExpr>(
"construct");
80 if (
const auto *
Field = Result.Nodes.getNodeAs<FieldDecl>(
"field")) {
81 const Expr *Init =
Field->getInClassInitializer();
82 diag(Construct->getExprLoc(),
"initializer for member %0 is redundant")
85 Init->getSourceRange(), *Result.SourceManager,
getLangOpts()));
89 const auto *Init = Result.Nodes.getNodeAs<CXXCtorInitializer>(
"init");
90 const auto *ConstructorDecl =
91 Result.Nodes.getNodeAs<CXXConstructorDecl>(
"constructor");
93 if (IgnoreBaseInCopyConstructors && ConstructorDecl->isCopyConstructor() &&
94 Init->isBaseInitializer())
97 if (Init->isAnyMemberInitializer()) {
98 diag(Init->getSourceLocation(),
"initializer for member %0 is redundant")
99 << Init->getAnyMember()
100 << FixItHint::CreateRemoval(Init->getSourceRange());
102 diag(Init->getSourceLocation(),
103 "initializer for base class %0 is redundant")
104 << Construct->getType()
105 << FixItHint::CreateRemoval(Init->getSourceRange());
CharSourceRange Range
SourceRange for the file 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.
DiagnosticBuilder diag(SourceLocation Loc, StringRef Description, DiagnosticIDs::Level Level=DiagnosticIDs::Warning)
Add a diagnostic with the check's name.
const LangOptions & getLangOpts() const
Returns the language options from the 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...
void registerMatchers(ast_matchers::MatchFinder *Finder) override
Override this to register AST matchers with Finder.
static SourceRange getFullInitRangeInclWhitespaces(SourceRange Range, const SourceManager &SM, const LangOptions &LangOpts)
Token getPreviousToken(SourceLocation Location, const SourceManager &SM, const LangOptions &LangOpts, bool SkipComments)
Returns previous token or tok::unknown if not found.
llvm::StringMap< ClangTidyValue > OptionMap