forked from OSchip/llvm-project
73 lines
2.9 KiB
C++
73 lines
2.9 KiB
C++
//===--- UnnecessaryCopyInitialization.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 "UnnecessaryCopyInitialization.h"
|
|
|
|
#include "../utils/LexerUtils.h"
|
|
#include "../utils/Matchers.h"
|
|
|
|
namespace clang {
|
|
namespace tidy {
|
|
namespace performance {
|
|
|
|
using namespace ::clang::ast_matchers;
|
|
|
|
namespace {
|
|
AST_MATCHER(QualType, isPointerType) { return Node->isPointerType(); }
|
|
} // namespace
|
|
|
|
void UnnecessaryCopyInitialization::registerMatchers(
|
|
ast_matchers::MatchFinder *Finder) {
|
|
auto ConstReference = referenceType(pointee(qualType(isConstQualified())));
|
|
auto ConstOrConstReference =
|
|
allOf(anyOf(ConstReference, isConstQualified()),
|
|
unless(allOf(isPointerType(), unless(pointerType(pointee(qualType(
|
|
isConstQualified())))))));
|
|
// Match method call expressions where the this argument is a const
|
|
// type or const reference. This returned const reference is highly likely to
|
|
// outlive the local const reference of the variable being declared.
|
|
// The assumption is that the const reference being returned either points
|
|
// to a global static variable or to a member of the called object.
|
|
auto ConstRefReturningMethodCallOfConstParam = cxxMemberCallExpr(
|
|
callee(cxxMethodDecl(returns(ConstReference))),
|
|
on(declRefExpr(to(varDecl(hasType(qualType(ConstOrConstReference)))))));
|
|
auto ConstRefReturningFunctionCall =
|
|
callExpr(callee(functionDecl(returns(ConstReference))),
|
|
unless(callee(cxxMethodDecl())));
|
|
Finder->addMatcher(
|
|
varDecl(
|
|
hasLocalStorage(), hasType(isConstQualified()),
|
|
hasType(matchers::isExpensiveToCopy()),
|
|
hasInitializer(cxxConstructExpr(
|
|
hasDeclaration(cxxConstructorDecl(isCopyConstructor())),
|
|
hasArgument(0, anyOf(ConstRefReturningFunctionCall,
|
|
ConstRefReturningMethodCallOfConstParam)))))
|
|
.bind("varDecl"),
|
|
this);
|
|
}
|
|
|
|
void UnnecessaryCopyInitialization::check(
|
|
const ast_matchers::MatchFinder::MatchResult &Result) {
|
|
const auto *Var = Result.Nodes.getNodeAs<VarDecl>("varDecl");
|
|
SourceLocation AmpLocation = Var->getLocation();
|
|
auto Token = lexer_utils::getPreviousNonCommentToken(*Result.Context,
|
|
Var->getLocation());
|
|
if (!Token.is(tok::unknown)) {
|
|
AmpLocation = Token.getLocation().getLocWithOffset(Token.getLength());
|
|
}
|
|
diag(Var->getLocation(),
|
|
"the const qualified variable '%0' is copy-constructed from a "
|
|
"const reference; consider making it a const reference")
|
|
<< Var->getName() << FixItHint::CreateInsertion(AmpLocation, "&");
|
|
}
|
|
|
|
} // namespace performance
|
|
} // namespace tidy
|
|
} // namespace clang
|