forked from OSchip/llvm-project
716 lines
25 KiB
C++
716 lines
25 KiB
C++
// RUN: %check_clang_tidy -std=c++17 %s performance-unnecessary-copy-initialization %t
|
|
|
|
template <typename T>
|
|
struct Iterator {
|
|
void operator++();
|
|
const T &operator*() const;
|
|
bool operator!=(const Iterator &) const;
|
|
typedef const T &const_reference;
|
|
};
|
|
|
|
struct ExpensiveToCopyType {
|
|
ExpensiveToCopyType();
|
|
virtual ~ExpensiveToCopyType();
|
|
const ExpensiveToCopyType &reference() const;
|
|
const ExpensiveToCopyType *pointer() const;
|
|
Iterator<ExpensiveToCopyType> begin() const;
|
|
Iterator<ExpensiveToCopyType> end() const;
|
|
void nonConstMethod();
|
|
bool constMethod() const;
|
|
template <typename A>
|
|
const A &templatedAccessor() const;
|
|
operator int() const; // Implicit conversion to int.
|
|
};
|
|
|
|
struct TrivialToCopyType {
|
|
const TrivialToCopyType &reference() const;
|
|
};
|
|
|
|
struct WeirdCopyCtorType {
|
|
WeirdCopyCtorType();
|
|
WeirdCopyCtorType(const WeirdCopyCtorType &w, bool oh_yes = true);
|
|
|
|
void nonConstMethod();
|
|
bool constMethod() const;
|
|
};
|
|
|
|
ExpensiveToCopyType global_expensive_to_copy_type;
|
|
|
|
const ExpensiveToCopyType &ExpensiveTypeReference();
|
|
const ExpensiveToCopyType &freeFunctionWithArg(const ExpensiveToCopyType &);
|
|
const ExpensiveToCopyType &freeFunctionWithDefaultArg(
|
|
const ExpensiveToCopyType *arg = nullptr);
|
|
const TrivialToCopyType &TrivialTypeReference();
|
|
|
|
void mutate(ExpensiveToCopyType &);
|
|
void mutate(ExpensiveToCopyType *);
|
|
void useAsConstPointer(const ExpensiveToCopyType *);
|
|
void useAsConstReference(const ExpensiveToCopyType &);
|
|
void useByValue(ExpensiveToCopyType);
|
|
|
|
void PositiveFunctionCall() {
|
|
const auto AutoAssigned = ExpensiveTypeReference();
|
|
// CHECK-MESSAGES: [[@LINE-1]]:14: warning: the const qualified variable 'AutoAssigned' is copy-constructed from a const reference; consider making it a const reference [performance-unnecessary-copy-initialization]
|
|
// CHECK-FIXES: const auto& AutoAssigned = ExpensiveTypeReference();
|
|
AutoAssigned.constMethod();
|
|
|
|
const auto AutoCopyConstructed(ExpensiveTypeReference());
|
|
// CHECK-MESSAGES: [[@LINE-1]]:14: warning: the const qualified variable 'AutoCopyConstructed'
|
|
// CHECK-FIXES: const auto& AutoCopyConstructed(ExpensiveTypeReference());
|
|
AutoCopyConstructed.constMethod();
|
|
|
|
const ExpensiveToCopyType VarAssigned = ExpensiveTypeReference();
|
|
// CHECK-MESSAGES: [[@LINE-1]]:29: warning: the const qualified variable 'VarAssigned'
|
|
// CHECK-FIXES: const ExpensiveToCopyType& VarAssigned = ExpensiveTypeReference();
|
|
VarAssigned.constMethod();
|
|
|
|
const ExpensiveToCopyType VarCopyConstructed(ExpensiveTypeReference());
|
|
// CHECK-MESSAGES: [[@LINE-1]]:29: warning: the const qualified variable 'VarCopyConstructed'
|
|
// CHECK-FIXES: const ExpensiveToCopyType& VarCopyConstructed(ExpensiveTypeReference());
|
|
VarCopyConstructed.constMethod();
|
|
}
|
|
|
|
void PositiveMethodCallConstReferenceParam(const ExpensiveToCopyType &Obj) {
|
|
const auto AutoAssigned = Obj.reference();
|
|
// CHECK-MESSAGES: [[@LINE-1]]:14: warning: the const qualified variable 'AutoAssigned'
|
|
// CHECK-FIXES: const auto& AutoAssigned = Obj.reference();
|
|
AutoAssigned.constMethod();
|
|
|
|
const auto AutoCopyConstructed(Obj.reference());
|
|
// CHECK-MESSAGES: [[@LINE-1]]:14: warning: the const qualified variable 'AutoCopyConstructed'
|
|
// CHECK-FIXES: const auto& AutoCopyConstructed(Obj.reference());
|
|
AutoCopyConstructed.constMethod();
|
|
|
|
const ExpensiveToCopyType VarAssigned = Obj.reference();
|
|
// CHECK-MESSAGES: [[@LINE-1]]:29: warning: the const qualified variable 'VarAssigned'
|
|
// CHECK-FIXES: const ExpensiveToCopyType& VarAssigned = Obj.reference();
|
|
VarAssigned.constMethod();
|
|
|
|
const ExpensiveToCopyType VarCopyConstructed(Obj.reference());
|
|
// CHECK-MESSAGES: [[@LINE-1]]:29: warning: the const qualified variable 'VarCopyConstructed'
|
|
// CHECK-FIXES: const ExpensiveToCopyType& VarCopyConstructed(Obj.reference());
|
|
VarCopyConstructed.constMethod();
|
|
}
|
|
|
|
void PositiveMethodCallConstParam(const ExpensiveToCopyType Obj) {
|
|
const auto AutoAssigned = Obj.reference();
|
|
// CHECK-MESSAGES: [[@LINE-1]]:14: warning: the const qualified variable 'AutoAssigned'
|
|
// CHECK-FIXES: const auto& AutoAssigned = Obj.reference();
|
|
AutoAssigned.constMethod();
|
|
|
|
const auto AutoCopyConstructed(Obj.reference());
|
|
// CHECK-MESSAGES: [[@LINE-1]]:14: warning: the const qualified variable 'AutoCopyConstructed'
|
|
// CHECK-FIXES: const auto& AutoCopyConstructed(Obj.reference());
|
|
AutoCopyConstructed.constMethod();
|
|
|
|
const ExpensiveToCopyType VarAssigned = Obj.reference();
|
|
// CHECK-MESSAGES: [[@LINE-1]]:29: warning: the const qualified variable 'VarAssigned'
|
|
// CHECK-FIXES: const ExpensiveToCopyType& VarAssigned = Obj.reference();
|
|
VarAssigned.constMethod();
|
|
|
|
const ExpensiveToCopyType VarCopyConstructed(Obj.reference());
|
|
// CHECK-MESSAGES: [[@LINE-1]]:29: warning: the const qualified variable 'VarCopyConstructed'
|
|
// CHECK-FIXES: const ExpensiveToCopyType& VarCopyConstructed(Obj.reference());
|
|
VarCopyConstructed.constMethod();
|
|
}
|
|
|
|
void PositiveMethodCallConstPointerParam(const ExpensiveToCopyType *const Obj) {
|
|
const auto AutoAssigned = Obj->reference();
|
|
// CHECK-MESSAGES: [[@LINE-1]]:14: warning: the const qualified variable 'AutoAssigned'
|
|
// CHECK-FIXES: const auto& AutoAssigned = Obj->reference();
|
|
AutoAssigned.constMethod();
|
|
|
|
const auto AutoCopyConstructed(Obj->reference());
|
|
// CHECK-MESSAGES: [[@LINE-1]]:14: warning: the const qualified variable 'AutoCopyConstructed'
|
|
// CHECK-FIXES: const auto& AutoCopyConstructed(Obj->reference());
|
|
AutoCopyConstructed.constMethod();
|
|
|
|
const ExpensiveToCopyType VarAssigned = Obj->reference();
|
|
// CHECK-MESSAGES: [[@LINE-1]]:29: warning: the const qualified variable 'VarAssigned'
|
|
// CHECK-FIXES: const ExpensiveToCopyType& VarAssigned = Obj->reference();
|
|
VarAssigned.constMethod();
|
|
|
|
const ExpensiveToCopyType VarCopyConstructed(Obj->reference());
|
|
// CHECK-MESSAGES: [[@LINE-1]]:29: warning: the const qualified variable 'VarCopyConstructed'
|
|
// CHECK-FIXES: const ExpensiveToCopyType& VarCopyConstructed(Obj->reference());
|
|
VarCopyConstructed.constMethod();
|
|
}
|
|
|
|
void PositiveLocalConstValue() {
|
|
const ExpensiveToCopyType Obj;
|
|
const auto UnnecessaryCopy = Obj.reference();
|
|
// CHECK-MESSAGES: [[@LINE-1]]:14: warning: the const qualified variable 'UnnecessaryCopy'
|
|
// CHECK-FIXES: const auto& UnnecessaryCopy = Obj.reference();
|
|
UnnecessaryCopy.constMethod();
|
|
}
|
|
|
|
void PositiveLocalConstRef() {
|
|
const ExpensiveToCopyType Obj;
|
|
const ExpensiveToCopyType &ConstReference = Obj.reference();
|
|
const auto UnnecessaryCopy = ConstReference.reference();
|
|
// CHECK-MESSAGES: [[@LINE-1]]:14: warning: the const qualified variable 'UnnecessaryCopy'
|
|
// CHECK-FIXES: const auto& UnnecessaryCopy = ConstReference.reference();
|
|
UnnecessaryCopy.constMethod();
|
|
}
|
|
|
|
void PositiveLocalConstPointer() {
|
|
const ExpensiveToCopyType Obj;
|
|
const ExpensiveToCopyType *const ConstPointer = &Obj;
|
|
const auto UnnecessaryCopy = ConstPointer->reference();
|
|
// CHECK-MESSAGES: [[@LINE-1]]:14: warning: the const qualified variable 'UnnecessaryCopy'
|
|
// CHECK-FIXES: const auto& UnnecessaryCopy = ConstPointer->reference();
|
|
UnnecessaryCopy.constMethod();
|
|
}
|
|
|
|
void NegativeFunctionCallTrivialType() {
|
|
const auto AutoAssigned = TrivialTypeReference();
|
|
const auto AutoCopyConstructed(TrivialTypeReference());
|
|
const TrivialToCopyType VarAssigned = TrivialTypeReference();
|
|
const TrivialToCopyType VarCopyConstructed(TrivialTypeReference());
|
|
}
|
|
|
|
void NegativeStaticLocalVar(const ExpensiveToCopyType &Obj) {
|
|
static const auto StaticVar = Obj.reference();
|
|
}
|
|
|
|
void PositiveFunctionCallExpensiveTypeNonConstVariable() {
|
|
auto AutoAssigned = ExpensiveTypeReference();
|
|
// CHECK-MESSAGES: [[@LINE-1]]:8: warning: the variable 'AutoAssigned' is copy-constructed from a const reference but is only used as const reference; consider making it a const reference [performance-unnecessary-copy-initialization]
|
|
// CHECK-FIXES: const auto& AutoAssigned = ExpensiveTypeReference();
|
|
AutoAssigned.constMethod();
|
|
|
|
auto AutoCopyConstructed(ExpensiveTypeReference());
|
|
// CHECK-MESSAGES: [[@LINE-1]]:8: warning: the variable 'AutoCopyConstructed'
|
|
// CHECK-FIXES: const auto& AutoCopyConstructed(ExpensiveTypeReference());
|
|
AutoCopyConstructed.constMethod();
|
|
|
|
ExpensiveToCopyType VarAssigned = ExpensiveTypeReference();
|
|
// CHECK-MESSAGES: [[@LINE-1]]:23: warning: the variable 'VarAssigned'
|
|
// CHECK-FIXES: const ExpensiveToCopyType& VarAssigned = ExpensiveTypeReference();
|
|
VarAssigned.constMethod();
|
|
|
|
ExpensiveToCopyType VarCopyConstructed(ExpensiveTypeReference());
|
|
// CHECK-MESSAGES: [[@LINE-1]]:23: warning: the variable 'VarCopyConstructed'
|
|
// CHECK-FIXES: const ExpensiveToCopyType& VarCopyConstructed(ExpensiveTypeReference());
|
|
VarCopyConstructed.constMethod();
|
|
}
|
|
|
|
void positiveNonConstVarInCodeBlock(const ExpensiveToCopyType &Obj) {
|
|
{
|
|
auto Assigned = Obj.reference();
|
|
// CHECK-MESSAGES: [[@LINE-1]]:10: warning: the variable 'Assigned'
|
|
// CHECK-FIXES: const auto& Assigned = Obj.reference();
|
|
Assigned.reference();
|
|
useAsConstReference(Assigned);
|
|
useByValue(Assigned);
|
|
}
|
|
}
|
|
|
|
void negativeNonConstVarWithNonConstUse(const ExpensiveToCopyType &Obj) {
|
|
{
|
|
auto NonConstInvoked = Obj.reference();
|
|
// CHECK-FIXES: auto NonConstInvoked = Obj.reference();
|
|
NonConstInvoked.nonConstMethod();
|
|
}
|
|
{
|
|
auto Reassigned = Obj.reference();
|
|
// CHECK-FIXES: auto Reassigned = Obj.reference();
|
|
Reassigned = ExpensiveToCopyType();
|
|
}
|
|
{
|
|
auto MutatedByReference = Obj.reference();
|
|
// CHECK-FIXES: auto MutatedByReference = Obj.reference();
|
|
mutate(MutatedByReference);
|
|
}
|
|
{
|
|
auto MutatedByPointer = Obj.reference();
|
|
// CHECK-FIXES: auto MutatedByPointer = Obj.reference();
|
|
mutate(&MutatedByPointer);
|
|
}
|
|
}
|
|
|
|
void PositiveMethodCallNonConstRefNotModified(ExpensiveToCopyType &Obj) {
|
|
const auto AutoAssigned = Obj.reference();
|
|
// CHECK-MESSAGES: [[@LINE-1]]:14: warning: the const qualified variable 'AutoAssigned'
|
|
// CHECK-FIXES: const auto& AutoAssigned = Obj.reference();
|
|
AutoAssigned.constMethod();
|
|
}
|
|
|
|
void NegativeMethodCallNonConstRefIsModified(ExpensiveToCopyType &Obj) {
|
|
const auto AutoAssigned = Obj.reference();
|
|
const auto AutoCopyConstructed(Obj.reference());
|
|
const ExpensiveToCopyType VarAssigned = Obj.reference();
|
|
const ExpensiveToCopyType VarCopyConstructed(Obj.reference());
|
|
mutate(&Obj);
|
|
}
|
|
|
|
void PositiveMethodCallNonConstNotModified(ExpensiveToCopyType Obj) {
|
|
const auto AutoAssigned = Obj.reference();
|
|
// CHECK-MESSAGES: [[@LINE-1]]:14: warning: the const qualified variable 'AutoAssigned'
|
|
// CHECK-FIXES: const auto& AutoAssigned = Obj.reference();
|
|
AutoAssigned.constMethod();
|
|
}
|
|
|
|
void NegativeMethodCallNonConstValueArgumentIsModified(ExpensiveToCopyType Obj) {
|
|
Obj.nonConstMethod();
|
|
const auto AutoAssigned = Obj.reference();
|
|
}
|
|
|
|
void PositiveMethodCallNonConstPointerNotModified(ExpensiveToCopyType *const Obj) {
|
|
const auto AutoAssigned = Obj->reference();
|
|
// CHECK-MESSAGES: [[@LINE-1]]:14: warning: the const qualified variable 'AutoAssigned'
|
|
// CHECK-FIXES: const auto& AutoAssigned = Obj->reference();
|
|
Obj->constMethod();
|
|
AutoAssigned.constMethod();
|
|
}
|
|
|
|
void NegativeMethodCallNonConstPointerIsModified(ExpensiveToCopyType *const Obj) {
|
|
const auto AutoAssigned = Obj->reference();
|
|
const auto AutoCopyConstructed(Obj->reference());
|
|
const ExpensiveToCopyType VarAssigned = Obj->reference();
|
|
const ExpensiveToCopyType VarCopyConstructed(Obj->reference());
|
|
mutate(Obj);
|
|
}
|
|
|
|
void PositiveLocalVarIsNotModified() {
|
|
ExpensiveToCopyType LocalVar;
|
|
const auto AutoAssigned = LocalVar.reference();
|
|
// CHECK-MESSAGES: [[@LINE-1]]:14: warning: the const qualified variable 'AutoAssigned'
|
|
// CHECK-FIXES: const auto& AutoAssigned = LocalVar.reference();
|
|
AutoAssigned.constMethod();
|
|
}
|
|
|
|
void NegativeLocalVarIsModified() {
|
|
ExpensiveToCopyType Obj;
|
|
const auto AutoAssigned = Obj.reference();
|
|
Obj = AutoAssigned;
|
|
}
|
|
|
|
struct NegativeConstructor {
|
|
NegativeConstructor(const ExpensiveToCopyType &Obj) : Obj(Obj) {}
|
|
ExpensiveToCopyType Obj;
|
|
};
|
|
|
|
#define UNNECESSARY_COPY_INIT_IN_MACRO_BODY(TYPE) \
|
|
void functionWith##TYPE(const TYPE &T) { \
|
|
auto AssignedInMacro = T.reference(); \
|
|
} \
|
|
// Ensure fix is not applied.
|
|
// CHECK-FIXES: auto AssignedInMacro = T.reference();
|
|
|
|
UNNECESSARY_COPY_INIT_IN_MACRO_BODY(ExpensiveToCopyType)
|
|
// CHECK-MESSAGES: [[@LINE-1]]:1: warning: the variable 'AssignedInMacro' is copy-constructed
|
|
|
|
#define UNNECESSARY_COPY_INIT_IN_MACRO_ARGUMENT(ARGUMENT) ARGUMENT
|
|
|
|
void PositiveMacroArgument(const ExpensiveToCopyType &Obj) {
|
|
UNNECESSARY_COPY_INIT_IN_MACRO_ARGUMENT(auto CopyInMacroArg = Obj.reference());
|
|
// CHECK-MESSAGES: [[@LINE-1]]:48: warning: the variable 'CopyInMacroArg' is copy-constructed
|
|
// Ensure fix is not applied.
|
|
// CHECK-FIXES: auto CopyInMacroArg = Obj.reference()
|
|
CopyInMacroArg.constMethod();
|
|
}
|
|
|
|
void PositiveLocalCopyConstMethodInvoked() {
|
|
ExpensiveToCopyType orig;
|
|
ExpensiveToCopyType copy_1 = orig;
|
|
// CHECK-MESSAGES: [[@LINE-1]]:23: warning: local copy 'copy_1' of the variable 'orig' is never modified; consider avoiding the copy [performance-unnecessary-copy-initialization]
|
|
// CHECK-FIXES: const ExpensiveToCopyType& copy_1 = orig;
|
|
copy_1.constMethod();
|
|
orig.constMethod();
|
|
}
|
|
|
|
void PositiveLocalCopyUsingExplicitCopyCtor() {
|
|
ExpensiveToCopyType orig;
|
|
ExpensiveToCopyType copy_2(orig);
|
|
// CHECK-MESSAGES: [[@LINE-1]]:23: warning: local copy 'copy_2'
|
|
// CHECK-FIXES: const ExpensiveToCopyType& copy_2(orig);
|
|
copy_2.constMethod();
|
|
orig.constMethod();
|
|
}
|
|
|
|
void PositiveLocalCopyCopyIsArgument(const ExpensiveToCopyType &orig) {
|
|
ExpensiveToCopyType copy_3 = orig;
|
|
// CHECK-MESSAGES: [[@LINE-1]]:23: warning: local copy 'copy_3'
|
|
// CHECK-FIXES: const ExpensiveToCopyType& copy_3 = orig;
|
|
copy_3.constMethod();
|
|
}
|
|
|
|
void PositiveLocalCopyUsedAsConstRef() {
|
|
ExpensiveToCopyType orig;
|
|
ExpensiveToCopyType copy_4 = orig;
|
|
// CHECK-MESSAGES: [[@LINE-1]]:23: warning: local copy 'copy_4'
|
|
// CHECK-FIXES: const ExpensiveToCopyType& copy_4 = orig;
|
|
useAsConstReference(orig);
|
|
copy_4.constMethod();
|
|
}
|
|
|
|
void PositiveLocalCopyTwice() {
|
|
ExpensiveToCopyType orig;
|
|
ExpensiveToCopyType copy_5 = orig;
|
|
// CHECK-MESSAGES: [[@LINE-1]]:23: warning: local copy 'copy_5'
|
|
// CHECK-FIXES: const ExpensiveToCopyType& copy_5 = orig;
|
|
ExpensiveToCopyType copy_6 = copy_5;
|
|
// CHECK-MESSAGES: [[@LINE-1]]:23: warning: local copy 'copy_6'
|
|
// CHECK-FIXES: const ExpensiveToCopyType& copy_6 = copy_5;
|
|
copy_5.constMethod();
|
|
copy_6.constMethod();
|
|
orig.constMethod();
|
|
}
|
|
|
|
|
|
void PositiveLocalCopyWeirdCopy() {
|
|
WeirdCopyCtorType orig;
|
|
WeirdCopyCtorType weird_1(orig);
|
|
// CHECK-MESSAGES: [[@LINE-1]]:21: warning: local copy 'weird_1'
|
|
// CHECK-FIXES: const WeirdCopyCtorType& weird_1(orig);
|
|
weird_1.constMethod();
|
|
|
|
WeirdCopyCtorType weird_2 = orig;
|
|
// CHECK-MESSAGES: [[@LINE-1]]:21: warning: local copy 'weird_2'
|
|
// CHECK-FIXES: const WeirdCopyCtorType& weird_2 = orig;
|
|
weird_2.constMethod();
|
|
}
|
|
|
|
void NegativeLocalCopySimpleTypes() {
|
|
int i1 = 0;
|
|
int i2 = i1;
|
|
}
|
|
|
|
void NegativeLocalCopyCopyIsModified() {
|
|
ExpensiveToCopyType orig;
|
|
ExpensiveToCopyType neg_copy_1 = orig;
|
|
neg_copy_1.nonConstMethod();
|
|
}
|
|
|
|
void NegativeLocalCopyOriginalIsModified() {
|
|
ExpensiveToCopyType orig;
|
|
ExpensiveToCopyType neg_copy_2 = orig;
|
|
orig.nonConstMethod();
|
|
}
|
|
|
|
void NegativeLocalCopyUsedAsRefArg() {
|
|
ExpensiveToCopyType orig;
|
|
ExpensiveToCopyType neg_copy_3 = orig;
|
|
mutate(neg_copy_3);
|
|
}
|
|
|
|
void NegativeLocalCopyUsedAsPointerArg() {
|
|
ExpensiveToCopyType orig;
|
|
ExpensiveToCopyType neg_copy_4 = orig;
|
|
mutate(&neg_copy_4);
|
|
}
|
|
|
|
void NegativeLocalCopyCopyFromGlobal() {
|
|
ExpensiveToCopyType neg_copy_5 = global_expensive_to_copy_type;
|
|
}
|
|
|
|
void NegativeLocalCopyCopyToStatic() {
|
|
ExpensiveToCopyType orig;
|
|
static ExpensiveToCopyType neg_copy_6 = orig;
|
|
}
|
|
|
|
void NegativeLocalCopyNonConstInForLoop() {
|
|
ExpensiveToCopyType orig;
|
|
for (ExpensiveToCopyType neg_copy_7 = orig; orig.constMethod();
|
|
orig.nonConstMethod()) {
|
|
orig.constMethod();
|
|
}
|
|
}
|
|
|
|
void NegativeLocalCopyWeirdNonCopy() {
|
|
WeirdCopyCtorType orig;
|
|
WeirdCopyCtorType neg_weird_1(orig, false);
|
|
WeirdCopyCtorType neg_weird_2(orig, true);
|
|
}
|
|
void WarningOnlyMultiDeclStmt() {
|
|
ExpensiveToCopyType orig;
|
|
ExpensiveToCopyType copy = orig, copy2;
|
|
// CHECK-MESSAGES: [[@LINE-1]]:23: warning: local copy 'copy' of the variable 'orig' is never modified; consider avoiding the copy [performance-unnecessary-copy-initialization]
|
|
// CHECK-FIXES: ExpensiveToCopyType copy = orig, copy2;
|
|
copy.constMethod();
|
|
}
|
|
|
|
class Element {};
|
|
class Container {
|
|
public:
|
|
class Iterator {
|
|
public:
|
|
void operator++();
|
|
Element operator*();
|
|
bool operator!=(const Iterator &);
|
|
WeirdCopyCtorType c;
|
|
};
|
|
const Iterator &begin() const;
|
|
const Iterator &end() const;
|
|
};
|
|
|
|
void implicitVarFalsePositive() {
|
|
for (const Element &E : Container()) {
|
|
}
|
|
}
|
|
|
|
// This should not trigger the check as the argument could introduce an alias.
|
|
void negativeInitializedFromFreeFunctionWithArg() {
|
|
ExpensiveToCopyType Orig;
|
|
const ExpensiveToCopyType Copy = freeFunctionWithArg(Orig);
|
|
}
|
|
|
|
void negativeInitializedFromFreeFunctionWithDefaultArg() {
|
|
const ExpensiveToCopyType Copy = freeFunctionWithDefaultArg();
|
|
}
|
|
|
|
void negativeInitialzedFromFreeFunctionWithNonDefaultArg() {
|
|
ExpensiveToCopyType Orig;
|
|
const ExpensiveToCopyType Copy = freeFunctionWithDefaultArg(&Orig);
|
|
}
|
|
|
|
namespace std {
|
|
inline namespace __1 {
|
|
|
|
template <class>
|
|
class function;
|
|
template <class R, class... ArgTypes>
|
|
class function<R(ArgTypes...)> {
|
|
public:
|
|
function();
|
|
function(const function &Other);
|
|
R operator()(ArgTypes... Args) const;
|
|
};
|
|
|
|
} // namespace __1
|
|
} // namespace std
|
|
|
|
void negativeStdFunction() {
|
|
std::function<int()> Orig;
|
|
std::function<int()> Copy = Orig;
|
|
int i = Orig();
|
|
}
|
|
|
|
using Functor = std::function<int()>;
|
|
|
|
void negativeAliasedStdFunction() {
|
|
Functor Orig;
|
|
Functor Copy = Orig;
|
|
int i = Orig();
|
|
}
|
|
|
|
typedef std::function<int()> TypedefFunc;
|
|
|
|
void negativeTypedefedStdFunction() {
|
|
TypedefFunc Orig;
|
|
TypedefFunc Copy = Orig;
|
|
int i = Orig();
|
|
}
|
|
|
|
namespace fake {
|
|
namespace std {
|
|
template <class R, class... Args>
|
|
struct function {
|
|
// Custom copy constructor makes it expensive to copy;
|
|
function(const function &);
|
|
void constMethod() const;
|
|
};
|
|
} // namespace std
|
|
|
|
void positiveFakeStdFunction(std::function<void(int)> F) {
|
|
auto Copy = F;
|
|
// CHECK-MESSAGES: [[@LINE-1]]:8: warning: local copy 'Copy' of the variable 'F' is never modified;
|
|
// CHECK-FIXES: const auto& Copy = F;
|
|
Copy.constMethod();
|
|
}
|
|
|
|
} // namespace fake
|
|
|
|
void positiveInvokedOnStdFunction(
|
|
std::function<void(const ExpensiveToCopyType &)> Update,
|
|
const ExpensiveToCopyType Orig) {
|
|
auto Copy = Orig.reference();
|
|
// CHECK-MESSAGES: [[@LINE-1]]:8: warning: the variable 'Copy' is copy-constructed from a const reference
|
|
// CHECK-FIXES: const auto& Copy = Orig.reference();
|
|
Update(Copy);
|
|
}
|
|
|
|
void negativeInvokedOnStdFunction(
|
|
std::function<void(ExpensiveToCopyType &)> Update,
|
|
const ExpensiveToCopyType Orig) {
|
|
auto Copy = Orig.reference();
|
|
Update(Copy);
|
|
}
|
|
|
|
void negativeCopiedFromReferenceToModifiedVar() {
|
|
ExpensiveToCopyType Orig;
|
|
const auto &Ref = Orig;
|
|
const auto NecessaryCopy = Ref;
|
|
Orig.nonConstMethod();
|
|
}
|
|
|
|
void positiveCopiedFromReferenceToConstVar() {
|
|
ExpensiveToCopyType Orig;
|
|
const auto &Ref = Orig;
|
|
const auto UnnecessaryCopy = Ref;
|
|
// CHECK-MESSAGES: [[@LINE-1]]:14: warning: local copy 'UnnecessaryCopy' of
|
|
// CHECK-FIXES: const auto& UnnecessaryCopy = Ref;
|
|
Orig.constMethod();
|
|
UnnecessaryCopy.constMethod();
|
|
}
|
|
|
|
void negativeCopiedFromGetterOfReferenceToModifiedVar() {
|
|
ExpensiveToCopyType Orig;
|
|
const auto &Ref = Orig.reference();
|
|
const auto NecessaryCopy = Ref.reference();
|
|
Orig.nonConstMethod();
|
|
}
|
|
|
|
void negativeAliasNonCanonicalPointerType() {
|
|
ExpensiveToCopyType Orig;
|
|
// The use of auto here hides that the type is a pointer type. The check needs
|
|
// to look at the canonical type to detect the aliasing through this pointer.
|
|
const auto Pointer = Orig.pointer();
|
|
const auto NecessaryCopy = Pointer->reference();
|
|
Orig.nonConstMethod();
|
|
}
|
|
|
|
void negativeAliasTypedefedType() {
|
|
typedef const ExpensiveToCopyType &ReferenceType;
|
|
ExpensiveToCopyType Orig;
|
|
// The typedef hides the fact that this is a reference type. The check needs
|
|
// to look at the canonical type to detect the aliasing.
|
|
ReferenceType Ref = Orig.reference();
|
|
const auto NecessaryCopy = Ref.reference();
|
|
Orig.nonConstMethod();
|
|
}
|
|
|
|
void positiveCopiedFromGetterOfReferenceToConstVar() {
|
|
ExpensiveToCopyType Orig;
|
|
const auto &Ref = Orig.reference();
|
|
auto UnnecessaryCopy = Ref.reference();
|
|
// CHECK-MESSAGES: [[@LINE-1]]:8: warning: the variable 'UnnecessaryCopy' is
|
|
// CHECK-FIXES: const auto& UnnecessaryCopy = Ref.reference();
|
|
Orig.constMethod();
|
|
UnnecessaryCopy.constMethod();
|
|
}
|
|
|
|
void positiveUnusedReferenceIsRemoved() {
|
|
// clang-format off
|
|
const auto AutoAssigned = ExpensiveTypeReference(); int i = 0; // Foo bar.
|
|
// CHECK-MESSAGES: [[@LINE-1]]:14: warning: the const qualified variable 'AutoAssigned' is copy-constructed from a const reference but is never used; consider removing the statement [performance-unnecessary-copy-initialization]
|
|
// CHECK-FIXES-NOT: const auto AutoAssigned = ExpensiveTypeReference();
|
|
// CHECK-FIXES: int i = 0; // Foo bar.
|
|
auto TrailingCommentRemoved = ExpensiveTypeReference(); // Trailing comment.
|
|
// CHECK-MESSAGES: [[@LINE-1]]:8: warning: the variable 'TrailingCommentRemoved' is copy-constructed from a const reference but is never used;
|
|
// CHECK-FIXES-NOT: auto TrailingCommentRemoved = ExpensiveTypeReference();
|
|
// CHECK-FIXES-NOT: // Trailing comment.
|
|
// clang-format on
|
|
|
|
auto UnusedAndUnnecessary = ExpensiveTypeReference();
|
|
// Comments on a new line should not be deleted.
|
|
// CHECK-MESSAGES: [[@LINE-2]]:8: warning: the variable 'UnusedAndUnnecessary' is copy-constructed
|
|
// CHECK-FIXES-NOT: auto UnusedAndUnnecessary = ExpensiveTypeReference();
|
|
// CHECK-FIXES: // Comments on a new line should not be deleted.
|
|
}
|
|
|
|
void negativeloopedOverObjectIsModified() {
|
|
ExpensiveToCopyType Orig;
|
|
for (const auto &Element : Orig) {
|
|
const auto Copy = Element;
|
|
Orig.nonConstMethod();
|
|
Copy.constMethod();
|
|
}
|
|
|
|
auto Lambda = []() {
|
|
ExpensiveToCopyType Orig;
|
|
for (const auto &Element : Orig) {
|
|
const auto Copy = Element;
|
|
Orig.nonConstMethod();
|
|
Copy.constMethod();
|
|
}
|
|
};
|
|
}
|
|
|
|
void negativeReferenceIsInitializedOutsideOfBlock() {
|
|
ExpensiveToCopyType Orig;
|
|
const auto &E2 = Orig;
|
|
if (1 != 2 * 3) {
|
|
const auto C2 = E2;
|
|
Orig.nonConstMethod();
|
|
C2.constMethod();
|
|
}
|
|
|
|
auto Lambda = []() {
|
|
ExpensiveToCopyType Orig;
|
|
const auto &E2 = Orig;
|
|
if (1 != 2 * 3) {
|
|
const auto C2 = E2;
|
|
Orig.nonConstMethod();
|
|
C2.constMethod();
|
|
}
|
|
};
|
|
}
|
|
|
|
void negativeStructuredBinding() {
|
|
// Structured bindings are not yet supported but can trigger false positives
|
|
// since the DecompositionDecl itself is unused and the check doesn't traverse
|
|
// VarDecls of the BindingDecls.
|
|
struct Pair {
|
|
ExpensiveToCopyType first;
|
|
ExpensiveToCopyType second;
|
|
};
|
|
|
|
Pair P;
|
|
const auto [C, D] = P;
|
|
C.constMethod();
|
|
D.constMethod();
|
|
}
|
|
|
|
template <typename A>
|
|
const A &templatedReference();
|
|
|
|
template <typename A, typename B>
|
|
void negativeTemplateTypes() {
|
|
A Orig;
|
|
// Different replaced template type params do not trigger the check. In some
|
|
// template instantiation this might not be a copy but an implicit
|
|
// conversion, so converting this to a reference might not work.
|
|
B AmbiguousCopy = Orig;
|
|
// CHECK-NOT-FIXES: B AmbiguousCopy = Orig;
|
|
|
|
B NecessaryCopy = templatedReference<A>();
|
|
// CHECK-NOT-FIXES: B NecessaryCopy = templatedReference<A>();
|
|
|
|
B NecessaryCopy2 = Orig.template templatedAccessor<A>();
|
|
|
|
// Non-dependent types in template still trigger the check.
|
|
const auto UnnecessaryCopy = ExpensiveTypeReference();
|
|
// CHECK-MESSAGES: [[@LINE-1]]:14: warning: the const qualified variable 'UnnecessaryCopy' is copy-constructed
|
|
// CHECK-FIXES: const auto& UnnecessaryCopy = ExpensiveTypeReference();
|
|
UnnecessaryCopy.constMethod();
|
|
}
|
|
|
|
void instantiateNegativeTemplateTypes() {
|
|
negativeTemplateTypes<ExpensiveToCopyType, ExpensiveToCopyType>();
|
|
// This template instantiation would not compile if the `AmbiguousCopy` above was made a reference.
|
|
negativeTemplateTypes<ExpensiveToCopyType, int>();
|
|
}
|
|
|
|
template <typename A>
|
|
void positiveSingleTemplateType() {
|
|
A Orig;
|
|
A SingleTmplParmTypeCopy = Orig;
|
|
// CHECK-MESSAGES: [[@LINE-1]]:5: warning: local copy 'SingleTmplParmTypeCopy' of the variable 'Orig' is never modified
|
|
// CHECK-FIXES: const A& SingleTmplParmTypeCopy = Orig;
|
|
SingleTmplParmTypeCopy.constMethod();
|
|
|
|
A UnnecessaryCopy2 = templatedReference<A>();
|
|
// CHECK-MESSAGES: [[@LINE-1]]:5: warning: the variable 'UnnecessaryCopy2' is copy-constructed from a const reference
|
|
// CHECK-FIXES: const A& UnnecessaryCopy2 = templatedReference<A>();
|
|
UnnecessaryCopy2.constMethod();
|
|
|
|
A UnnecessaryCopy3 = Orig.template templatedAccessor<A>();
|
|
// CHECK-MESSAGES: [[@LINE-1]]:5: warning: the variable 'UnnecessaryCopy3' is copy-constructed from a const reference
|
|
// CHECK-FIXES: const A& UnnecessaryCopy3 = Orig.template templatedAccessor<A>();
|
|
UnnecessaryCopy3.constMethod();
|
|
}
|
|
|
|
void instantiatePositiveSingleTemplateType() { positiveSingleTemplateType<ExpensiveToCopyType>(); }
|