You've already forked linux-packaging-mono
							
							
		
			
	
	
		
			362 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
		
		
			
		
	
	
			362 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
|   | // RUN: %check_clang_tidy %s performance-unnecessary-value-param %t
 | ||
|  | 
 | ||
|  | // CHECK-FIXES: #include <utility>
 | ||
|  | 
 | ||
|  | struct ExpensiveToCopyType { | ||
|  |   const ExpensiveToCopyType & constReference() const { | ||
|  |     return *this; | ||
|  |   } | ||
|  |   void nonConstMethod(); | ||
|  |   virtual ~ExpensiveToCopyType(); | ||
|  | }; | ||
|  | 
 | ||
|  | void mutate(ExpensiveToCopyType &); | ||
|  | void mutate(ExpensiveToCopyType *); | ||
|  | void useAsConstReference(const ExpensiveToCopyType &); | ||
|  | void useByValue(ExpensiveToCopyType); | ||
|  | 
 | ||
|  | // This class simulates std::pair<>. It is trivially copy constructible
 | ||
|  | // and trivially destructible, but not trivially copy assignable.
 | ||
|  | class SomewhatTrivial { | ||
|  |  public: | ||
|  |   SomewhatTrivial(); | ||
|  |   SomewhatTrivial(const SomewhatTrivial&) = default; | ||
|  |   ~SomewhatTrivial() = default; | ||
|  |   SomewhatTrivial& operator=(const SomewhatTrivial&); | ||
|  | }; | ||
|  | 
 | ||
|  | struct MoveOnlyType { | ||
|  |   MoveOnlyType(const MoveOnlyType &) = delete; | ||
|  |   MoveOnlyType(MoveOnlyType &&) = default; | ||
|  |   ~MoveOnlyType(); | ||
|  |   void constMethod() const; | ||
|  | }; | ||
|  | 
 | ||
|  | struct ExpensiveMovableType { | ||
|  |   ExpensiveMovableType(); | ||
|  |   ExpensiveMovableType(ExpensiveMovableType &&); | ||
|  |   ExpensiveMovableType(const ExpensiveMovableType &) = default; | ||
|  |   ExpensiveMovableType &operator=(const ExpensiveMovableType &) = default; | ||
|  |   ExpensiveMovableType &operator=(ExpensiveMovableType &&); | ||
|  |   ~ExpensiveMovableType(); | ||
|  | }; | ||
|  | 
 | ||
|  | void positiveExpensiveConstValue(const ExpensiveToCopyType Obj); | ||
|  | // CHECK-FIXES: void positiveExpensiveConstValue(const ExpensiveToCopyType& Obj);
 | ||
|  | void positiveExpensiveConstValue(const ExpensiveToCopyType Obj) { | ||
|  |   // CHECK-MESSAGES: [[@LINE-1]]:60: warning: the const qualified parameter 'Obj' is copied for each invocation; consider making it a reference [performance-unnecessary-value-param]
 | ||
|  |   // CHECK-FIXES: void positiveExpensiveConstValue(const ExpensiveToCopyType& Obj) {
 | ||
|  | } | ||
|  | 
 | ||
|  | void positiveExpensiveValue(ExpensiveToCopyType Obj); | ||
|  | // CHECK-FIXES: void positiveExpensiveValue(const ExpensiveToCopyType& Obj);
 | ||
|  | void positiveExpensiveValue(ExpensiveToCopyType Obj) { | ||
|  |   // CHECK-MESSAGES: [[@LINE-1]]:49: warning: the parameter 'Obj' is copied for each invocation but only used as a const reference; consider making it a const reference [performance-unnecessary-value-param]
 | ||
|  |   // CHECK-FIXES: void positiveExpensiveValue(const ExpensiveToCopyType& Obj) {
 | ||
|  |   Obj.constReference(); | ||
|  |   useAsConstReference(Obj); | ||
|  |   auto Copy = Obj; | ||
|  |   useByValue(Obj); | ||
|  | } | ||
|  | 
 | ||
