xref: /llvm-project/clang-tools-extra/test/clang-tidy/checkers/modernize/use-transparent-functors.cpp (revision 89a1d03e2b379e325daa5249411e414bbd995b5e)
1*89a1d03eSRichard // RUN: %check_clang_tidy -std=c++14-or-later %s modernize-use-transparent-functors %t
2*89a1d03eSRichard 
3*89a1d03eSRichard namespace std {
4*89a1d03eSRichard template<class T>
5*89a1d03eSRichard struct remove_reference;
6*89a1d03eSRichard 
7*89a1d03eSRichard template <class T>
8*89a1d03eSRichard constexpr T &&forward(typename std::remove_reference<T>::type &t);
9*89a1d03eSRichard 
10*89a1d03eSRichard template <class T>
11*89a1d03eSRichard constexpr T &&forward(typename std::remove_reference<T>::type &&t);
12*89a1d03eSRichard 
13*89a1d03eSRichard template <typename T = void>
14*89a1d03eSRichard struct plus {
15*89a1d03eSRichard   constexpr T operator()(const T &Lhs, const T &Rhs) const;
16*89a1d03eSRichard };
17*89a1d03eSRichard 
18*89a1d03eSRichard template <>
19*89a1d03eSRichard struct plus<void> {
20*89a1d03eSRichard   template <typename T, typename U>
21*89a1d03eSRichard   constexpr auto operator()(T &&Lhs, U &&Rhs) const ->
22*89a1d03eSRichard     decltype(forward<T>(Lhs) + forward<U>(Rhs));
23*89a1d03eSRichard };
24*89a1d03eSRichard 
25*89a1d03eSRichard template <typename T = void>
26*89a1d03eSRichard struct less {
27*89a1d03eSRichard   constexpr bool operator()(const T &Lhs, const T &Rhs) const;
28*89a1d03eSRichard };
29*89a1d03eSRichard 
30*89a1d03eSRichard template <>
31*89a1d03eSRichard struct less<void> {
32*89a1d03eSRichard   template <typename T, typename U>
33*89a1d03eSRichard   constexpr bool operator()(T &&Lhs, U &&Rhs) const;
34*89a1d03eSRichard };
35*89a1d03eSRichard 
36*89a1d03eSRichard template <typename T = void>
37*89a1d03eSRichard struct logical_not {
38*89a1d03eSRichard   constexpr bool operator()(const T &Arg) const;
39*89a1d03eSRichard };
40*89a1d03eSRichard 
41*89a1d03eSRichard template <>
42*89a1d03eSRichard struct logical_not<void> {
43*89a1d03eSRichard   template <typename T>
44*89a1d03eSRichard   constexpr bool operator()(T &&Arg) const;
45*89a1d03eSRichard };
46*89a1d03eSRichard 
47*89a1d03eSRichard template <typename T>
48*89a1d03eSRichard class allocator;
49*89a1d03eSRichard 
50*89a1d03eSRichard template <
51*89a1d03eSRichard     class Key,
52*89a1d03eSRichard     class Compare = std::less<>,
53*89a1d03eSRichard     class Allocator = std::allocator<Key>>
54*89a1d03eSRichard class set {};
55*89a1d03eSRichard 
56*89a1d03eSRichard template <
57*89a1d03eSRichard     class Key,
58*89a1d03eSRichard     class Compare = std::less<Key>,
59*89a1d03eSRichard     class Allocator = std::allocator<Key>>
60*89a1d03eSRichard class set2 {};
61*89a1d03eSRichard 
62*89a1d03eSRichard template <class InputIt, class UnaryPredicate>
63*89a1d03eSRichard InputIt find_if(InputIt first, InputIt last,
64*89a1d03eSRichard                 UnaryPredicate p);
65*89a1d03eSRichard 
66*89a1d03eSRichard template <class RandomIt, class Compare>
67*89a1d03eSRichard void sort(RandomIt first, RandomIt last, Compare comp);
68*89a1d03eSRichard 
69*89a1d03eSRichard class iterator {};
70*89a1d03eSRichard class string {};
71*89a1d03eSRichard }
72*89a1d03eSRichard 
main()73*89a1d03eSRichard int main() {
74*89a1d03eSRichard   using std::set;
75*89a1d03eSRichard   using std::less;
76*89a1d03eSRichard   std::set<int, std::less<int>> s;
77*89a1d03eSRichard   // CHECK-MESSAGES: :[[@LINE-1]]:17: warning: prefer transparent functors 'less<>' [modernize-use-transparent-functors]
78*89a1d03eSRichard   // CHECK-FIXES: {{^}}  std::set<int, std::less<>> s;{{$}}
79*89a1d03eSRichard   set<int, std::less<int>> s2;
80*89a1d03eSRichard   // CHECK-MESSAGES: :[[@LINE-1]]:12: warning: prefer transparent functors
81*89a1d03eSRichard   // CHECK-FIXES: {{^}}  set<int, std::less<>> s2;{{$}}
82*89a1d03eSRichard   set<int, less<int>> s3;
83*89a1d03eSRichard   // CHECK-MESSAGES: :[[@LINE-1]]:12: warning: prefer transparent functors
84*89a1d03eSRichard   // CHECK-FIXES: {{^}}  set<int, less<>> s3;{{$}}
85*89a1d03eSRichard   std::set<int, std::less<>> s4;
86*89a1d03eSRichard   std::set<char *, std::less<std::string>> s5;
87*89a1d03eSRichard   std::set<set<int, less<int>>, std::less<>> s6;
88*89a1d03eSRichard   // CHECK-MESSAGES: :[[@LINE-1]]:21: warning: prefer transparent functors
89*89a1d03eSRichard   // CHECK-FIXES: {{^}}  std::set<set<int, less<>>, std::less<>> s6;{{$}}
90*89a1d03eSRichard   std::iterator begin, end;
91*89a1d03eSRichard   sort(begin, end, std::less<int>());
92*89a1d03eSRichard   // CHECK-MESSAGES: :[[@LINE-1]]:20: warning: prefer transparent functors
93*89a1d03eSRichard   std::sort(begin, end, std::less<>());
94*89a1d03eSRichard   find_if(begin, end, std::logical_not<bool>());
95*89a1d03eSRichard   // CHECK-MESSAGES: :[[@LINE-1]]:23: warning: prefer transparent functors
96*89a1d03eSRichard   std::find_if(begin, end, std::logical_not<>());
97*89a1d03eSRichard   using my_set = std::set<int, std::less<int>>;
98*89a1d03eSRichard   // CHECK-MESSAGES: :[[@LINE-1]]:32: warning: prefer transparent functors
99*89a1d03eSRichard   // CHECK-FIXES: {{^}}  using my_set = std::set<int, std::less<>>;{{$}}
100*89a1d03eSRichard   using my_set2 = std::set<char*, std::less<std::string>>;
101*89a1d03eSRichard   using my_less = std::less<std::string>;
102*89a1d03eSRichard   find_if(begin, end, my_less());
103*89a1d03eSRichard   // CHECK-MESSAGES: :[[@LINE-1]]:23: warning: prefer transparent functors
104*89a1d03eSRichard   std::set2<int> control;
105*89a1d03eSRichard }
106*89a1d03eSRichard 
107*89a1d03eSRichard struct ImplicitTypeLoc : std::set2<std::less<int>> {
108*89a1d03eSRichard   // CHECK-MESSAGES: :[[@LINE-1]]:36: warning: prefer transparent functors
ImplicitTypeLocImplicitTypeLoc109*89a1d03eSRichard   ImplicitTypeLoc() {}
110*89a1d03eSRichard };
111