blob: c696c4c65c7e971adcec2b4cfe1e0e8e21012313 [file] [log] [blame]
Aaron Ballman9392ced2015-08-20 15:52:52 +00001//===--- MoveConstructorInitCheck.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 "MoveConstructorInitCheck.h"
Aaron Ballmanfc4e0422015-10-06 16:27:03 +000011#include "../utils/Matchers.h"
Aaron Ballman9392ced2015-08-20 15:52:52 +000012#include "clang/AST/ASTContext.h"
13#include "clang/ASTMatchers/ASTMatchFinder.h"
Aaron Ballmanfc4e0422015-10-06 16:27:03 +000014#include "clang/Frontend/CompilerInstance.h"
15#include "clang/Lex/Lexer.h"
16#include "clang/Lex/Preprocessor.h"
Aaron Ballman9392ced2015-08-20 15:52:52 +000017
18using namespace clang::ast_matchers;
19
20namespace clang {
21namespace tidy {
Etienne Bergeron456177b2016-05-02 18:00:29 +000022namespace misc {
Aaron Ballman9392ced2015-08-20 15:52:52 +000023
Aaron Ballmanfc4e0422015-10-06 16:27:03 +000024namespace {
25
26unsigned int
27parmVarDeclRefExprOccurences(const ParmVarDecl &MovableParam,
28 const CXXConstructorDecl &ConstructorDecl,
29 ASTContext &Context) {
30 unsigned int Occurrences = 0;
31 auto AllDeclRefs =
32 findAll(declRefExpr(to(parmVarDecl(equalsNode(&MovableParam)))));
33 Occurrences += match(AllDeclRefs, *ConstructorDecl.getBody(), Context).size();
34 for (const auto *Initializer : ConstructorDecl.inits()) {
35 Occurrences += match(AllDeclRefs, *Initializer->getInit(), Context).size();
36 }
37 return Occurrences;
38}
39
40} // namespace
41
42MoveConstructorInitCheck::MoveConstructorInitCheck(StringRef Name,
43 ClangTidyContext *Context)
44 : ClangTidyCheck(Name, Context),
Etienne Bergeron2a4c00f2016-05-03 02:54:05 +000045 IncludeStyle(utils::IncludeSorter::parseIncludeStyle(
Aaron Ballmanc00ad6c2016-01-08 15:50:51 +000046 Options.get("IncludeStyle", "llvm"))),
Alexander Kornienko5d08bb72016-05-20 13:42:40 +000047 UseCERTSemantics(Options.get("UseCERTSemantics", 0) != 0) {}
Aaron Ballmanfc4e0422015-10-06 16:27:03 +000048
Aaron Ballman9392ced2015-08-20 15:52:52 +000049void MoveConstructorInitCheck::registerMatchers(MatchFinder *Finder) {
Aaron Ballman327e97b2015-08-28 19:27:19 +000050 // Only register the matchers for C++11; the functionality currently does not
51 // provide any benefit to other languages, despite being benign.
Aaron Ballmanbf891092015-08-31 15:28:57 +000052 if (!getLangOpts().CPlusPlus11)
53 return;
54
55 Finder->addMatcher(
Aaron Ballmanb9ea09c2015-09-17 13:31:25 +000056 cxxConstructorDecl(
57 unless(isImplicit()),
58 allOf(isMoveConstructor(),
59 hasAnyConstructorInitializer(
60 cxxCtorInitializer(
61 withInitializer(cxxConstructExpr(hasDeclaration(
62 cxxConstructorDecl(isCopyConstructor())
63 .bind("ctor")))))
Aaron Ballmanfc4e0422015-10-06 16:27:03 +000064 .bind("move-init")))),
65 this);
66
67 auto NonConstValueMovableAndExpensiveToCopy =
68 qualType(allOf(unless(pointerType()), unless(isConstQualified()),
69 hasDeclaration(cxxRecordDecl(hasMethod(cxxConstructorDecl(
70 isMoveConstructor(), unless(isDeleted()))))),
71 matchers::isExpensiveToCopy()));
Aaron Ballmanc00ad6c2016-01-08 15:50:51 +000072
73 // This checker is also used to implement cert-oop11-cpp, but when using that
74 // form of the checker, we do not want to diagnose movable parameters.
Alexander Kornienko5d08bb72016-05-20 13:42:40 +000075 if (!UseCERTSemantics) {
Aaron Ballmanc00ad6c2016-01-08 15:50:51 +000076 Finder->addMatcher(
77 cxxConstructorDecl(
78 allOf(
79 unless(isMoveConstructor()),
80 hasAnyConstructorInitializer(withInitializer(cxxConstructExpr(
81 hasDeclaration(cxxConstructorDecl(isCopyConstructor())),
82 hasArgument(
83 0,
84 declRefExpr(
85 to(parmVarDecl(
86 hasType(
87 NonConstValueMovableAndExpensiveToCopy))
88 .bind("movable-param")))
89 .bind("init-arg")))))))
90 .bind("ctor-decl"),
91 this);
Alexander Kornienko5d08bb72016-05-20 13:42:40 +000092 }
Aaron Ballman9392ced2015-08-20 15:52:52 +000093}
94
95void MoveConstructorInitCheck::check(const MatchFinder::MatchResult &Result) {
Aaron Ballmanfc4e0422015-10-06 16:27:03 +000096 if (Result.Nodes.getNodeAs<CXXCtorInitializer>("move-init") != nullptr)
97 handleMoveConstructor(Result);
98 if (Result.Nodes.getNodeAs<ParmVarDecl>("movable-param") != nullptr)
99 handleParamNotMoved(Result);
100}
101
102void MoveConstructorInitCheck::handleParamNotMoved(
103 const MatchFinder::MatchResult &Result) {
104 const auto *MovableParam =
105 Result.Nodes.getNodeAs<ParmVarDecl>("movable-param");
106 const auto *ConstructorDecl =
107 Result.Nodes.getNodeAs<CXXConstructorDecl>("ctor-decl");
108 const auto *InitArg = Result.Nodes.getNodeAs<DeclRefExpr>("init-arg");
109 // If the parameter is referenced more than once it is not safe to move it.
110 if (parmVarDeclRefExprOccurences(*MovableParam, *ConstructorDecl,
111 *Result.Context) > 1)
112 return;
Felix Berger99a0ddd2016-05-03 23:07:44 +0000113 auto DiagOut = diag(InitArg->getLocStart(),
114 "value argument %0 can be moved to avoid copy")
115 << MovableParam;
Aaron Ballmanfc4e0422015-10-06 16:27:03 +0000116 DiagOut << FixItHint::CreateReplacement(
117 InitArg->getSourceRange(),
118 (Twine("std::move(") + MovableParam->getName() + ")").str());
119 if (auto IncludeFixit = Inserter->CreateIncludeInsertion(
120 Result.SourceManager->getFileID(InitArg->getLocStart()), "utility",
121 /*IsAngled=*/true)) {
122 DiagOut << *IncludeFixit;
123 }
124}
125
126void MoveConstructorInitCheck::handleMoveConstructor(
127 const MatchFinder::MatchResult &Result) {
Aaron Ballman9392ced2015-08-20 15:52:52 +0000128 const auto *CopyCtor = Result.Nodes.getNodeAs<CXXConstructorDecl>("ctor");
Aaron Ballmanfc4e0422015-10-06 16:27:03 +0000129 const auto *Initializer = Result.Nodes.getNodeAs<CXXCtorInitializer>("move-init");
Aaron Ballman9392ced2015-08-20 15:52:52 +0000130
131 // Do not diagnose if the expression used to perform the initialization is a
132 // trivially-copyable type.
133 QualType QT = Initializer->getInit()->getType();
134 if (QT.isTriviallyCopyableType(*Result.Context))
135 return;
Kirill Bobyrev11cea452016-08-01 12:06:18 +0000136
Aaron Ballman9392ced2015-08-20 15:52:52 +0000137 const auto *RD = QT->getAsCXXRecordDecl();
138 if (RD && RD->isTriviallyCopyable())
139 return;
140
141 // Diagnose when the class type has a move constructor available, but the
142 // ctor-initializer uses the copy constructor instead.
143 const CXXConstructorDecl *Candidate = nullptr;
144 for (const auto *Ctor : CopyCtor->getParent()->ctors()) {
145 if (Ctor->isMoveConstructor() && Ctor->getAccess() <= AS_protected &&
146 !Ctor->isDeleted()) {
147 // The type has a move constructor that is at least accessible to the
148 // initializer.
149 //
150 // FIXME: Determine whether the move constructor is a viable candidate
151 // for the ctor-initializer, perhaps provide a fixit that suggests
152 // using std::move().
153 Candidate = Ctor;
154 break;
155 }
156 }
157
158 if (Candidate) {
159 // There's a move constructor candidate that the caller probably intended
160 // to call instead.
161 diag(Initializer->getSourceLocation(),
162 "move constructor initializes %0 by calling a copy constructor")
163 << (Initializer->isBaseInitializer() ? "base class" : "class member");
164 diag(CopyCtor->getLocation(), "copy constructor being called",
165 DiagnosticIDs::Note);
166 diag(Candidate->getLocation(), "candidate move constructor here",
167 DiagnosticIDs::Note);
168 }
169}
170
Aaron Ballmanfc4e0422015-10-06 16:27:03 +0000171void MoveConstructorInitCheck::registerPPCallbacks(CompilerInstance &Compiler) {
Etienne Bergeron2a4c00f2016-05-03 02:54:05 +0000172 Inserter.reset(new utils::IncludeInserter(
173 Compiler.getSourceManager(), Compiler.getLangOpts(), IncludeStyle));
Aaron Ballmanfc4e0422015-10-06 16:27:03 +0000174 Compiler.getPreprocessor().addPPCallbacks(Inserter->CreatePPCallbacks());
175}
176
177void MoveConstructorInitCheck::storeOptions(ClangTidyOptions::OptionMap &Opts) {
Etienne Bergeron2a4c00f2016-05-03 02:54:05 +0000178 Options.store(Opts, "IncludeStyle",
179 utils::IncludeSorter::toString(IncludeStyle));
Alexander Kornienko5d08bb72016-05-20 13:42:40 +0000180 Options.store(Opts, "UseCERTSemantics", UseCERTSemantics ? 1 : 0);
Aaron Ballmanfc4e0422015-10-06 16:27:03 +0000181}
182
Etienne Bergeron456177b2016-05-02 18:00:29 +0000183} // namespace misc
Aaron Ballman9392ced2015-08-20 15:52:52 +0000184} // namespace tidy
185} // namespace clang