|  | void positiveWithComment(const ExpensiveToCopyType /* important */ S); | ||
|  | // CHECK-FIXES: void positiveWithComment(const ExpensiveToCopyType& /* important */ S);
 | ||
|  | void positiveWithComment(const ExpensiveToCopyType /* important */ S) { | ||
|  |   // CHECK-MESSAGES: [[@LINE-1]]:68: warning: the const qualified
 | ||
|  |   // CHECK-FIXES: void positiveWithComment(const ExpensiveToCopyType& /* important */ S) {
 | ||
|  | } | ||
|  | 
 | ||
|  | void positiveUnnamedParam(const ExpensiveToCopyType) { | ||
|  |   // CHECK-MESSAGES: [[@LINE-1]]:52: warning: the const qualified parameter #1
 | ||
|  |   // CHECK-FIXES: void positiveUnnamedParam(const ExpensiveToCopyType&) {
 | ||
|  | } | ||
|  | 
 | ||
|  | void positiveAndNegative(const ExpensiveToCopyType ConstCopy, const ExpensiveToCopyType& ConstRef, ExpensiveToCopyType Copy); | ||
|  | // CHECK-FIXES: void positiveAndNegative(const ExpensiveToCopyType& ConstCopy, const ExpensiveToCopyType& ConstRef, const ExpensiveToCopyType& Copy);
 | ||
|  | void positiveAndNegative(const ExpensiveToCopyType ConstCopy, const ExpensiveToCopyType& ConstRef, ExpensiveToCopyType Copy) { | ||
|  |   // CHECK-MESSAGES: [[@LINE-1]]:52: warning: the const qualified parameter 'ConstCopy'
 | ||
|  |   // CHECK-MESSAGES: [[@LINE-2]]:120: warning: the parameter 'Copy'
 | ||
|  |   // CHECK-FIXES: void positiveAndNegative(const ExpensiveToCopyType& ConstCopy, const ExpensiveToCopyType& ConstRef, const ExpensiveToCopyType& Copy) {
 | ||
|  | } | ||
|  | 
 | ||
|  | struct PositiveConstValueConstructor { | ||
|  |   PositiveConstValueConstructor(const ExpensiveToCopyType ConstCopy) {} | ||
|  |   // CHECK-MESSAGES: [[@LINE-1]]:59: warning: the const qualified parameter 'ConstCopy'
 | ||
|  |   // CHECK-FIXES: PositiveConstValueConstructor(const ExpensiveToCopyType& ConstCopy) {}
 | ||
|  | }; | ||
|  | 
 | ||
|  | template <typename T> void templateWithNonTemplatizedParameter(const ExpensiveToCopyType S, T V) { | ||
|  |   // CHECK-MESSAGES: [[@LINE-1]]:90: warning: the const qualified parameter 'S'
 | ||
|  |   // CHECK-FIXES: template <typename T> void templateWithNonTemplatizedParameter(const ExpensiveToCopyType& S, T V) {
 | ||
|  | } | ||
|  | 
 | ||
|  | void instantiated() { | ||
|  |   templateWithNonTemplatizedParameter(ExpensiveToCopyType(), ExpensiveToCopyType()); | ||
|  |   templateWithNonTemplatizedParameter(ExpensiveToCopyType(), 5); | ||
|  | } | ||
|  | 
 | ||
|  | template <typename T> void negativeTemplateType(const T V) { | ||
|  | } | ||
|  | 
 | ||
|  | void negativeArray(const ExpensiveToCopyType[]) { | ||
|  | } | ||
|  | 
 | ||
|  | void negativePointer(ExpensiveToCopyType* Obj) { | ||
|  | } | ||
|  | 
 | ||
|  | void negativeConstPointer(const ExpensiveToCopyType* Obj) { | ||
|  | } | ||
|  | 
 | ||
|  | void negativeConstReference(const ExpensiveToCopyType& Obj) { | ||
|  | } | ||
|  | 
 | ||
