You've already forked linux-packaging-mono
acceptance-tests
data
debian
docs
external
Newtonsoft.Json
api-doc-tools
api-snapshot
aspnetwebstack
bdwgc
binary-reference-assemblies
bockbuild
boringssl
cecil
cecil-legacy
corefx
corert
helix-binaries
ikdasm
ikvm
illinker-test-assets
linker
llvm-project
clang
clang-tools-extra
change-namespace
clang-apply-replacements
clang-move
clang-query
clang-reorder-fields
clang-tidy
android
boost
bugprone
cert
cppcoreguidelines
fuchsia
google
hicpp
llvm
misc
modernize
mpi
objc
performance
CMakeLists.txt
FasterStringFindCheck.cpp
FasterStringFindCheck.h
ForRangeCopyCheck.cpp
ForRangeCopyCheck.h
ImplicitConversionInLoopCheck.cpp
ImplicitConversionInLoopCheck.h
InefficientAlgorithmCheck.cpp
InefficientAlgorithmCheck.h
InefficientStringConcatenationCheck.cpp
InefficientStringConcatenationCheck.h
InefficientVectorOperationCheck.cpp
InefficientVectorOperationCheck.h
MoveConstArgCheck.cpp
MoveConstArgCheck.h
MoveConstructorInitCheck.cpp
MoveConstructorInitCheck.h
NoexceptMoveConstructorCheck.cpp
NoexceptMoveConstructorCheck.h
PerformanceTidyModule.cpp
TypePromotionInMathFnCheck.cpp
TypePromotionInMathFnCheck.h
UnnecessaryCopyInitialization.cpp
UnnecessaryCopyInitialization.h
UnnecessaryValueParamCheck.cpp
UnnecessaryValueParamCheck.h
plugin
readability
tool
utils
CMakeLists.txt
ClangTidy.cpp
ClangTidy.h
ClangTidyDiagnosticConsumer.cpp
ClangTidyDiagnosticConsumer.h
ClangTidyModule.cpp
ClangTidyModule.h
ClangTidyModuleRegistry.h
ClangTidyOptions.cpp
ClangTidyOptions.h
add_new_check.py
rename_check.py
clang-tidy-vs
clangd
docs
include-fixer
modularize
pp-trace
test
tool-template
unittests
.arcconfig
.gitignore
CMakeLists.txt
CODE_OWNERS.TXT
LICENSE.TXT
README.txt
compiler-rt
libcxx
libcxxabi
libunwind
lld
lldb
llvm
openmp
polly
nuget-buildtasks
nunit-lite
roslyn-binaries
rx
xunit-binaries
how-to-bump-roslyn-binaries.md
ikvm-native
llvm
m4
man
mcs
mk
mono
msvc
netcore
po
runtime
samples
scripts
support
tools
COPYING.LIB
LICENSE
Makefile.am
Makefile.in
NEWS
README.md
acinclude.m4
aclocal.m4
autogen.sh
code_of_conduct.md
compile
config.guess
config.h.in
config.rpath
config.sub
configure.REMOVED.git-id
configure.ac.REMOVED.git-id
depcomp
install-sh
ltmain.sh.REMOVED.git-id
missing
mkinstalldirs
mono-uninstalled.pc.in
test-driver
winconfig.h
204 lines
8.5 KiB
C++
204 lines
8.5 KiB
C++
//===--- UnnecessaryValueParamCheck.cpp - clang-tidy-----------------------===//
|
|
//
|
|
// The LLVM Compiler Infrastructure
|
|
//
|
|
// This file is distributed under the University of Illinois Open Source
|
|
// License. See LICENSE.TXT for details.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#include "UnnecessaryValueParamCheck.h"
|
|
|
|
#include "../utils/DeclRefExprUtils.h"
|
|
#include "../utils/FixItHintUtils.h"
|
|
#include "../utils/Matchers.h"
|
|
#include "../utils/TypeTraits.h"
|
|
#include "clang/Frontend/CompilerInstance.h"
|
|
#include "clang/Lex/Lexer.h"
|
|
#include "clang/Lex/Preprocessor.h"
|
|
|
|
using namespace clang::ast_matchers;
|
|
|
|
namespace clang {
|
|
namespace tidy {
|
|
namespace performance {
|
|
|
|
namespace {
|
|
|
|
std::string paramNameOrIndex(StringRef Name, size_t Index) {
|
|
return (Name.empty() ? llvm::Twine('#') + llvm::Twine(Index + 1)
|
|
: llvm::Twine('\'') + Name + llvm::Twine('\''))
|
|
.str();
|
|
}
|
|
|
|
template <typename S>
|
|
bool isSubset(const S &SubsetCandidate, const S &SupersetCandidate) {
|
|
for (const auto &E : SubsetCandidate)
|
|
if (SupersetCandidate.count(E) == 0)
|
|
return false;
|
|
return true;
|
|
}
|
|
|
|
bool isReferencedOutsideOfCallExpr(const FunctionDecl &Function,
|
|
ASTContext &Context) {
|
|
auto Matches = match(declRefExpr(to(functionDecl(equalsNode(&Function))),
|
|
unless(hasAncestor(callExpr()))),
|
|
Context);
|
|
return !Matches.empty();
|
|
}
|
|
|
|
bool hasLoopStmtAncestor(const DeclRefExpr &DeclRef, const Decl &Decl,
|
|
ASTContext &Context) {
|
|
auto Matches =
|
|
match(decl(forEachDescendant(declRefExpr(
|
|
equalsNode(&DeclRef),
|
|
unless(hasAncestor(stmt(anyOf(forStmt(), cxxForRangeStmt(),
|
|
whileStmt(), doStmt()))))))),
|
|
Decl, Context);
|
|
return Matches.empty();
|
|
}
|
|
|
|
bool isExplicitTemplateSpecialization(const FunctionDecl &Function) {
|
|
if (const auto *SpecializationInfo = Function.getTemplateSpecializationInfo())
|
|
if (SpecializationInfo->getTemplateSpecializationKind() ==
|
|
TSK_ExplicitSpecialization)
|
|
return true;
|
|
if (const auto *Method = llvm::dyn_cast<CXXMethodDecl>(&Function))
|
|
if (Method->getTemplatedKind() == FunctionDecl::TK_MemberSpecialization &&
|
|
Method->getMemberSpecializationInfo()->isExplicitSpecialization())
|
|
return true;
|
|
return false;
|
|
}
|
|
|
|
} // namespace
|
|
|
|
UnnecessaryValueParamCheck::UnnecessaryValueParamCheck(
|
|
StringRef Name, ClangTidyContext *Context)
|
|
: ClangTidyCheck(Name, Context),
|
|
IncludeStyle(utils::IncludeSorter::parseIncludeStyle(
|
|
Options.getLocalOrGlobal("IncludeStyle", "llvm"))) {}
|
|
|
|
void UnnecessaryValueParamCheck::registerMatchers(MatchFinder *Finder) {
|
|
const auto ExpensiveValueParamDecl =
|
|
parmVarDecl(hasType(hasCanonicalType(allOf(
|
|
unless(referenceType()), matchers::isExpensiveToCopy()))),
|
|
decl().bind("param"));
|
|
Finder->addMatcher(
|
|
functionDecl(hasBody(stmt()), isDefinition(), unless(isImplicit()),
|
|
unless(cxxMethodDecl(anyOf(isOverride(), isFinal()))),
|
|
has(typeLoc(forEach(ExpensiveValueParamDecl))),
|
|
unless(isInstantiated()), decl().bind("functionDecl")),
|
|
this);
|
|
}
|
|
|
|
void UnnecessaryValueParamCheck::check(const MatchFinder::MatchResult &Result) {
|
|
const auto *Param = Result.Nodes.getNodeAs<ParmVarDecl>("param");
|
|
const auto *Function = Result.Nodes.getNodeAs<FunctionDecl>("functionDecl");
|
|
const size_t Index = std::find(Function->parameters().begin(),
|
|
Function->parameters().end(), Param) -
|
|
Function->parameters().begin();
|
|
bool IsConstQualified =
|
|
Param->getType().getCanonicalType().isConstQualified();
|
|
|
|
auto AllDeclRefExprs = utils::decl_ref_expr::allDeclRefExprs(
|
|
*Param, *Function, *Result.Context);
|
|
auto ConstDeclRefExprs = utils::decl_ref_expr::constReferenceDeclRefExprs(
|
|
*Param, *Function, *Result.Context);
|
|
|
|
// Do not trigger on non-const value parameters when they are not only used as
|
|
// const.
|
|
if (!isSubset(AllDeclRefExprs, ConstDeclRefExprs))
|
|
return;
|
|
|
|
// If the parameter is non-const, check if it has a move constructor and is
|
|
// only referenced once to copy-construct another object or whether it has a
|
|
// move assignment operator and is only referenced once when copy-assigned.
|
|
// In this case wrap DeclRefExpr with std::move() to avoid the unnecessary
|
|
// copy.
|
|
if (!IsConstQualified && AllDeclRefExprs.size() == 1) {
|
|
auto CanonicalType = Param->getType().getCanonicalType();
|
|
const auto &DeclRefExpr = **AllDeclRefExprs.begin();
|
|
|
|
if (!hasLoopStmtAncestor(DeclRefExpr, *Function, *Result.Context) &&
|
|
((utils::type_traits::hasNonTrivialMoveConstructor(CanonicalType) &&
|
|
utils::decl_ref_expr::isCopyConstructorArgument(
|
|
DeclRefExpr, *Function, *Result.Context)) ||
|
|
(utils::type_traits::hasNonTrivialMoveAssignment(CanonicalType) &&
|
|
utils::decl_ref_expr::isCopyAssignmentArgument(
|
|
DeclRefExpr, *Function, *Result.Context)))) {
|
|
handleMoveFix(*Param, DeclRefExpr, *Result.Context);
|
|
return;
|
|
}
|
|
}
|
|
|
|
auto Diag =
|
|
diag(Param->getLocation(),
|
|
IsConstQualified ? "the const qualified parameter %0 is "
|
|
"copied for each invocation; consider "
|
|
"making it a reference"
|
|
: "the parameter %0 is copied for each "
|
|
"invocation but only used as a const reference; "
|
|
"consider making it a const reference")
|
|
<< paramNameOrIndex(Param->getName(), Index);
|
|
// Do not propose fixes when:
|
|
// 1. the ParmVarDecl is in a macro, since we cannot place them correctly
|
|
// 2. the function is virtual as it might break overrides
|
|
// 3. the function is referenced outside of a call expression within the
|
|
// compilation unit as the signature change could introduce build errors.
|
|
// 4. the function is an explicit template specialization.
|
|
const auto *Method = llvm::dyn_cast<CXXMethodDecl>(Function);
|
|
if (Param->getLocStart().isMacroID() || (Method && Method->isVirtual()) ||
|
|
isReferencedOutsideOfCallExpr(*Function, *Result.Context) ||
|
|
isExplicitTemplateSpecialization(*Function))
|
|
return;
|
|
for (const auto *FunctionDecl = Function; FunctionDecl != nullptr;
|
|
FunctionDecl = FunctionDecl->getPreviousDecl()) {
|
|
const auto &CurrentParam = *FunctionDecl->getParamDecl(Index);
|
|
Diag << utils::fixit::changeVarDeclToReference(CurrentParam,
|
|
*Result.Context);
|
|
// The parameter of each declaration needs to be checked individually as to
|
|
// whether it is const or not as constness can differ between definition and
|
|
// declaration.
|
|
if (!CurrentParam.getType().getCanonicalType().isConstQualified())
|
|
Diag << utils::fixit::changeVarDeclToConst(CurrentParam);
|
|
}
|
|
}
|
|
|
|
void UnnecessaryValueParamCheck::registerPPCallbacks(
|
|
CompilerInstance &Compiler) {
|
|
Inserter.reset(new utils::IncludeInserter(
|
|
Compiler.getSourceManager(), Compiler.getLangOpts(), IncludeStyle));
|
|
Compiler.getPreprocessor().addPPCallbacks(Inserter->CreatePPCallbacks());
|
|
}
|
|
|
|
void UnnecessaryValueParamCheck::storeOptions(
|
|
ClangTidyOptions::OptionMap &Opts) {
|
|
Options.store(Opts, "IncludeStyle",
|
|
utils::IncludeSorter::toString(IncludeStyle));
|
|
}
|
|
|
|
void UnnecessaryValueParamCheck::handleMoveFix(const ParmVarDecl &Var,
|
|
const DeclRefExpr &CopyArgument,
|
|
const ASTContext &Context) {
|
|
auto Diag = diag(CopyArgument.getLocStart(),
|
|
"parameter %0 is passed by value and only copied once; "
|
|
"consider moving it to avoid unnecessary copies")
|
|
<< &Var;
|
|
// Do not propose fixes in macros since we cannot place them correctly.
|
|
if (CopyArgument.getLocStart().isMacroID())
|
|
return;
|
|
const auto &SM = Context.getSourceManager();
|
|
auto EndLoc = Lexer::getLocForEndOfToken(CopyArgument.getLocation(), 0, SM,
|
|
Context.getLangOpts());
|
|
Diag << FixItHint::CreateInsertion(CopyArgument.getLocStart(), "std::move(")
|
|
<< FixItHint::CreateInsertion(EndLoc, ")");
|
|
if (auto IncludeFixit = Inserter->CreateIncludeInsertion(
|
|
SM.getFileID(CopyArgument.getLocStart()), "utility",
|
|
/*IsAngled=*/true))
|
|
Diag << *IncludeFixit;
|
|
}
|
|
|
|
} // namespace performance
|
|
} // namespace tidy
|
|
} // namespace clang
|