clang-tools 22.0.0git
OptionalValueConversionCheck.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
10#include "../utils/LexerUtils.h"
11#include "../utils/Matchers.h"
13#include "clang/AST/ASTContext.h"
14#include "clang/ASTMatchers/ASTMatchFinder.h"
15#include "clang/ASTMatchers/ASTMatchers.h"
16#include <array>
17
18using namespace clang::ast_matchers;
19using clang::ast_matchers::internal::Matcher;
20
21namespace clang::tidy::bugprone {
22
23namespace {
24
25AST_MATCHER_P(QualType, hasCleanType, Matcher<QualType>, InnerMatcher) {
26 return InnerMatcher.matches(
27 Node.getNonReferenceType().getUnqualifiedType().getCanonicalType(),
28 Finder, Builder);
29}
30
31constexpr std::array<StringRef, 2> MakeSmartPtrList{
32 "::std::make_unique",
33 "::std::make_shared",
34};
35constexpr StringRef MakeOptional = "::std::make_optional";
36
37} // namespace
38
40 StringRef Name, ClangTidyContext *Context)
41 : ClangTidyCheck(Name, Context),
42 OptionalTypes(utils::options::parseStringList(
43 Options.get("OptionalTypes",
44 "::std::optional;::absl::optional;::boost::optional"))),
45 ValueMethods(utils::options::parseStringList(
46 Options.get("ValueMethods", "::value$;::get$"))) {}
47
48std::optional<TraversalKind>
52
54 auto BindOptionalType = qualType(
55 hasCleanType(qualType(hasDeclaration(namedDecl(
56 matchers::matchesAnyListedName(OptionalTypes))))
57 .bind("optional-type")));
58
59 auto EqualsBoundOptionalType =
60 qualType(hasCleanType(equalsBoundNode("optional-type")));
61
62 auto OptionalDerefMatcherImpl = callExpr(
63 anyOf(
64 cxxOperatorCallExpr(hasOverloadedOperatorName("*"),
65 hasUnaryOperand(hasType(EqualsBoundOptionalType)))
66 .bind("op-call"),
67 cxxMemberCallExpr(thisPointerType(EqualsBoundOptionalType),
68 callee(cxxMethodDecl(anyOf(
69 hasOverloadedOperatorName("*"),
70 matchers::matchesAnyListedName(ValueMethods)))))
71 .bind("member-call")),
72 hasType(qualType().bind("value-type")));
73
74 auto StdMoveCallMatcher =
75 callExpr(argumentCountIs(1), callee(functionDecl(hasName("::std::move"))),
76 hasArgument(0, ignoringImpCasts(OptionalDerefMatcherImpl)));
77 auto OptionalDerefMatcher =
78 ignoringImpCasts(anyOf(OptionalDerefMatcherImpl, StdMoveCallMatcher));
79
80 Finder->addMatcher(
81 expr(anyOf(
82 // construct optional
83 cxxConstructExpr(argumentCountIs(1), hasType(BindOptionalType),
84 hasArgument(0, OptionalDerefMatcher)),
85 // known template methods in std
86 callExpr(
87 argumentCountIs(1),
88 anyOf(
89 // match std::make_unique std::make_shared
90 callee(functionDecl(
91 matchers::matchesAnyListedName(MakeSmartPtrList),
92 hasTemplateArgument(
93 0, refersToType(BindOptionalType)))),
94 // match first std::make_optional by limit argument count
95 // (1) and template count (1).
96 // 1. template< class T > constexpr
97 // std::optional<decay_t<T>> make_optional(T&& value);
98 // 2. template< class T, class... Args > constexpr
99 // std::optional<T> make_optional(Args&&... args);
100 callee(functionDecl(templateArgumentCountIs(1),
101 hasName(MakeOptional),
102 returns(BindOptionalType)))),
103 hasArgument(0, OptionalDerefMatcher)),
104 callExpr(
105
106 argumentCountIs(1),
107
108 hasArgument(0, OptionalDerefMatcher))),
109 unless(anyOf(hasAncestor(typeLoc()),
110 hasAncestor(expr(matchers::hasUnevaluatedContext())))))
111 .bind("expr"),
112 this);
113}
114
117 Options.store(Opts, "OptionalTypes",
119 Options.store(Opts, "ValueMethods",
121}
122
124 const MatchFinder::MatchResult &Result) {
125 const auto *MatchedExpr = Result.Nodes.getNodeAs<Expr>("expr");
126 const auto *OptionalType = Result.Nodes.getNodeAs<QualType>("optional-type");
127 const auto *ValueType = Result.Nodes.getNodeAs<QualType>("value-type");
128
129 diag(MatchedExpr->getExprLoc(),
130 "conversion from %0 into %1 and back into %0, remove potentially "
131 "error-prone optional dereference")
132 << *OptionalType << ValueType->getUnqualifiedType();
133
134 if (const auto *OperatorExpr =
135 Result.Nodes.getNodeAs<CXXOperatorCallExpr>("op-call")) {
136 diag(OperatorExpr->getExprLoc(), "remove '*' to silence this warning",
137 DiagnosticIDs::Note)
138 << FixItHint::CreateRemoval(CharSourceRange::getTokenRange(
139 OperatorExpr->getBeginLoc(), OperatorExpr->getExprLoc()));
140 return;
141 }
142 if (const auto *CallExpr =
143 Result.Nodes.getNodeAs<CXXMemberCallExpr>("member-call")) {
144 const SourceLocation Begin =
145 utils::lexer::getPreviousToken(CallExpr->getExprLoc(),
146 *Result.SourceManager, getLangOpts())
147 .getLocation();
148 auto Diag =
149 diag(CallExpr->getExprLoc(),
150 "remove call to %0 to silence this warning", DiagnosticIDs::Note);
151 Diag << CallExpr->getMethodDecl()
152 << FixItHint::CreateRemoval(
153 CharSourceRange::getTokenRange(Begin, CallExpr->getEndLoc()));
154 if (const auto *Member =
155 llvm::dyn_cast<MemberExpr>(CallExpr->getCallee()->IgnoreImplicit());
156 Member && Member->isArrow())
157 Diag << FixItHint::CreateInsertion(CallExpr->getBeginLoc(), "*");
158 return;
159 }
160}
161
162} // namespace clang::tidy::bugprone
Every ClangTidyCheck reports errors through a DiagnosticsEngine provided by this context.
void registerMatchers(ast_matchers::MatchFinder *Finder) override
void storeOptions(ClangTidyOptions::OptionMap &Opts) override
OptionalValueConversionCheck(StringRef Name, ClangTidyContext *Context)
void check(const ast_matchers::MatchFinder::MatchResult &Result) override
std::optional< TraversalKind > getCheckTraversalKind() const override
AST_MATCHER_P(Stmt, isStatementIdenticalToBoundNode, std::string, ID)
inline ::clang::ast_matchers::internal::Matcher< NamedDecl > matchesAnyListedName(llvm::ArrayRef< StringRef > NameList)
Token getPreviousToken(SourceLocation Location, const SourceManager &SM, const LangOptions &LangOpts, bool SkipComments)
Returns previous token or tok::unknown if not found.
std::string serializeStringList(ArrayRef< StringRef > Strings)
Serialize a sequence of names that can be parsed by parseStringList.
llvm::StringMap< ClangTidyValue > OptionMap