|  | void negativeReference(ExpensiveToCopyType& Obj) { | ||
|  | } | ||
|  | 
 | ||
|  | void negativeUniversalReference(ExpensiveToCopyType&& Obj) { | ||
|  | } | ||
|  | 
 | ||
|  | void negativeSomewhatTrivialConstValue(const SomewhatTrivial Somewhat) { | ||
|  | } | ||
|  | 
 | ||
|  | void negativeSomewhatTrivialValue(SomewhatTrivial Somewhat) { | ||
|  | } | ||
|  | 
 | ||
|  | void negativeConstBuiltIn(const int I) { | ||
|  | } | ||
|  | 
 | ||
|  | void negativeValueBuiltIn(int I) { | ||
|  | } | ||
|  | 
 | ||
|  | void negativeValueIsMutatedByReference(ExpensiveToCopyType Obj) { | ||
|  |   mutate(Obj); | ||
|  | } | ||
|  | 
 | ||
|  | void negativeValueIsMutatatedByPointer(ExpensiveToCopyType Obj) { | ||
|  |   mutate(&Obj); | ||
|  | } | ||
|  | 
 | ||
|  | void negativeValueIsReassigned(ExpensiveToCopyType Obj) { | ||
|  |   Obj = ExpensiveToCopyType(); | ||
|  | } | ||
|  | 
 | ||
|  | void negativeValueNonConstMethodIsCalled(ExpensiveToCopyType Obj) { | ||
|  |   Obj.nonConstMethod(); | ||
|  | } | ||
|  | 
 | ||
|  | struct PositiveValueUnusedConstructor { | ||
|  |   PositiveValueUnusedConstructor(ExpensiveToCopyType Copy) {} | ||
|  |   // CHECK-MESSAGES: [[@LINE-1]]:54: warning: the parameter 'Copy'
 | ||
|  |   // CHECK-FIXES: PositiveValueUnusedConstructor(const ExpensiveToCopyType& Copy) {}
 | ||
|  | }; | ||
|  | 
 | ||
|  | struct PositiveValueCopiedConstructor { | ||
|  |   PositiveValueCopiedConstructor(ExpensiveToCopyType Copy) : Field(Copy) {} | ||
|  |   // CHECK-MESSAGES: [[@LINE-1]]:54: warning: the parameter 'Copy'
 | ||
|  |   // CHECK-FIXES: PositiveValueCopiedConstructor(const ExpensiveToCopyType& Copy) : Field(Copy) {}
 | ||
|  |   ExpensiveToCopyType Field; | ||
|  | }; | ||
|  | 
 | ||
|  | struct PositiveValueMovableConstructor { | ||
|  |   PositiveValueMovableConstructor(ExpensiveMovableType Copy) : Field(Copy) {} | ||
|  |   // CHECK-MESSAGES: [[@LINE-1]]:70: warning: parameter 'Copy'
 | ||
|  |   // CHECK-FIXES: PositiveValueMovableConstructor(ExpensiveMovableType Copy) : Field(std::move(Copy)) {}
 | ||
|  |   ExpensiveMovableType Field; | ||
|  | }; | ||
|  | 
 | ||
|  | struct NegativeValueMovedConstructor { | ||
|  |   NegativeValueMovedConstructor(ExpensiveMovableType Copy) : Field(static_cast<ExpensiveMovableType &&>(Copy)) {} | ||
|  |   ExpensiveMovableType Field; | ||
|  | }; | ||
|  | 
 | ||
|  | template <typename T> | ||
|  | struct Container { | ||
|  |   typedef const T & const_reference; | ||
|  | }; | ||
|  | 
 | ||
|  | void NegativeTypedefParam(const Container<ExpensiveToCopyType>::const_reference Param) { | ||
|  | } | ||
|  | 
 | ||
|  | #define UNNECESSARY_VALUE_PARAM_IN_MACRO_BODY()         \
 | ||
|  |   void inMacro(const ExpensiveToCopyType T) {           \ | ||
|  |   }                                                     \ | ||
|  | // Ensure fix is not applied.
 | ||
