1 //===--- InitVariablesCheck.cpp - clang-tidy ------------------------------===//
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
7 //===----------------------------------------------------------------------===//
9 #include "InitVariablesCheck.h"
11 #include "clang/AST/ASTContext.h"
12 #include "clang/ASTMatchers/ASTMatchFinder.h"
13 #include "clang/Lex/PPCallbacks.h"
14 #include "clang/Lex/Preprocessor.h"
17 using namespace clang::ast_matchers
;
19 namespace clang::tidy::cppcoreguidelines
{
22 AST_MATCHER(VarDecl
, isLocalVarDecl
) { return Node
.isLocalVarDecl(); }
25 InitVariablesCheck::InitVariablesCheck(StringRef Name
,
26 ClangTidyContext
*Context
)
27 : ClangTidyCheck(Name
, Context
),
28 IncludeInserter(Options
.getLocalOrGlobal("IncludeStyle",
29 utils::IncludeSorter::IS_LLVM
),
30 areDiagsSelfContained()),
31 MathHeader(Options
.get("MathHeader", "<math.h>")) {}
33 void InitVariablesCheck::storeOptions(ClangTidyOptions::OptionMap
&Opts
) {
34 Options
.store(Opts
, "IncludeStyle", IncludeInserter
.getStyle());
35 Options
.store(Opts
, "MathHeader", MathHeader
);
38 void InitVariablesCheck::registerMatchers(MatchFinder
*Finder
) {
39 std::string BadDecl
= "badDecl";
41 varDecl(unless(hasInitializer(anything())), unless(isInstantiated()),
42 isLocalVarDecl(), unless(isStaticLocal()), isDefinition(),
43 unless(hasParent(cxxCatchStmt())),
44 optionally(hasParent(declStmt(hasParent(
45 cxxForRangeStmt(hasLoopVariable(varDecl().bind(BadDecl
))))))),
46 unless(equalsBoundNode(BadDecl
)))
51 void InitVariablesCheck::registerPPCallbacks(const SourceManager
&SM
,
53 Preprocessor
*ModuleExpanderPP
) {
54 IncludeInserter
.registerPreprocessor(PP
);
57 void InitVariablesCheck::check(const MatchFinder::MatchResult
&Result
) {
58 const auto *MatchedDecl
= Result
.Nodes
.getNodeAs
<VarDecl
>("vardecl");
59 const ASTContext
&Context
= *Result
.Context
;
60 const SourceManager
&Source
= Context
.getSourceManager();
62 // Clang diagnostic error may cause the variable to be an invalid int vardecl
63 if (MatchedDecl
->isInvalidDecl())
66 // We want to warn about cases where the type name
67 // comes from a macro like this:
69 // TYPENAME_FROM_MACRO var;
71 // but not if the entire declaration comes from
74 // DEFINE_SOME_VARIABLE();
76 // or if the definition comes from a macro like SWAP
77 // that uses an internal temporary variable.
79 // Thus check that the variable name does
80 // not come from a macro expansion.
81 if (MatchedDecl
->getEndLoc().isMacroID())
84 QualType TypePtr
= MatchedDecl
->getType();
85 std::optional
<const char *> InitializationString
;
86 bool AddMathInclude
= false;
88 if (TypePtr
->isEnumeralType())
89 InitializationString
= nullptr;
90 else if (TypePtr
->isBooleanType())
91 InitializationString
= " = false";
92 else if (TypePtr
->isIntegerType())
93 InitializationString
= " = 0";
94 else if (TypePtr
->isFloatingType()) {
95 InitializationString
= " = NAN";
96 AddMathInclude
= true;
97 } else if (TypePtr
->isAnyPointerType()) {
98 if (getLangOpts().CPlusPlus11
)
99 InitializationString
= " = nullptr";
101 InitializationString
= " = NULL";
104 if (InitializationString
) {
106 diag(MatchedDecl
->getLocation(), "variable %0 is not initialized")
108 if (*InitializationString
!= nullptr)
109 Diagnostic
<< FixItHint::CreateInsertion(
110 MatchedDecl
->getLocation().getLocWithOffset(
111 MatchedDecl
->getName().size()),
112 *InitializationString
);
113 if (AddMathInclude
) {
114 Diagnostic
<< IncludeInserter
.createIncludeInsertion(
115 Source
.getFileID(MatchedDecl
->getBeginLoc()), MathHeader
);
119 } // namespace clang::tidy::cppcoreguidelines