clang-tools 23.0.0git
utils/UseRangesCheck.cpp
Go to the documentation of this file.
1//===----------------------------------------------------------------------===//
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 "UseRangesCheck.h"
10#include "Matchers.h"
11#include "clang/AST/ASTContext.h"
12#include "clang/AST/Decl.h"
13#include "clang/AST/Expr.h"
14#include "clang/ASTMatchers/ASTMatchFinder.h"
15#include "clang/ASTMatchers/ASTMatchers.h"
16#include "clang/ASTMatchers/ASTMatchersInternal.h"
17#include "clang/Basic/Diagnostic.h"
18#include "clang/Basic/LLVM.h"
19#include "clang/Basic/SourceLocation.h"
20#include "clang/Basic/SourceManager.h"
21#include "clang/Lex/Lexer.h"
22#include "llvm/ADT/ArrayRef.h"
23#include "llvm/ADT/STLExtras.h"
24#include "llvm/ADT/SmallBitVector.h"
25#include "llvm/ADT/SmallVector.h"
26#include "llvm/ADT/StringRef.h"
27#include "llvm/ADT/Twine.h"
28#include "llvm/Support/raw_ostream.h"
29#include <cassert>
30#include <optional>
31#include <string>
32
33using namespace clang::ast_matchers;
34
35static constexpr const char BoundCall[] = "CallExpr";
36static constexpr const char FuncDecl[] = "FuncDecl";
37static constexpr const char ArgName[] = "ArgName";
38
39namespace clang::tidy::utils {
40
41static std::string getFullPrefix(ArrayRef<UseRangesCheck::Indexes> Signature) {
42 std::string Output;
43 llvm::raw_string_ostream OS(Output);
44 for (const UseRangesCheck::Indexes &Item : Signature)
45 OS << Item.BeginArg << ":" << Item.EndArg << ":"
46 << (Item.ReplaceArg == UseRangesCheck::Indexes::First ? '0' : '1');
47 return Output;
48}
49
50namespace {
51
52AST_MATCHER(Expr, hasSideEffects) {
53 return Node.HasSideEffects(Finder->getASTContext());
54}
55} // namespace
56
57static auto
58makeExprMatcher(const ast_matchers::internal::Matcher<Expr> &ArgumentMatcher,
59 ArrayRef<StringRef> MethodNames,
60 ArrayRef<StringRef> FreeNames) {
61 return expr(
62 anyOf(cxxMemberCallExpr(argumentCountIs(0),
63 callee(cxxMethodDecl(hasAnyName(MethodNames))),
64 on(ArgumentMatcher)),
65 callExpr(argumentCountIs(1), hasArgument(0, ArgumentMatcher),
66 hasDeclaration(functionDecl(hasAnyName(FreeNames))))));
67}
68
69static ast_matchers::internal::Matcher<CallExpr>
70makeMatcherPair(StringRef State, const UseRangesCheck::Indexes &Indexes,
71 ArrayRef<StringRef> BeginFreeNames,
72 ArrayRef<StringRef> EndFreeNames,
73 const std::optional<UseRangesCheck::ReverseIteratorDescriptor>
74 &ReverseDescriptor) {
75 std::string ArgBound = (ArgName + llvm::Twine(Indexes.BeginArg)).str();
76 const SmallString<64> ID = {BoundCall, State};
77 ast_matchers::internal::Matcher<CallExpr> ArgumentMatcher = allOf(
78 hasArgument(Indexes.BeginArg,
79 makeExprMatcher(expr(unless(hasSideEffects())).bind(ArgBound),
80 {"begin", "cbegin"}, BeginFreeNames)),
81 hasArgument(Indexes.EndArg,
83 expr(matchers::isStatementIdenticalToBoundNode(ArgBound)),
84 {"end", "cend"}, EndFreeNames)));
85 if (ReverseDescriptor) {
86 ArgBound.push_back('R');
87 const SmallVector<StringRef> RBegin{
88 llvm::make_first_range(ReverseDescriptor->FreeReverseNames)};
89 const SmallVector<StringRef> REnd{
90 llvm::make_second_range(ReverseDescriptor->FreeReverseNames)};
91 ArgumentMatcher = anyOf(
92 ArgumentMatcher,
93 allOf(hasArgument(
94 Indexes.BeginArg,
95 makeExprMatcher(expr(unless(hasSideEffects())).bind(ArgBound),
96 {"rbegin", "crbegin"}, RBegin)),
97 hasArgument(
98 Indexes.EndArg,
100 expr(matchers::isStatementIdenticalToBoundNode(ArgBound)),
101 {"rend", "crend"}, REnd))));
102 }
103 return callExpr(argumentCountAtLeast(
104 std::max(Indexes.BeginArg, Indexes.EndArg) + 1),
105 ArgumentMatcher)
106 .bind(ID);
107}
108
109void UseRangesCheck::registerMatchers(MatchFinder *Finder) {
110 auto Replaces = getReplacerMap();
111 ReverseDescriptor = getReverseDescriptor();
112 auto BeginEndNames = getFreeBeginEndMethods();
113 const SmallVector<StringRef, 4> BeginNames{
114 llvm::make_first_range(BeginEndNames)};
115 const SmallVector<StringRef, 4> EndNames{
116 llvm::make_second_range(BeginEndNames)};
117 Replacers.clear();
118 llvm::DenseSet<Replacer *> SeenRepl;
119 for (auto I = Replaces.begin(), E = Replaces.end(); I != E; ++I) {
120 auto Replacer = I->getValue();
121 if (!SeenRepl.insert(Replacer.get()).second)
122 continue;
123 Replacers.push_back(Replacer);
124 assert(!Replacer->getReplacementSignatures().empty() &&
125 llvm::all_of(Replacer->getReplacementSignatures(),
126 [](const Signature &Index) { return !Index.empty(); }));
127 std::vector<StringRef> Names(1, I->getKey());
128 for (auto J = std::next(I); J != E; ++J)
129 if (J->getValue() == Replacer)
130 Names.push_back(J->getKey());
131
132 std::vector<ast_matchers::internal::DynTypedMatcher> TotalMatchers;
133 // As we match on the first matched signature, we need to sort the
134 // signatures in order of length(longest to shortest). This way any
135 // signature that is a subset of another signature will be matched after the
136 // other.
138 llvm::sort(SigVec, [](auto &L, auto &R) { return R.size() < L.size(); });
139 for (const auto &Signature : SigVec) {
140 std::vector<ast_matchers::internal::DynTypedMatcher> Matchers;
141 for (const auto &ArgPair : Signature)
142 Matchers.push_back(makeMatcherPair(getFullPrefix(Signature), ArgPair,
143 BeginNames, EndNames,
144 ReverseDescriptor));
145 TotalMatchers.push_back(
146 ast_matchers::internal::DynTypedMatcher::constructVariadic(
147 ast_matchers::internal::DynTypedMatcher::VO_AllOf,
148 ASTNodeKind::getFromNodeKind<CallExpr>(), std::move(Matchers)));
149 }
150 Finder->addMatcher(
151 callExpr(
152 callee(functionDecl(hasAnyName(Names))
153 .bind((FuncDecl + Twine(Replacers.size() - 1).str()))),
154 ast_matchers::internal::DynTypedMatcher::constructVariadic(
155 ast_matchers::internal::DynTypedMatcher::VO_AnyOf,
156 ASTNodeKind::getFromNodeKind<CallExpr>(),
157 std::move(TotalMatchers))
158 .convertTo<CallExpr>()),
159 this);
160 }
161}
162
163static void removeFunctionArgs(const DiagnosticBuilder &Diag,
164 const CallExpr &Call, ArrayRef<unsigned> Indexes,
165 const ASTContext &Ctx) {
166 SmallVector<unsigned> Sorted(Indexes);
167 llvm::sort(Sorted);
168 // Keep track of commas removed
169 llvm::SmallBitVector Commas(Call.getNumArgs());
170 // The first comma is actually the '(' which we can't remove
171 Commas[0] = true;
172 for (const unsigned Index : Sorted) {
173 const Expr *Arg = Call.getArg(Index);
174 if (Commas[Index]) {
175 if (Index >= Commas.size()) {
176 Diag << FixItHint::CreateRemoval(Arg->getSourceRange());
177 } else {
178 // Remove the next comma
179 Commas[Index + 1] = true;
180 Diag << FixItHint::CreateRemoval(CharSourceRange::getTokenRange(
181 {Arg->getBeginLoc(),
182 Lexer::getLocForEndOfToken(
183 Arg->getEndLoc(), 0, Ctx.getSourceManager(), Ctx.getLangOpts())
184 .getLocWithOffset(1)}));
185 }
186 } else {
187 Diag << FixItHint::CreateRemoval(CharSourceRange::getTokenRange(
188 Arg->getBeginLoc().getLocWithOffset(-1), Arg->getEndLoc()));
189 Commas[Index] = true;
190 }
191 }
192}
193
194static bool isResultUsed(const DynTypedNode &Node,
195 const ast_matchers::MatchFinder::MatchResult &Result) {
196 const DynTypedNodeList Parents = Result.Context->getParents(Node);
197 assert(Parents.size() == 1 &&
198 "Expected exactly one parent for a matched algorithm call");
199 const DynTypedNode &Parent = Parents[0];
200 if (Parent.get<CompoundStmt>())
201 return false;
202 if (const auto *Cleanups = Parent.get<ExprWithCleanups>())
203 return isResultUsed(DynTypedNode::create(*Cleanups), Result);
204 if (const auto *Temporary = Parent.get<CXXBindTemporaryExpr>())
205 return isResultUsed(DynTypedNode::create(*Temporary), Result);
206 return true;
207}
208
209static bool isResultUsed(const CallExpr &Call,
210 const ast_matchers::MatchFinder::MatchResult &Result) {
211 return isResultUsed(DynTypedNode::create(Call), Result);
212}
213
214static void insertAccessor(const DiagnosticBuilder &Diag, const CallExpr &Call,
215 StringRef Accessor, const ASTContext &Ctx) {
216 const SourceLocation End = Lexer::getLocForEndOfToken(
217 Call.getEndLoc(), 0, Ctx.getSourceManager(), Ctx.getLangOpts());
218 if (End.isValid())
219 Diag << FixItHint::CreateInsertion(End, Accessor);
220}
221
222void UseRangesCheck::check(const MatchFinder::MatchResult &Result) {
223 const Replacer *Replacer = nullptr;
224 const FunctionDecl *Function = nullptr;
225 for (const auto &[Node, Value] : Result.Nodes.getMap()) {
226 StringRef NodeStr(Node);
227 if (!NodeStr.consume_front(FuncDecl))
228 continue;
229 Function = Value.get<FunctionDecl>();
230 size_t Index = 0;
231 if (NodeStr.getAsInteger(10, Index))
232 llvm_unreachable("Unable to extract replacer index");
233 assert(Index < Replacers.size());
234 Replacer = Replacers[Index].get();
235 break;
236 }
237 assert(Replacer && Function);
238 SmallString<64> Buffer;
239 for (const Signature &Sig : Replacer->getReplacementSignatures()) {
240 Buffer.assign({BoundCall, getFullPrefix(Sig)});
241 const auto *Call = Result.Nodes.getNodeAs<CallExpr>(Buffer);
242 if (!Call)
243 continue;
244
245 // FIXME: This check specifically handles `CXXNullPtrLiteralExpr`, but
246 // a more general solution might be needed.
247 if (Function->getName() == "find") {
248 const unsigned ValueArgIndex = 2;
249 if (Call->getNumArgs() <= ValueArgIndex)
250 continue;
251 const Expr *ValueExpr =
252 Call->getArg(ValueArgIndex)->IgnoreParenImpCasts();
253 if (isa<CXXNullPtrLiteralExpr>(ValueExpr))
254 return;
255 }
256
257 const bool ResultUsed = isResultUsed(*Call, Result);
258 auto ResultPolicy = Replacer->getResultUsePolicy(*Function, false);
259
260 auto Diag = createDiag(*Call);
261 if (auto ReplaceName = Replacer->getReplaceName(*Function))
262 Diag << FixItHint::CreateReplacement(Call->getCallee()->getSourceRange(),
263 *ReplaceName);
264 if (auto Include = Replacer->getHeaderInclusion(*Function))
265 Diag << Inserter.createIncludeInsertion(
266 Result.SourceManager->getFileID(Call->getBeginLoc()), *Include);
268 for (const auto &[First, Second, Replace] : Sig) {
269 auto ArgNode = ArgName + std::to_string(First);
270 if (const auto *ArgExpr = Result.Nodes.getNodeAs<Expr>(ArgNode)) {
271 Diag << FixItHint::CreateReplacement(
272 Call->getArg(Replace == Indexes::Second ? Second : First)
273 ->getSourceRange(),
274 Lexer::getSourceText(
275 CharSourceRange::getTokenRange(ArgExpr->getSourceRange()),
276 Result.Context->getSourceManager(),
277 Result.Context->getLangOpts()));
278 } else {
279 assert(ReverseDescriptor && "Couldn't find forward argument");
280 ArgNode.push_back('R');
281 ArgExpr = Result.Nodes.getNodeAs<Expr>(ArgNode);
282 assert(ArgExpr && "Couldn't find forward or reverse argument");
283 if (ReverseDescriptor->ReverseHeader)
284 Diag << Inserter.createIncludeInsertion(
285 Result.SourceManager->getFileID(Call->getBeginLoc()),
286 *ReverseDescriptor->ReverseHeader);
287 const StringRef ArgText = Lexer::getSourceText(
288 CharSourceRange::getTokenRange(ArgExpr->getSourceRange()),
289 Result.Context->getSourceManager(), Result.Context->getLangOpts());
290 SmallString<128> ReplaceText;
291 if (ReverseDescriptor->IsPipeSyntax)
292 ReplaceText.assign(
293 {ArgText, " | ", ReverseDescriptor->ReverseAdaptorName});
294 else
295 ReplaceText.assign(
296 {ReverseDescriptor->ReverseAdaptorName, "(", ArgText, ")"});
297 Diag << FixItHint::CreateReplacement(
298 Call->getArg(Replace == Indexes::Second ? Second : First)
299 ->getSourceRange(),
300 ReplaceText);
301 }
302 ToRemove.push_back(Replace == Indexes::Second ? First : Second);
303 }
304 removeFunctionArgs(Diag, *Call, ToRemove, *Result.Context);
305 using ResultPolicyKind = Replacer::ResultUsePolicy::Kind;
306 if (ResultUsed && ResultPolicy.PolicyKind ==
307 ResultPolicyKind::AppendAccessorForUsedResult) {
308 insertAccessor(Diag, *Call, ResultPolicy.Accessor, *Result.Context);
309 }
310 return;
311 }
312 llvm_unreachable("No valid signature found");
313}
314
316 const LangOptions &LangOpts) const {
317 return LangOpts.CPlusPlus11;
318}
319
321 : ClangTidyCheck(Name, Context),
322 Inserter(Options.getLocalOrGlobal("IncludeStyle",
323 utils::IncludeSorter::IS_LLVM),
324 areDiagsSelfContained()) {}
325
326void UseRangesCheck::registerPPCallbacks(const SourceManager &,
327 Preprocessor *PP, Preprocessor *) {
328 Inserter.registerPreprocessor(PP);
329}
330
332 Options.store(Opts, "IncludeStyle", Inserter.getStyle());
333}
334
335std::optional<std::string>
337 return std::nullopt;
338}
339
341UseRangesCheck::Replacer::getResultUsePolicy(const NamedDecl &, bool) const {
342 return {};
343}
344
345DiagnosticBuilder UseRangesCheck::createDiag(const CallExpr &Call) {
346 return diag(Call.getBeginLoc(), "use a ranges version of this algorithm");
347}
348
349std::optional<UseRangesCheck::ReverseIteratorDescriptor>
351 return std::nullopt;
352}
353
354ArrayRef<std::pair<StringRef, StringRef>>
356 return {};
357}
358
359std::optional<TraversalKind> UseRangesCheck::getCheckTraversalKind() const {
360 return TK_IgnoreUnlessSpelledInSource;
361}
362} // namespace clang::tidy::utils
Every ClangTidyCheck reports errors through a DiagnosticsEngine provided by this context.
virtual ArrayRef< Signature > getReplacementSignatures() const =0
Gets an array of all the possible overloads for a function with indexes where begin and end arguments...
virtual ResultUsePolicy getResultUsePolicy(const NamedDecl &OriginalName, bool IsStructuredBinding) const
Gets how to handle fix-its when the original algorithm result is used.
virtual std::optional< std::string > getHeaderInclusion(const NamedDecl &OriginalName) const
Gets the header needed to access the replaced function Return std::nullopt if no new header is needed...
virtual std::optional< std::string > getReplaceName(const NamedDecl &OriginalName) const =0
Gets the name to replace a function with, return std::nullopt for a replacement where we just call a ...
void storeOptions(ClangTidyOptions::OptionMap &Opts) override
virtual DiagnosticBuilder createDiag(const CallExpr &Call)
Create a diagnostic for the CallExpr Override this to support custom diagnostic messages.
void registerMatchers(ast_matchers::MatchFinder *Finder) final
std::optional< TraversalKind > getCheckTraversalKind() const override
void check(const ast_matchers::MatchFinder::MatchResult &Result) final
void registerPPCallbacks(const SourceManager &SM, Preprocessor *PP, Preprocessor *ModuleExpanderPP) final
virtual ReplacerMap getReplacerMap() const =0
Gets a map of function to replace and methods to create the replacements.
bool isLanguageVersionSupported(const LangOptions &LangOpts) const override
UseRangesCheck(StringRef Name, ClangTidyContext *Context)
virtual std::optional< ReverseIteratorDescriptor > getReverseDescriptor() const
virtual ArrayRef< std::pair< StringRef, StringRef > > getFreeBeginEndMethods() const
Gets the fully qualified names of begin and end functions.
static std::string getFullPrefix(ArrayRef< UseRangesCheck::Indexes > Signature)
static ast_matchers::internal::Matcher< CallExpr > makeMatcherPair(StringRef State, const UseRangesCheck::Indexes &Indexes, ArrayRef< StringRef > BeginFreeNames, ArrayRef< StringRef > EndFreeNames, const std::optional< UseRangesCheck::ReverseIteratorDescriptor > &ReverseDescriptor)
static bool isResultUsed(const DynTypedNode &Node, const ast_matchers::MatchFinder::MatchResult &Result)
IncludeSorter(const SourceManager *SourceMgr, FileID FileID, StringRef FileName, IncludeStyle Style)
Class used by IncludeInserterCallback to record the names of the / inclusions in a given source file ...
static void insertAccessor(const DiagnosticBuilder &Diag, const CallExpr &Call, StringRef Accessor, const ASTContext &Ctx)
static auto makeExprMatcher(const ast_matchers::internal::Matcher< Expr > &ArgumentMatcher, ArrayRef< StringRef > MethodNames, ArrayRef< StringRef > FreeNames)
static void removeFunctionArgs(const DiagnosticBuilder &Diag, const CallExpr &Call, ArrayRef< unsigned > Indexes, const ASTContext &Ctx)
llvm::StringMap< ClangTidyValue > OptionMap
static constexpr const char ArgName[]
static constexpr const char BoundCall[]
static constexpr const char FuncDecl[]