|  | // CHECK-FIXES: void inMacro(const ExpensiveToCopyType T) {
 | ||
|  | 
 | ||
|  | UNNECESSARY_VALUE_PARAM_IN_MACRO_BODY() | ||
|  | // CHECK-MESSAGES: [[@LINE-1]]:1: warning: the const qualified parameter 'T'
 | ||
|  | 
 | ||
|  | #define UNNECESSARY_VALUE_PARAM_IN_MACRO_ARGUMENT(ARGUMENT)     \
 | ||
|  |   ARGUMENT | ||
|  | 
 | ||
|  | UNNECESSARY_VALUE_PARAM_IN_MACRO_ARGUMENT(void inMacroArgument(const ExpensiveToCopyType InMacroArg) {}) | ||
|  | // CHECK-MESSAGES: [[@LINE-1]]:90: warning: the const qualified parameter 'InMacroArg'
 | ||
|  | // CHECK-FIXES: void inMacroArgument(const ExpensiveToCopyType InMacroArg) {}
 | ||
|  | 
 | ||
|  | struct VirtualMethod { | ||
|  |   virtual ~VirtualMethod() {} | ||
|  |   virtual void handle(ExpensiveToCopyType T) const = 0; | ||
|  | }; | ||
|  | 
 | ||
|  | struct NegativeOverriddenMethod : public VirtualMethod { | ||
|  |   void handle(ExpensiveToCopyType Overridden) const { | ||
|  |     // CHECK-FIXES: handle(ExpensiveToCopyType Overridden) const {
 | ||
|  |   } | ||
|  | }; | ||
|  | 
 | ||
|  | struct VirtualMethodWarningOnly { | ||
|  |   virtual void methodWithExpensiveValueParam(ExpensiveToCopyType T) {} | ||
|  |   // CHECK-MESSAGES: [[@LINE-1]]:66: warning: the parameter 'T' is copied
 | ||
|  |   // CHECK-FIXES: virtual void methodWithExpensiveValueParam(ExpensiveToCopyType T) {}
 | ||
|  |   virtual ~VirtualMethodWarningOnly() {} | ||
|  | }; | ||
|  | 
 | ||
|  | struct PositiveNonVirualMethod { | ||
|  |   void method(const ExpensiveToCopyType T) {} | ||
|  |   // CHECK-MESSAGES: [[@LINE-1]]:41: warning: the const qualified parameter 'T' is copied
 | ||
|  |   // CHECK-FIXES: void method(const ExpensiveToCopyType& T) {}
 | ||
|  | }; | ||
|  | 
 | ||
|  | struct NegativeDeletedMethod { | ||
|  |   ~NegativeDeletedMethod() {} | ||
|  |   NegativeDeletedMethod& operator=(NegativeDeletedMethod N) = delete; | ||
|  |   // CHECK-FIXES: NegativeDeletedMethod& operator=(NegativeDeletedMethod N) = delete;
 | ||
|  | }; | ||
|  | 
 | ||
|  | void NegativeMoveOnlyTypePassedByValue(MoveOnlyType M) { | ||
|  |   M.constMethod(); | ||
|  | } | ||
|  | 
 | ||
|  | void PositiveMoveOnCopyConstruction(ExpensiveMovableType E) { | ||
|  |   auto F = E; | ||
|  |   // CHECK-MESSAGES: [[@LINE-1]]:12: warning: parameter 'E' is passed by value and only copied once; consider moving it to avoid unnecessary copies [performance-unnecessary-value-param]
 | ||
|  |   // CHECK-FIXES: auto F = std::move(E);
 | ||
|  | } | ||
|  | 
 | ||
