clang-tools 23.0.0git
llvm/RedundantCastingCheck.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 "clang/AST/ASTTypeTraits.h"
11#include "clang/AST/Decl.h"
12#include "clang/AST/DeclCXX.h"
13#include "clang/AST/Expr.h"
14#include "clang/AST/ExprCXX.h"
15#include "clang/AST/NestedNameSpecifierBase.h"
16#include "clang/AST/ParentMapContext.h"
17#include "clang/AST/TemplateBase.h"
18#include "clang/AST/TypeBase.h"
19#include "clang/ASTMatchers/ASTMatchFinder.h"
20#include "clang/ASTMatchers/ASTMatchers.h"
21#include "clang/Lex/Lexer.h"
22#include "llvm/ADT/STLExtras.h"
23
24using namespace clang::ast_matchers;
25
27
28namespace {
29AST_MATCHER(Expr, isMacroID) { return Node.getExprLoc().isMacroID(); }
30AST_MATCHER_P(OverloadExpr, hasAnyUnresolvedName, ArrayRef<StringRef>, Names) {
31 auto DeclName = Node.getName();
32 if (!DeclName.isIdentifier())
33 return false;
34 const IdentifierInfo *II = DeclName.getAsIdentifierInfo();
35 return llvm::any_of(Names, [II](StringRef Name) { return II->isStr(Name); });
36}
37} // namespace
38
39static constexpr StringRef FunctionNames[] = {
40 "cast", "cast_or_null", "cast_if_present",
41 "dyn_cast", "dyn_cast_or_null", "dyn_cast_if_present"};
42
43void RedundantCastingCheck::registerMatchers(MatchFinder *Finder) {
44 auto IsInLLVMNamespace = hasDeclContext(
45 namespaceDecl(hasName("llvm"), hasDeclContext(translationUnitDecl())));
46 auto AnyCalleeName =
47 allOf(unless(isMacroID()), unless(cxxMemberCallExpr()),
48 callee(expr(ignoringImpCasts(
49 declRefExpr(
50 to(namedDecl(hasAnyName(FunctionNames), IsInLLVMNamespace)),
51 templateArgumentLocCountIs(1))
52 .bind("callee")))));
53 auto AnyCalleeNameInUninstantiatedTemplate =
54 allOf(unless(isMacroID()), unless(cxxMemberCallExpr()),
55 callee(expr(ignoringImpCasts(
56 unresolvedLookupExpr(hasAnyUnresolvedName(FunctionNames),
57 templateArgumentLocCountIs(1))
58 .bind("callee")))));
59 Finder->addMatcher(callExpr(AnyCalleeName, argumentCountIs(1),
60 optionally(hasParent(
61 callExpr(AnyCalleeName).bind("parent_cast"))))
62 .bind("call"),
63 this);
64 Finder->addMatcher(
65 callExpr(
66 AnyCalleeNameInUninstantiatedTemplate, argumentCountIs(1),
67 optionally(hasAncestor(
68 namespaceDecl(hasName("llvm"), hasParent(translationUnitDecl()))
69 .bind("llvm_ns"))))
70 .bind("call"),
71 this);
72}
73
74static QualType stripPointerOrReference(QualType Ty) {
75 QualType Pointee = Ty->getPointeeType();
76 if (Pointee.isNull())
77 return Ty;
78 return Pointee;
79}
80
81static bool isLLVMNamespace(NestedNameSpecifier NNS) {
82 if (NNS.getKind() != NestedNameSpecifier::Kind::Namespace)
83 return false;
84 auto Pair = NNS.getAsNamespaceAndPrefix();
85 if (Pair.Namespace->getNamespace()->getName() != "llvm")
86 return false;
87 const NestedNameSpecifier::Kind Kind = Pair.Prefix.getKind();
88 return Kind == NestedNameSpecifier::Kind::Null ||
89 Kind == NestedNameSpecifier::Kind::Global;
90}
91
92void RedundantCastingCheck::check(const MatchFinder::MatchResult &Result) {
93 const auto &Nodes = Result.Nodes;
94 const auto *Call = Nodes.getNodeAs<CallExpr>("call");
95
96 CanQualType RetTy;
97 std::string FuncName;
98 if (const auto *ResolvedCallee = Nodes.getNodeAs<DeclRefExpr>("callee")) {
99 const auto *F = cast<FunctionDecl>(ResolvedCallee->getDecl());
100 RetTy = stripPointerOrReference(F->getReturnType())
101 ->getCanonicalTypeUnqualified();
102 FuncName = F->getName();
103 } else if (const auto *UnresolvedCallee =
104 Nodes.getNodeAs<UnresolvedLookupExpr>("callee")) {
105 const bool IsExplicitlyLLVM =
106 isLLVMNamespace(UnresolvedCallee->getQualifier());
107 const auto *CallerNS = Nodes.getNodeAs<NamedDecl>("llvm_ns");
108 if (!IsExplicitlyLLVM && !CallerNS)
109 return;
110 auto TArg = UnresolvedCallee->template_arguments()[0].getArgument();
111 if (TArg.getKind() != TemplateArgument::Type)
112 return;
113
114 RetTy = TArg.getAsType()->getCanonicalTypeUnqualified();
115 FuncName = UnresolvedCallee->getName().getAsString();
116 } else {
117 llvm_unreachable("");
118 }
119
120 const auto *Arg = Call->getArg(0);
121 const QualType ArgTy = Arg->getType();
122 const QualType ArgPointeeTy = stripPointerOrReference(ArgTy);
123 const CanQualType FromTy = ArgPointeeTy->getCanonicalTypeUnqualified();
124 const auto *FromDecl = FromTy->getAsCXXRecordDecl();
125 const auto *RetDecl = RetTy->getAsCXXRecordDecl();
126 const bool IsDerived =
127 FromDecl && RetDecl && FromDecl->isDerivedFrom(RetDecl);
128 if (FromTy != RetTy && !IsDerived)
129 return;
130
131 QualType ParentTy;
132 if (const auto *ParentCast = Nodes.getNodeAs<Expr>("parent_cast")) {
133 ParentTy = ParentCast->getType();
134 } else {
135 // IgnoreUnlessSpelledInSource prevents matching implicit casts
136 const TraversalKindScope TmpTraversalKind(*Result.Context, TK_AsIs);
137 for (const DynTypedNode Parent : Result.Context->getParents(*Call)) {
138 if (const auto *ParentCastExpr = Parent.get<CastExpr>()) {
139 ParentTy = ParentCastExpr->getType();
140 break;
141 }
142 }
143 }
144 if (!ParentTy.isNull()) {
145 const CXXRecordDecl *ParentDecl = ParentTy->getAsCXXRecordDecl();
146 if (FromDecl && ParentDecl) {
147 CXXBasePaths Paths(/*FindAmbiguities=*/true,
148 /*RecordPaths=*/false,
149 /*DetectVirtual=*/false);
150 const bool IsDerivedFromParent =
151 FromDecl && ParentDecl && FromDecl->isDerivedFrom(ParentDecl, Paths);
152 // For the following case a direct `cast<A>(d)` would be ambiguous:
153 // struct A {};
154 // struct B : A {};
155 // struct C : A {};
156 // struct D : B, C {};
157 // So we should not warn for `A *a = cast<C>(d)`.
158 if (IsDerivedFromParent &&
159 Paths.isAmbiguous(ParentTy->getCanonicalTypeUnqualified()))
160 return;
161 }
162 }
163
164 auto GetText = [&](SourceRange R) {
165 return Lexer::getSourceText(CharSourceRange::getTokenRange(R),
166 *Result.SourceManager, getLangOpts());
167 };
168 StringRef ArgText = GetText(Arg->getSourceRange());
169 diag(Call->getExprLoc(), "redundant use of '%0'")
170 << FuncName
171 << FixItHint::CreateReplacement(Call->getSourceRange(), ArgText);
172 // printing the canonical type for a template parameter prints as e.g.
173 // 'type-parameter-0-0'
174 const QualType DiagFromTy(ArgPointeeTy->getUnqualifiedDesugaredType(), 0);
175 diag(Arg->getExprLoc(),
176 "source expression has%select{| pointee}0 type %1%select{|, which is a "
177 "subtype of %3}2",
178 DiagnosticIDs::Note)
179 << Arg->getSourceRange() << ArgTy->isPointerType() << DiagFromTy
180 << (FromTy != RetTy) << RetTy;
181}
182
183} // namespace clang::tidy::llvm_check
void check(const ast_matchers::MatchFinder::MatchResult &Result) override
void registerMatchers(ast_matchers::MatchFinder *Finder) override
static bool isLLVMNamespace(NestedNameSpecifier NNS)
static QualType stripPointerOrReference(QualType Ty)
static constexpr StringRef FunctionNames[]
AST_MATCHER_P(Stmt, isStatementIdenticalToBoundNode, std::string, ID)
AST_MATCHER(BinaryOperator, isRelationalOperator)