blob: 1e2184e324fc132652a6437ed5b6850c6d59a6c6 [file] [log] [blame]
Yuanfang Chenf24c1e62019-08-21 20:00:01 +00001//===--- DynamicStaticInitializersCheck.cpp - clang-tidy ------------------===//
2//
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
6//
7//===----------------------------------------------------------------------===//
8
9#include "DynamicStaticInitializersCheck.h"
10#include "clang/AST/ASTContext.h"
11#include "clang/ASTMatchers/ASTMatchFinder.h"
12
13using namespace clang::ast_matchers;
14
15namespace clang {
16namespace tidy {
17namespace bugprone {
18
19AST_MATCHER(clang::VarDecl, hasConstantDeclaration) {
20 const Expr *Init = Node.getInit();
21 if (Init && !Init->isValueDependent()) {
22 if (Node.isConstexpr())
23 return true;
24 return Node.checkInitIsICE();
25 }
26 return false;
27}
28
29DynamicStaticInitializersCheck::DynamicStaticInitializersCheck(StringRef Name,
30 ClangTidyContext *Context)
31 : ClangTidyCheck(Name, Context),
32 RawStringHeaderFileExtensions(Options.getLocalOrGlobal(
33 "HeaderFileExtensions", utils::defaultHeaderFileExtensions())) {
34 if (!utils::parseHeaderFileExtensions(RawStringHeaderFileExtensions,
35 HeaderFileExtensions, ',')) {
36 llvm::errs() << "Invalid header file extension: "
37 << RawStringHeaderFileExtensions << "\n";
38 }
39}
40
41void DynamicStaticInitializersCheck::storeOptions(
42 ClangTidyOptions::OptionMap &Opts) {
43 Options.store(Opts, "HeaderFileExtensions", RawStringHeaderFileExtensions);
44}
45
46void DynamicStaticInitializersCheck::registerMatchers(MatchFinder *Finder) {
Yuanfang Chenf24c1e62019-08-21 20:00:01 +000047 Finder->addMatcher(
48 varDecl(hasGlobalStorage(), unless(hasConstantDeclaration())).bind("var"),
49 this);
50}
51
52void DynamicStaticInitializersCheck::check(const MatchFinder::MatchResult &Result) {
53 const auto *Var = Result.Nodes.getNodeAs<VarDecl>("var");
54 SourceLocation Loc = Var->getLocation();
55 if (!Loc.isValid() || !utils::isPresumedLocInHeaderFile(Loc, *Result.SourceManager,
56 HeaderFileExtensions))
57 return;
58 // If the initializer is a constant expression, then the compiler
59 // doesn't have to dynamically initialize it.
60 diag(Loc, "static variable %0 may be dynamically initialized in this header file")
61 << Var;
62}
63
64} // namespace bugprone
65} // namespace tidy
66} // namespace clang