|  | void PositiveConstRefNotMoveSinceReferencedMultipleTimes(ExpensiveMovableType E) { | ||
|  |   // CHECK-MESSAGES: [[@LINE-1]]:79: warning: the parameter 'E' is copied
 | ||
|  |   // CHECK-FIXES: void PositiveConstRefNotMoveSinceReferencedMultipleTimes(const ExpensiveMovableType& E) {
 | ||
|  |   auto F = E; | ||
|  |   auto G = E; | ||
|  | } | ||
|  | 
 | ||
|  | void PositiveMoveOnCopyAssignment(ExpensiveMovableType E) { | ||
|  |   ExpensiveMovableType F; | ||
|  |   F = E; | ||
|  |   // CHECK-MESSAGES: [[@LINE-1]]:7: warning: parameter 'E' is passed by value
 | ||
|  |   // CHECK-FIXES: F = std::move(E);
 | ||
|  | } | ||
|  | 
 | ||
|  | struct NotCopyAssigned { | ||
|  |   NotCopyAssigned &operator=(const ExpensiveMovableType &); | ||
|  | }; | ||
|  | 
 | ||
|  | void PositiveNoMoveForNonCopyAssigmentOperator(ExpensiveMovableType E) { | ||
|  |   // CHECK-MESSAGES: [[@LINE-1]]:69: warning: the parameter 'E' is copied
 | ||
|  |   // CHECK-FIXES: void PositiveNoMoveForNonCopyAssigmentOperator(const ExpensiveMovableType& E) {
 | ||
|  |   NotCopyAssigned N; | ||
|  |   N = E; | ||
|  | } | ||
|  | 
 | ||
|  | // The argument could be moved but is not since copy statement is inside a loop.
 | ||
