2017-02-16 00:32:55 +08:00
|
|
|
// RUN: %check_clang_tidy %s misc-unconventional-assign-operator %t -- -- -std=c++11 -isystem %S/Inputs/Headers -fno-delayed-template-parsing
|
2016-05-04 20:02:22 +08:00
|
|
|
|
2016-05-04 20:17:55 +08:00
|
|
|
namespace std {
|
|
|
|
template <typename T>
|
|
|
|
struct remove_reference { typedef T type; };
|
|
|
|
template <typename T>
|
|
|
|
struct remove_reference<T &> { typedef T type; };
|
|
|
|
template <typename T>
|
|
|
|
struct remove_reference<T &&> { typedef T type; };
|
|
|
|
template <typename T>
|
|
|
|
typename remove_reference<T>::type &&move(T &&t);
|
|
|
|
}
|
|
|
|
|
2015-02-10 01:50:40 +08:00
|
|
|
|
|
|
|
struct Good {
|
|
|
|
Good& operator=(const Good&);
|
|
|
|
Good& operator=(Good&&);
|
|
|
|
|
|
|
|
// Assign from other types is fine too.
|
|
|
|
Good& operator=(int);
|
|
|
|
};
|
|
|
|
|
|
|
|
struct AlsoGood {
|
|
|
|
// By value is also fine.
|
|
|
|
AlsoGood& operator=(AlsoGood);
|
|
|
|
};
|
|
|
|
|
2016-05-04 20:02:22 +08:00
|
|
|
struct BadReturnType {
|
|
|
|
void operator=(const BadReturnType&);
|
|
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:3: warning: operator=() should return 'BadReturnType&' [misc-unconventional-assign-operator]
|
|
|
|
const BadReturnType& operator=(BadReturnType&&);
|
2015-02-10 01:50:40 +08:00
|
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:3: warning: operator=() should return 'Bad
|
2016-03-24 18:12:08 +08:00
|
|
|
void operator=(int);
|
|
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:3: warning: operator=() should return 'Bad
|
2015-02-10 01:50:40 +08:00
|
|
|
};
|
2016-05-04 20:02:22 +08:00
|
|
|
|
|
|
|
struct BadReturnType2 {
|
|
|
|
BadReturnType2&& operator=(const BadReturnType2&);
|
2015-02-10 01:50:40 +08:00
|
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:3: warning: operator=() should return 'Bad
|
2016-05-04 20:02:22 +08:00
|
|
|
int operator=(BadReturnType2&&);
|
2015-02-10 01:50:40 +08:00
|
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:3: warning: operator=() should return 'Bad
|
|
|
|
};
|
|
|
|
|
|
|
|
struct BadArgument {
|
|
|
|
BadArgument& operator=(BadArgument&);
|
|
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:3: warning: operator=() should take 'BadArgument const&', 'BadArgument&&' or 'BadArgument'
|
|
|
|
BadArgument& operator=(const BadArgument&&);
|
|
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:3: warning: operator=() should take 'BadAr
|
|
|
|
};
|
|
|
|
|
|
|
|
struct BadModifier {
|
|
|
|
BadModifier& operator=(const BadModifier&) const;
|
|
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:3: warning: operator=() should not be marked 'const'
|
|
|
|
};
|
|
|
|
|
|
|
|
struct Deleted {
|
|
|
|
// We don't check the return value of deleted operators.
|
|
|
|
void operator=(const Deleted&) = delete;
|
|
|
|
void operator=(Deleted&&) = delete;
|
|
|
|
};
|
|
|
|
|
|
|
|
class Private {
|
|
|
|
// We don't check the return value of private operators.
|
|
|
|
// Pre-C++11 way of disabling assignment.
|
|
|
|
void operator=(const Private &);
|
|
|
|
};
|
2015-10-13 23:24:33 +08:00
|
|
|
|
|
|
|
struct Virtual {
|
|
|
|
virtual Virtual& operator=(const Virtual &);
|
|
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:3: warning: operator=() should not be marked 'virtual'
|
|
|
|
};
|
2016-05-04 20:02:22 +08:00
|
|
|
|
|
|
|
class BadReturnStatement {
|
|
|
|
int n;
|
|
|
|
|
|
|
|
public:
|
|
|
|
BadReturnStatement& operator=(BadReturnStatement&& rhs) {
|
|
|
|
n = std::move(rhs.n);
|
|
|
|
return rhs;
|
|
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:5: warning: operator=() should always return '*this'
|
|
|
|
}
|
|
|
|
|
|
|
|
// Do not check if return type is different from '&BadReturnStatement'
|
|
|
|
int operator=(int i) {
|
|
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:3: warning: operator=() should return 'Bad
|
|
|
|
n = i;
|
|
|
|
return n;
|
|
|
|
}
|
|
|
|
};
|
2017-02-15 22:01:41 +08:00
|
|
|
|
|
|
|
namespace pr31531 {
|
|
|
|
enum E { e };
|
|
|
|
// This declaration makes the 'return *this' below have an unresolved operator
|
|
|
|
// in the class template, but not in an instantiation.
|
|
|
|
E operator*(E, E);
|
|
|
|
|
|
|
|
template <typename>
|
|
|
|
struct UnresolvedOperator {
|
|
|
|
UnresolvedOperator &operator=(const UnresolvedOperator &) { return *this; }
|
|
|
|
};
|
|
|
|
|
|
|
|
UnresolvedOperator<int> UnresolvedOperatorInt;
|
|
|
|
|
|
|
|
template <typename>
|
|
|
|
struct Template {
|
|
|
|
Template &operator=(const Template &) { return this; }
|
|
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:43: warning: operator=() should always return '*this'
|
|
|
|
};
|
|
|
|
|
|
|
|
Template<int> TemplateInt;
|
|
|
|
}
|