blob: 6c60edeed26ddc59da763134e788d1d57ffa6c11 [file] [log] [blame]
Daniel Jasper9c6df882015-07-20 01:06:44 +00001//===--- UnusedParametersCheck.cpp - clang-tidy----------------------------===//
2//
3// The LLVM Compiler Infrastructure
4//
5// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
7//
8//===----------------------------------------------------------------------===//
9
10#include "UnusedParametersCheck.h"
11#include "clang/AST/ASTContext.h"
12#include "clang/ASTMatchers/ASTMatchFinder.h"
Samuel Benzaquenb4396272015-11-11 18:40:36 +000013#include "clang/Lex/Lexer.h"
Daniel Jasper9c6df882015-07-20 01:06:44 +000014
15using namespace clang::ast_matchers;
16
17namespace clang {
18namespace tidy {
Etienne Bergeron456177b2016-05-02 18:00:29 +000019namespace misc {
20
Haojian Wu7d158532016-04-01 07:57:30 +000021namespace {
22bool isOverrideMethod(const FunctionDecl *Function) {
23 if (const auto *MD = dyn_cast<CXXMethodDecl>(Function))
24 return MD->size_overridden_methods() > 0 || MD->hasAttr<OverrideAttr>();
25 return false;
26}
27} // namespace
Daniel Jasper9c6df882015-07-20 01:06:44 +000028
29void UnusedParametersCheck::registerMatchers(MatchFinder *Finder) {
Daniel Jasper9fe55a322015-07-27 13:46:37 +000030 Finder->addMatcher(functionDecl().bind("function"), this);
Daniel Jasper9c6df882015-07-20 01:06:44 +000031}
32
Samuel Benzaquenb4396272015-11-11 18:40:36 +000033template <typename T>
34static CharSourceRange removeNode(const MatchFinder::MatchResult &Result,
35 const T *PrevNode, const T *Node,
36 const T *NextNode) {
37 if (NextNode)
38 return CharSourceRange::getCharRange(Node->getLocStart(),
39 NextNode->getLocStart());
Daniel Jasper82efabb2015-07-20 03:42:38 +000040
Samuel Benzaquenb4396272015-11-11 18:40:36 +000041 if (PrevNode)
42 return CharSourceRange::getTokenRange(
43 Lexer::getLocForEndOfToken(PrevNode->getLocEnd(), 0,
44 *Result.SourceManager,
45 Result.Context->getLangOpts()),
46 Node->getLocEnd());
Daniel Jasper82efabb2015-07-20 03:42:38 +000047
Samuel Benzaquenb4396272015-11-11 18:40:36 +000048 return CharSourceRange::getTokenRange(Node->getSourceRange());
Daniel Jasper82efabb2015-07-20 03:42:38 +000049}
50
Samuel Benzaquenb4396272015-11-11 18:40:36 +000051static FixItHint removeParameter(const MatchFinder::MatchResult &Result,
52 const FunctionDecl *Function, unsigned Index) {
53 return FixItHint::CreateRemoval(removeNode(
54 Result, Index > 0 ? Function->getParamDecl(Index - 1) : nullptr,
55 Function->getParamDecl(Index),
56 Index + 1 < Function->getNumParams() ? Function->getParamDecl(Index + 1)
57 : nullptr));
58}
59
60static FixItHint removeArgument(const MatchFinder::MatchResult &Result,
61 const CallExpr *Call, unsigned Index) {
62 return FixItHint::CreateRemoval(removeNode(
63 Result, Index > 0 ? Call->getArg(Index - 1) : nullptr,
64 Call->getArg(Index),
65 Index + 1 < Call->getNumArgs() ? Call->getArg(Index + 1) : nullptr));
Daniel Jasper82efabb2015-07-20 03:42:38 +000066}
67
Daniel Jasper9fe55a322015-07-27 13:46:37 +000068void UnusedParametersCheck::warnOnUnusedParameter(
69 const MatchFinder::MatchResult &Result, const FunctionDecl *Function,
70 unsigned ParamIndex) {
71 const auto *Param = Function->getParamDecl(ParamIndex);
Benjamin Kramera62e2232016-04-07 14:55:25 +000072 auto MyDiag = diag(Param->getLocation(), "parameter %0 is unused") << Param;
Daniel Jasper9c6df882015-07-20 01:06:44 +000073
Haojian Wu7d158532016-04-01 07:57:30 +000074 auto DeclRefExpr =
75 declRefExpr(to(equalsNode(Function)),
76 unless(hasAncestor(callExpr(callee(equalsNode(Function))))));
Daniel Jasper82efabb2015-07-20 03:42:38 +000077
78 // Comment out parameter name for non-local functions.
Daniel Jasper542482e2015-07-28 13:19:12 +000079 if (Function->isExternallyVisible() ||
80 !Result.SourceManager->isInMainFile(Function->getLocation()) ||
Haojian Wu7d158532016-04-01 07:57:30 +000081 !ast_matchers::match(DeclRefExpr, *Result.Context).empty() ||
82 isOverrideMethod(Function)) {
Daniel Jasper82efabb2015-07-20 03:42:38 +000083 SourceRange RemovalRange(Param->getLocation(), Param->getLocEnd());
Daniel Jasper718029b2015-07-28 10:39:25 +000084 // Note: We always add a space before the '/*' to not accidentally create a
85 // '*/*' for pointer types, which doesn't start a comment. clang-format will
86 // clean this up afterwards.
Daniel Jasper82efabb2015-07-20 03:42:38 +000087 MyDiag << FixItHint::CreateReplacement(
88 RemovalRange, (Twine(" /*") + Param->getName() + "*/").str());
89 return;
90 }
91
Daniel Jasper82efabb2015-07-20 03:42:38 +000092 // Fix all redeclarations.
93 for (const FunctionDecl *FD : Function->redecls())
Daniel Jasperea223a72015-08-14 13:39:57 +000094 if (FD->param_size())
Samuel Benzaquenb4396272015-11-11 18:40:36 +000095 MyDiag << removeParameter(Result, FD, ParamIndex);
Daniel Jasper82efabb2015-07-20 03:42:38 +000096
97 // Fix all call sites.
98 auto CallMatches = ast_matchers::match(
99 decl(forEachDescendant(
100 callExpr(callee(functionDecl(equalsNode(Function)))).bind("x"))),
101 *Result.Context->getTranslationUnitDecl(), *Result.Context);
102 for (const auto &Match : CallMatches)
Samuel Benzaquenb4396272015-11-11 18:40:36 +0000103 MyDiag << removeArgument(Result, Match.getNodeAs<CallExpr>("x"),
104 ParamIndex);
Daniel Jasper9c6df882015-07-20 01:06:44 +0000105}
106
Daniel Jasper9fe55a322015-07-27 13:46:37 +0000107void UnusedParametersCheck::check(const MatchFinder::MatchResult &Result) {
108 const auto *Function = Result.Nodes.getNodeAs<FunctionDecl>("function");
Daniel Jasperb3a74c62015-08-10 15:45:46 +0000109 if (!Function->doesThisDeclarationHaveABody() ||
Mandeep Singh Grang7c7ea7d2016-11-08 07:50:19 +0000110 !Function->hasWrittenPrototype() || Function->isTemplateInstantiation())
Daniel Jasper9fe55a322015-07-27 13:46:37 +0000111 return;
Daniel Jasperf6311912015-09-22 09:20:20 +0000112 if (const auto *Method = dyn_cast<CXXMethodDecl>(Function))
113 if (Method->isLambdaStaticInvoker())
114 return;
Daniel Jasper9fe55a322015-07-27 13:46:37 +0000115 for (unsigned i = 0, e = Function->getNumParams(); i != e; ++i) {
116 const auto *Param = Function->getParamDecl(i);
117 if (Param->isUsed() || Param->isReferenced() || !Param->getDeclName() ||
118 Param->hasAttr<UnusedAttr>())
119 continue;
120 warnOnUnusedParameter(Result, Function, i);
121 }
122}
123
Etienne Bergeron456177b2016-05-02 18:00:29 +0000124} // namespace misc
Daniel Jasper9c6df882015-07-20 01:06:44 +0000125} // namespace tidy
126} // namespace clang