|  | void PositiveNoMoveInsideLoop(ExpensiveMovableType E) { | ||
|  |   // CHECK-MESSAGES: [[@LINE-1]]:52: warning: the parameter 'E' is copied
 | ||
|  |   // CHECK-FIXES: void PositiveNoMoveInsideLoop(const ExpensiveMovableType& E) {
 | ||
|  |   for (;;) { | ||
|  |     auto F = E; | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | void PositiveConstRefNotMoveConstructible(ExpensiveToCopyType T) { | ||
|  |   // CHECK-MESSAGES: [[@LINE-1]]:63: warning: the parameter 'T' is copied
 | ||
|  |   // CHECK-FIXES: void PositiveConstRefNotMoveConstructible(const ExpensiveToCopyType& T) {
 | ||
|  |   auto U = T; | ||
|  | } | ||
|  | 
 | ||
|  | void PositiveConstRefNotMoveAssignable(ExpensiveToCopyType A) { | ||
|  |   // CHECK-MESSAGES: [[@LINE-1]]:60: warning: the parameter 'A' is copied
 | ||
|  |   // CHECK-FIXES: void PositiveConstRefNotMoveAssignable(const ExpensiveToCopyType& A) {
 | ||
|  |   ExpensiveToCopyType B; | ||
|  |   B = A; | ||
|  | } | ||
|  | 
 | ||
|  | // Case where parameter in declaration is already const-qualified but not in
 | ||
|  | // implementation. Make sure a second 'const' is not added to the declaration.
 | ||
|  | void PositiveConstDeclaration(const ExpensiveToCopyType A); | ||
|  | // CHECK-FIXES: void PositiveConstDeclaration(const ExpensiveToCopyType& A);
 | ||
|  | void PositiveConstDeclaration(ExpensiveToCopyType A) { | ||
|  |   // CHECK-MESSAGES: [[@LINE-1]]:51: warning: the parameter 'A' is copied
 | ||
|  |   // CHECK-FIXES: void PositiveConstDeclaration(const ExpensiveToCopyType& A) {
 | ||
|  | } | ||
|  | 
 | ||
|  | void PositiveNonConstDeclaration(ExpensiveToCopyType A); | ||
|  | // CHECK-FIXES: void PositiveNonConstDeclaration(const ExpensiveToCopyType& A);
 | ||
|  | void PositiveNonConstDeclaration(const ExpensiveToCopyType A) { | ||
|  |   // CHECK-MESSAGES: [[@LINE-1]]:60: warning: the const qualified parameter 'A'
 | ||
|  |   // CHECK-FIXES: void PositiveNonConstDeclaration(const ExpensiveToCopyType& A) {
 | ||
|  | } | ||
|  | 
 | ||
|  | void PositiveOnlyMessageAsReferencedInCompilationUnit(ExpensiveToCopyType A) { | ||
|  |   // CHECK-MESSAGES: [[@LINE-1]]:75: warning: the parameter 'A' is copied
 | ||
|  |   // CHECK-FIXES: void PositiveOnlyMessageAsReferencedInCompilationUnit(ExpensiveToCopyType A) {
 | ||
|  | } | ||
|  | 
 | ||
|  | void ReferenceFunctionOutsideOfCallExpr() { | ||
|  |   void (*ptr)(ExpensiveToCopyType) = &PositiveOnlyMessageAsReferencedInCompilationUnit; | ||
|  | } | ||
|  | 
 | ||
|  | void PositiveMessageAndFixAsFunctionIsCalled(ExpensiveToCopyType A) { | ||
|  |   // CHECK-MESSAGES: [[@LINE-1]]:66: warning: the parameter 'A' is copied
 | ||
|  |   // CHECK-FIXES: void PositiveMessageAndFixAsFunctionIsCalled(const ExpensiveToCopyType& A) {
 | ||
|  | } | ||
|  | 
 | ||
|  | void ReferenceFunctionByCallingIt() { | ||
|  |   PositiveMessageAndFixAsFunctionIsCalled(ExpensiveToCopyType()); | ||
|  | } | ||
|  | 
 | ||
|  | // Virtual method overrides of dependent types cannot be recognized unless they
 | ||
|  | // are marked as override or final. Test that check is not triggered on methods
 | ||
|  | // marked with override or final.
 | ||
|  | template <typename T> | ||
|  | struct NegativeDependentTypeInterface { | ||
|  |   virtual void Method(ExpensiveToCopyType E) = 0; | ||
|  | }; | ||
|  | 
 | ||
|  | template <typename T> | ||
|  | struct NegativeOverrideImpl : public NegativeDependentTypeInterface<T> { | ||
|  |   void Method(ExpensiveToCopyType E) override {} | ||
|  | }; | ||
|  | 
 | ||
|  | template <typename T> | ||
|  | struct NegativeFinalImpl : public NegativeDependentTypeInterface<T> { | ||
|  |   void Method(ExpensiveToCopyType E) final {} | ||
|  | }; | ||
|  | 
 | ||
|  | struct PositiveConstructor { | ||
|  |   PositiveConstructor(ExpensiveToCopyType E) : E(E) {} | ||
|  |   // CHECK-MESSAGES: [[@LINE-1]]:43: warning: the parameter 'E' is copied
 | ||
|  |   // CHECK-FIXES: PositiveConstructor(const ExpensiveToCopyType& E) : E(E) {}
 | ||
|  | 
 | ||
|  |   ExpensiveToCopyType E; | ||
|  | }; | ||
|  | 
 | ||
|  | struct NegativeUsingConstructor : public PositiveConstructor { | ||
|  |   using PositiveConstructor::PositiveConstructor; | ||
|  | }; | ||
|  | 
 | ||
|  | void fun() { | ||
|  |   ExpensiveToCopyType E; | ||
|  |   NegativeUsingConstructor S(E); | ||
|  | } | ||
|  | 
 | ||
|  | template<typename T> | ||
|  | void templateFunction(T) { | ||
|  | } | ||
|  | 
 | ||
|  | template<> | ||
|  | void templateFunction<ExpensiveToCopyType>(ExpensiveToCopyType E) { | ||
|  |   // CHECK-MESSAGES: [[@LINE-1]]:64: warning: the parameter 'E' is copied
 | ||
|  |   // CHECK-FIXES: void templateFunction<ExpensiveToCopyType>(ExpensiveToCopyType E) {
 | ||
|  |   E.constReference(); | ||
|  | } |