annotate clang-tools-extra/clang-tidy/utils/RenamerClangTidyCheck.h @ 150:1d019706d866

LLVM10
author anatofuz
date Thu, 13 Feb 2020 15:10:13 +0900
parents
children 0572611fdcc8
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
150
anatofuz
parents:
diff changeset
1 //===--- RenamderClangTidyCheck.h - clang-tidy ------------------*- C++ -*-===//
anatofuz
parents:
diff changeset
2 //
anatofuz
parents:
diff changeset
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
anatofuz
parents:
diff changeset
4 // See https://llvm.org/LICENSE.txt for license information.
anatofuz
parents:
diff changeset
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
anatofuz
parents:
diff changeset
6 //
anatofuz
parents:
diff changeset
7 //===----------------------------------------------------------------------===//
anatofuz
parents:
diff changeset
8
anatofuz
parents:
diff changeset
9 #ifndef LLVM_CLANG_TOOLS_EXTRA_CLANG_TIDY_RENAMERCLANGTIDYCHECK_H
anatofuz
parents:
diff changeset
10 #define LLVM_CLANG_TOOLS_EXTRA_CLANG_TIDY_RENAMERCLANGTIDYCHECK_H
anatofuz
parents:
diff changeset
11
anatofuz
parents:
diff changeset
12 #include "../ClangTidyCheck.h"
anatofuz
parents:
diff changeset
13 #include "llvm/ADT/DenseMap.h"
anatofuz
parents:
diff changeset
14 #include "llvm/ADT/DenseSet.h"
anatofuz
parents:
diff changeset
15 #include "llvm/ADT/Optional.h"
anatofuz
parents:
diff changeset
16 #include <string>
anatofuz
parents:
diff changeset
17 #include <utility>
anatofuz
parents:
diff changeset
18
anatofuz
parents:
diff changeset
19 namespace clang {
anatofuz
parents:
diff changeset
20
anatofuz
parents:
diff changeset
21 class MacroInfo;
anatofuz
parents:
diff changeset
22
anatofuz
parents:
diff changeset
23 namespace tidy {
anatofuz
parents:
diff changeset
24
anatofuz
parents:
diff changeset
25 /// Base class for clang-tidy checks that want to flag declarations and/or
anatofuz
parents:
diff changeset
26 /// macros for renaming based on customizable criteria.
anatofuz
parents:
diff changeset
27 class RenamerClangTidyCheck : public ClangTidyCheck {
anatofuz
parents:
diff changeset
28 public:
anatofuz
parents:
diff changeset
29 RenamerClangTidyCheck(StringRef CheckName, ClangTidyContext *Context);
anatofuz
parents:
diff changeset
30 ~RenamerClangTidyCheck();
anatofuz
parents:
diff changeset
31
anatofuz
parents:
diff changeset
32 /// Derived classes should not implement any matching logic themselves; this
anatofuz
parents:
diff changeset
33 /// class will do the matching and call the derived class'
anatofuz
parents:
diff changeset
34 /// GetDeclFailureInfo() and GetMacroFailureInfo() for determining whether a
anatofuz
parents:
diff changeset
35 /// given identifier passes or fails the check.
anatofuz
parents:
diff changeset
36 void registerMatchers(ast_matchers::MatchFinder *Finder) override final;
anatofuz
parents:
diff changeset
37 void
anatofuz
parents:
diff changeset
38 check(const ast_matchers::MatchFinder::MatchResult &Result) override final;
anatofuz
parents:
diff changeset
39 void registerPPCallbacks(const SourceManager &SM, Preprocessor *PP,
anatofuz
parents:
diff changeset
40 Preprocessor *ModuleExpanderPP) override final;
anatofuz
parents:
diff changeset
41 void onEndOfTranslationUnit() override final;
anatofuz
parents:
diff changeset
42
anatofuz
parents:
diff changeset
43 /// This enum will be used in %select of the diagnostic message.
anatofuz
parents:
diff changeset
44 /// Each value below IgnoreFailureThreshold should have an error message.
anatofuz
parents:
diff changeset
45 enum class ShouldFixStatus {
anatofuz
parents:
diff changeset
46 ShouldFix,
anatofuz
parents:
diff changeset
47
anatofuz
parents:
diff changeset
48 /// The fixup will conflict with a language keyword,
anatofuz
parents:
diff changeset
49 /// so we can't fix it automatically.
anatofuz
parents:
diff changeset
50 ConflictsWithKeyword,
anatofuz
parents:
diff changeset
51
anatofuz
parents:
diff changeset
52 /// The fixup will conflict with a macro
anatofuz
parents:
diff changeset
53 /// definition, so we can't fix it
anatofuz
parents:
diff changeset
54 /// automatically.
anatofuz
parents:
diff changeset
55 ConflictsWithMacroDefinition,
anatofuz
parents:
diff changeset
56
anatofuz
parents:
diff changeset
57 /// Values pass this threshold will be ignored completely
anatofuz
parents:
diff changeset
58 /// i.e no message, no fixup.
anatofuz
parents:
diff changeset
59 IgnoreFailureThreshold,
anatofuz
parents:
diff changeset
60
anatofuz
parents:
diff changeset
61 /// If the identifier was used or declared within a macro we
anatofuz
parents:
diff changeset
62 /// won't offer a fixup for safety reasons.
anatofuz
parents:
diff changeset
63 InsideMacro,
anatofuz
parents:
diff changeset
64 };
anatofuz
parents:
diff changeset
65
anatofuz
parents:
diff changeset
66 /// Information describing a failed check
anatofuz
parents:
diff changeset
67 struct FailureInfo {
anatofuz
parents:
diff changeset
68 std::string KindName; // Tag or misc info to be used as derived classes need
anatofuz
parents:
diff changeset
69 std::string Fixup; // The name that will be proposed as a fix-it hint
anatofuz
parents:
diff changeset
70 };
anatofuz
parents:
diff changeset
71
anatofuz
parents:
diff changeset
72 /// Holds an identifier name check failure, tracking the kind of the
anatofuz
parents:
diff changeset
73 /// identifier, its possible fixup and the starting locations of all the
anatofuz
parents:
diff changeset
74 /// identifier usages.
anatofuz
parents:
diff changeset
75 struct NamingCheckFailure {
anatofuz
parents:
diff changeset
76 FailureInfo Info;
anatofuz
parents:
diff changeset
77
anatofuz
parents:
diff changeset
78 /// Whether the failure should be fixed or not.
anatofuz
parents:
diff changeset
79 ///
anatofuz
parents:
diff changeset
80 /// e.g.: if the identifier was used or declared within a macro we won't
anatofuz
parents:
diff changeset
81 /// offer a fixup for safety reasons.
anatofuz
parents:
diff changeset
82 bool ShouldFix() const {
anatofuz
parents:
diff changeset
83 return FixStatus == ShouldFixStatus::ShouldFix && !Info.Fixup.empty();
anatofuz
parents:
diff changeset
84 }
anatofuz
parents:
diff changeset
85
anatofuz
parents:
diff changeset
86 bool ShouldNotify() const {
anatofuz
parents:
diff changeset
87 return FixStatus < ShouldFixStatus::IgnoreFailureThreshold;
anatofuz
parents:
diff changeset
88 }
anatofuz
parents:
diff changeset
89
anatofuz
parents:
diff changeset
90 ShouldFixStatus FixStatus = ShouldFixStatus::ShouldFix;
anatofuz
parents:
diff changeset
91
anatofuz
parents:
diff changeset
92 /// A set of all the identifier usages starting SourceLocation, in
anatofuz
parents:
diff changeset
93 /// their encoded form.
anatofuz
parents:
diff changeset
94 llvm::DenseSet<unsigned> RawUsageLocs;
anatofuz
parents:
diff changeset
95
anatofuz
parents:
diff changeset
96 NamingCheckFailure() = default;
anatofuz
parents:
diff changeset
97 };
anatofuz
parents:
diff changeset
98
anatofuz
parents:
diff changeset
99 using NamingCheckId = std::pair<SourceLocation, std::string>;
anatofuz
parents:
diff changeset
100
anatofuz
parents:
diff changeset
101 using NamingCheckFailureMap =
anatofuz
parents:
diff changeset
102 llvm::DenseMap<NamingCheckId, NamingCheckFailure>;
anatofuz
parents:
diff changeset
103
anatofuz
parents:
diff changeset
104 /// Check Macros for style violations.
anatofuz
parents:
diff changeset
105 void checkMacro(SourceManager &sourceMgr, const Token &MacroNameTok,
anatofuz
parents:
diff changeset
106 const MacroInfo *MI);
anatofuz
parents:
diff changeset
107
anatofuz
parents:
diff changeset
108 /// Add a usage of a macro if it already has a violation.
anatofuz
parents:
diff changeset
109 void expandMacro(const Token &MacroNameTok, const MacroInfo *MI);
anatofuz
parents:
diff changeset
110
anatofuz
parents:
diff changeset
111 protected:
anatofuz
parents:
diff changeset
112 /// Overridden by derived classes, returns information about if and how a Decl
anatofuz
parents:
diff changeset
113 /// failed the check. A 'None' result means the Decl did not fail the check.
anatofuz
parents:
diff changeset
114 virtual llvm::Optional<FailureInfo>
anatofuz
parents:
diff changeset
115 GetDeclFailureInfo(const NamedDecl *Decl, const SourceManager &SM) const = 0;
anatofuz
parents:
diff changeset
116
anatofuz
parents:
diff changeset
117 /// Overridden by derived classes, returns information about if and how a
anatofuz
parents:
diff changeset
118 /// macro failed the check. A 'None' result means the macro did not fail the
anatofuz
parents:
diff changeset
119 /// check.
anatofuz
parents:
diff changeset
120 virtual llvm::Optional<FailureInfo>
anatofuz
parents:
diff changeset
121 GetMacroFailureInfo(const Token &MacroNameTok,
anatofuz
parents:
diff changeset
122 const SourceManager &SM) const = 0;
anatofuz
parents:
diff changeset
123
anatofuz
parents:
diff changeset
124 /// Represents customized diagnostic text and how arguments should be applied.
anatofuz
parents:
diff changeset
125 /// Example usage:
anatofuz
parents:
diff changeset
126 ///
anatofuz
parents:
diff changeset
127 /// return DiagInfo{"my %1 very %2 special %3 text",
anatofuz
parents:
diff changeset
128 /// [=](DiagnosticBuilder &diag) {
anatofuz
parents:
diff changeset
129 /// diag << arg1 << arg2 << arg3;
anatofuz
parents:
diff changeset
130 /// }};
anatofuz
parents:
diff changeset
131 struct DiagInfo {
anatofuz
parents:
diff changeset
132 std::string Text;
anatofuz
parents:
diff changeset
133 llvm::unique_function<void(DiagnosticBuilder &)> ApplyArgs;
anatofuz
parents:
diff changeset
134 };
anatofuz
parents:
diff changeset
135
anatofuz
parents:
diff changeset
136 /// Overridden by derived classes, returns a description of the diagnostic
anatofuz
parents:
diff changeset
137 /// that should be emitted for the given failure. The base class will then
anatofuz
parents:
diff changeset
138 /// further customize the diagnostic by adding info about whether the fix-it
anatofuz
parents:
diff changeset
139 /// can be automatically applied or not.
anatofuz
parents:
diff changeset
140 virtual DiagInfo GetDiagInfo(const NamingCheckId &ID,
anatofuz
parents:
diff changeset
141 const NamingCheckFailure &Failure) const = 0;
anatofuz
parents:
diff changeset
142
anatofuz
parents:
diff changeset
143 private:
anatofuz
parents:
diff changeset
144 NamingCheckFailureMap NamingCheckFailures;
anatofuz
parents:
diff changeset
145 };
anatofuz
parents:
diff changeset
146
anatofuz
parents:
diff changeset
147 } // namespace tidy
anatofuz
parents:
diff changeset
148 } // namespace clang
anatofuz
parents:
diff changeset
149
anatofuz
parents:
diff changeset
150 #endif // LLVM_CLANG_TOOLS_EXTRA_CLANG_TIDY_RENAMERCLANGTIDYCHECK_H