2019-05-20 17:26:47 +08:00
|
|
|
// RUN: %check_clang_tidy %s modernize-pass-by-value %t -- -- -fno-delayed-template-parsing
|
2015-08-14 21:17:11 +08:00
|
|
|
|
|
|
|
namespace {
|
|
|
|
// POD types are trivially move constructible.
|
2017-01-13 03:20:35 +08:00
|
|
|
struct POD {
|
|
|
|
int a, b, c;
|
|
|
|
};
|
|
|
|
|
2015-08-14 21:17:11 +08:00
|
|
|
struct Movable {
|
|
|
|
int a, b, c;
|
2017-01-13 03:20:35 +08:00
|
|
|
Movable() = default;
|
|
|
|
Movable(const Movable &) {}
|
|
|
|
Movable(Movable &&) {}
|
2015-08-14 21:17:11 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
struct NotMovable {
|
|
|
|
NotMovable() = default;
|
|
|
|
NotMovable(const NotMovable &) = default;
|
|
|
|
NotMovable(NotMovable &&) = delete;
|
|
|
|
int a, b, c;
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
struct A {
|
2016-05-24 23:13:44 +08:00
|
|
|
A(const Movable &M) : M(M) {}
|
2015-08-14 21:17:11 +08:00
|
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:5: warning: pass by value and use std::move [modernize-pass-by-value]
|
|
|
|
// CHECK-FIXES: A(Movable M) : M(std::move(M)) {}
|
|
|
|
Movable M;
|
|
|
|
};
|
|
|
|
|
|
|
|
// Test that we aren't modifying other things than a parameter.
|
|
|
|
Movable GlobalObj;
|
|
|
|
struct B {
|
|
|
|
B(const Movable &M) : M(GlobalObj) {}
|
|
|
|
// CHECK-FIXES: B(const Movable &M) : M(GlobalObj) {}
|
|
|
|
Movable M;
|
|
|
|
};
|
|
|
|
|
|
|
|
// Test that a parameter with more than one reference to it won't be changed.
|
|
|
|
struct C {
|
|
|
|
// Tests extra-reference in body.
|
|
|
|
C(const Movable &M) : M(M) { this->i = M.a; }
|
|
|
|
// CHECK-FIXES: C(const Movable &M) : M(M) { this->i = M.a; }
|
|
|
|
|
|
|
|
// Tests extra-reference in init-list.
|
|
|
|
C(const Movable &M, int) : M(M), i(M.a) {}
|
|
|
|
// CHECK-FIXES: C(const Movable &M, int) : M(M), i(M.a) {}
|
|
|
|
Movable M;
|
|
|
|
int i;
|
|
|
|
};
|
|
|
|
|
|
|
|
// Test that both declaration and definition are updated.
|
|
|
|
struct D {
|
2016-05-24 23:13:44 +08:00
|
|
|
D(const Movable &M);
|
2015-08-14 21:17:11 +08:00
|
|
|
// CHECK-FIXES: D(Movable M);
|
|
|
|
Movable M;
|
|
|
|
};
|
2016-05-24 23:13:44 +08:00
|
|
|
D::D(const Movable &M) : M(M) {}
|
2015-08-14 21:17:11 +08:00
|
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:6: warning: pass by value and use std::move
|
|
|
|
// CHECK-FIXES: D::D(Movable M) : M(std::move(M)) {}
|
|
|
|
|
|
|
|
// Test with default parameter.
|
|
|
|
struct E {
|
2016-05-24 23:13:44 +08:00
|
|
|
E(const Movable &M = Movable()) : M(M) {}
|
2015-08-14 21:17:11 +08:00
|
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:5: warning: pass by value and use std::move
|
|
|
|
// CHECK-FIXES: E(Movable M = Movable()) : M(std::move(M)) {}
|
|
|
|
Movable M;
|
|
|
|
};
|
|
|
|
|
|
|
|
// Test with object that can't be moved.
|
|
|
|
struct F {
|
|
|
|
F(const NotMovable &NM) : NM(NM) {}
|
|
|
|
// CHECK-FIXES: F(const NotMovable &NM) : NM(NM) {}
|
|
|
|
NotMovable NM;
|
|
|
|
};
|
|
|
|
|
|
|
|
// Test unnamed parameter in declaration.
|
|
|
|
struct G {
|
2016-05-24 23:13:44 +08:00
|
|
|
G(const Movable &);
|
2015-08-14 21:17:11 +08:00
|
|
|
// CHECK-FIXES: G(Movable );
|
|
|
|
Movable M;
|
|
|
|
};
|
2016-05-24 23:13:44 +08:00
|
|
|
G::G(const Movable &M) : M(M) {}
|
2015-08-14 21:17:11 +08:00
|
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:6: warning: pass by value and use std::move
|
|
|
|
// CHECK-FIXES: G::G(Movable M) : M(std::move(M)) {}
|
|
|
|
|
|
|
|
// Test parameter with and without qualifier.
|
|
|
|
namespace ns_H {
|
|
|
|
typedef ::Movable HMovable;
|
|
|
|
}
|
|
|
|
struct H {
|
2016-05-24 23:13:44 +08:00
|
|
|
H(const ns_H::HMovable &M);
|
2015-08-14 21:17:11 +08:00
|
|
|
// CHECK-FIXES: H(ns_H::HMovable M);
|
|
|
|
ns_H::HMovable M;
|
|
|
|
};
|
|
|
|
using namespace ns_H;
|
2016-05-24 23:13:44 +08:00
|
|
|
H::H(const HMovable &M) : M(M) {}
|
2015-08-14 21:17:11 +08:00
|
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:6: warning: pass by value and use std::move
|
|
|
|
// CHECK-FIXES: H(HMovable M) : M(std::move(M)) {}
|
|
|
|
|
|
|
|
// Try messing up with macros.
|
|
|
|
#define MOVABLE_PARAM(Name) const Movable & Name
|
|
|
|
// CHECK-FIXES: #define MOVABLE_PARAM(Name) const Movable & Name
|
|
|
|
struct I {
|
|
|
|
I(MOVABLE_PARAM(M)) : M(M) {}
|
|
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:5: warning: pass by value and use std::move
|
|
|
|
// CHECK-FIXES: I(MOVABLE_PARAM(M)) : M(M) {}
|
|
|
|
Movable M;
|
|
|
|
};
|
|
|
|
#undef MOVABLE_PARAM
|
|
|
|
|
|
|
|
// Test that templates aren't modified.
|
|
|
|
template <typename T> struct J {
|
|
|
|
J(const T &M) : M(M) {}
|
|
|
|
// CHECK-FIXES: J(const T &M) : M(M) {}
|
|
|
|
T M;
|
|
|
|
};
|
|
|
|
J<Movable> j1(Movable());
|
|
|
|
J<NotMovable> j2(NotMovable());
|
|
|
|
|
2020-02-28 22:17:16 +08:00
|
|
|
template<class T>
|
|
|
|
struct MovableTemplateT
|
|
|
|
{
|
|
|
|
MovableTemplateT() {}
|
|
|
|
MovableTemplateT(const MovableTemplateT& o) { }
|
|
|
|
MovableTemplateT(MovableTemplateT&& o) { }
|
|
|
|
};
|
|
|
|
|
|
|
|
template <class T>
|
|
|
|
struct J2 {
|
|
|
|
J2(const MovableTemplateT<T>& A);
|
|
|
|
// CHECK-FIXES: J2(const MovableTemplateT<T>& A);
|
|
|
|
MovableTemplateT<T> M;
|
|
|
|
};
|
|
|
|
|
|
|
|
template <class T>
|
|
|
|
J2<T>::J2(const MovableTemplateT<T>& A) : M(A) {}
|
|
|
|
// CHECK-FIXES: J2<T>::J2(const MovableTemplateT<T>& A) : M(A) {}
|
|
|
|
J2<int> j3(MovableTemplateT<int>{});
|
|
|
|
|
2015-08-14 21:17:11 +08:00
|
|
|
struct K_Movable {
|
|
|
|
K_Movable() = default;
|
|
|
|
K_Movable(const K_Movable &) = default;
|
|
|
|
K_Movable(K_Movable &&o) { dummy = o.dummy; }
|
|
|
|
int dummy;
|
|
|
|
};
|
|
|
|
|
|
|
|
// Test with movable type with an user defined move constructor.
|
|
|
|
struct K {
|
2016-05-24 23:13:44 +08:00
|
|
|
K(const K_Movable &M) : M(M) {}
|
2015-08-14 21:17:11 +08:00
|
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:5: warning: pass by value and use std::move
|
|
|
|
// CHECK-FIXES: K(K_Movable M) : M(std::move(M)) {}
|
|
|
|
K_Movable M;
|
|
|
|
};
|
|
|
|
|
|
|
|
template <typename T> struct L {
|
2016-05-24 23:13:44 +08:00
|
|
|
L(const Movable &M) : M(M) {}
|
2015-08-14 21:17:11 +08:00
|
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:5: warning: pass by value and use std::move
|
|
|
|
// CHECK-FIXES: L(Movable M) : M(std::move(M)) {}
|
|
|
|
Movable M;
|
|
|
|
};
|
|
|
|
L<int> l(Movable());
|
|
|
|
|
|
|
|
// Test with a non-instantiated template class.
|
|
|
|
template <typename T> struct N {
|
2016-05-24 23:13:44 +08:00
|
|
|
N(const Movable &M) : M(M) {}
|
2015-08-14 21:17:11 +08:00
|
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:5: warning: pass by value and use std::move
|
|
|
|
// CHECK-FIXES: N(Movable M) : M(std::move(M)) {}
|
|
|
|
|
|
|
|
Movable M;
|
|
|
|
T A;
|
|
|
|
};
|
|
|
|
|
|
|
|
// Test with value parameter.
|
|
|
|
struct O {
|
2016-05-24 23:13:44 +08:00
|
|
|
O(Movable M) : M(M) {}
|
2017-01-13 03:20:35 +08:00
|
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:5: warning: pass by value and use std::move
|
|
|
|
// CHECK-FIXES: O(Movable M) : M(std::move(M)) {}
|
2015-08-14 21:17:11 +08:00
|
|
|
Movable M;
|
|
|
|
};
|
|
|
|
|
|
|
|
// Test with a const-value parameter.
|
|
|
|
struct P {
|
|
|
|
P(const Movable M) : M(M) {}
|
|
|
|
// CHECK-FIXES: P(const Movable M) : M(M) {}
|
|
|
|
Movable M;
|
|
|
|
};
|
|
|
|
|
|
|
|
// Test with multiples parameters where some need to be changed and some don't.
|
|
|
|
// need to.
|
|
|
|
struct Q {
|
2016-05-24 23:13:44 +08:00
|
|
|
Q(const Movable &A, const Movable &B, const Movable &C, double D)
|
|
|
|
: A(A), B(B), C(C), D(D) {}
|
2015-08-14 21:17:11 +08:00
|
|
|
// CHECK-MESSAGES: :[[@LINE-2]]:23: warning: pass by value and use std::move
|
|
|
|
// CHECK-MESSAGES: :[[@LINE-3]]:41: warning: pass by value and use std::move
|
|
|
|
// CHECK-FIXES: Q(const Movable &A, Movable B, Movable C, double D)
|
|
|
|
// CHECK-FIXES: : A(A), B(std::move(B)), C(std::move(C)), D(D) {}
|
|
|
|
const Movable &A;
|
|
|
|
Movable B;
|
|
|
|
Movable C;
|
|
|
|
double D;
|
|
|
|
};
|
|
|
|
|
|
|
|
// Test that value-parameters with a nested name specifier are left as-is.
|
|
|
|
namespace ns_R {
|
|
|
|
typedef ::Movable RMovable;
|
|
|
|
}
|
|
|
|
struct R {
|
2016-05-24 23:13:44 +08:00
|
|
|
R(ns_R::RMovable M) : M(M) {}
|
2017-01-13 03:20:35 +08:00
|
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:5: warning: pass by value and use std::move
|
|
|
|
// CHECK-FIXES: R(ns_R::RMovable M) : M(std::move(M)) {}
|
2015-08-14 21:17:11 +08:00
|
|
|
ns_R::RMovable M;
|
|
|
|
};
|
|
|
|
|
|
|
|
// Test with rvalue parameter.
|
|
|
|
struct S {
|
|
|
|
S(Movable &&M) : M(M) {}
|
|
|
|
// CHECK-FIXES: S(Movable &&M) : M(M) {}
|
|
|
|
Movable M;
|
|
|
|
};
|
2016-05-24 23:00:16 +08:00
|
|
|
|
2016-05-25 00:54:26 +08:00
|
|
|
template <typename T, int N> struct array { T A[N]; };
|
|
|
|
|
2016-05-24 23:00:16 +08:00
|
|
|
// Test that types that are trivially copyable will not use std::move. This will
|
2017-11-29 00:41:03 +08:00
|
|
|
// cause problems with performance-move-const-arg, as it will revert it.
|
2016-05-24 23:00:16 +08:00
|
|
|
struct T {
|
2016-05-25 00:54:26 +08:00
|
|
|
T(array<int, 10> a) : a_(a) {}
|
|
|
|
// CHECK-FIXES: T(array<int, 10> a) : a_(a) {}
|
|
|
|
array<int, 10> a_;
|
2016-05-24 23:00:16 +08:00
|
|
|
};
|
2017-01-13 03:20:35 +08:00
|
|
|
|
|
|
|
struct U {
|
|
|
|
U(const POD &M) : M(M) {}
|
2021-12-09 02:31:30 +08:00
|
|
|
// CHECK-FIXES: U(const POD &M) : M(M) {}
|
2017-01-13 03:20:35 +08:00
|
|
|
POD M;
|
|
|
|
};
|
2021-12-09 02:31:30 +08:00
|
|
|
|
|
|
|
// The rewrite can't look through `typedefs` and `using`.
|
|
|
|
// Test that we don't partially rewrite one decl without rewriting the other.
|
|
|
|
using MovableConstRef = const Movable &;
|
|
|
|
struct V {
|
|
|
|
V(MovableConstRef M);
|
|
|
|
// CHECK-FIXES: V(MovableConstRef M);
|
|
|
|
Movable M;
|
|
|
|
};
|
|
|
|
V::V(const Movable &M) : M(M) {}
|
|
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:6: warning: pass by value and use std::move
|
|
|
|
// CHECK-FIXES: V::V(const Movable &M) : M(M) {}
|
2022-01-03 21:25:29 +08:00
|
|
|
|
|
|
|
// Test with paired lvalue/rvalue overloads.
|
|
|
|
struct W1 {
|
|
|
|
W1(const Movable &M) : M(M) {}
|
|
|
|
W1(Movable &&M);
|
|
|
|
Movable M;
|
|
|
|
};
|
|
|
|
struct W2 {
|
|
|
|
W2(const Movable &M, int) : M(M) {}
|
|
|
|
W2(Movable &&M, int);
|
|
|
|
Movable M;
|
|
|
|
};
|
|
|
|
struct W3 {
|
|
|
|
W3(const W1 &, const Movable &M) : M(M) {}
|
|
|
|
W3(W1 &&, Movable &&M);
|
|
|
|
Movable M;
|
|
|
|
};
|