xref: /llvm-project/libcxx/test/std/utilities/function.objects/func.bind_front/bind_front.pass.cpp (revision c8917048e3aa2be03b6588b817730abdbce23c85)
184a50f59Szoecarver //===----------------------------------------------------------------------===//
284a50f59Szoecarver //
384a50f59Szoecarver // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
484a50f59Szoecarver // See https://llvm.org/LICENSE.txt for license information.
584a50f59Szoecarver // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
684a50f59Szoecarver //
784a50f59Szoecarver //===----------------------------------------------------------------------===//
884a50f59Szoecarver 
9e35677c0SLouis Dionne // UNSUPPORTED: c++03, c++11, c++14, c++17
1084a50f59Szoecarver 
1184a50f59Szoecarver // functional
1284a50f59Szoecarver 
13f599e7a7SLouis Dionne // template <class F, class... Args>
14f599e7a7SLouis Dionne // constexpr unspecified bind_front(F&&, Args&&...);
1584a50f59Szoecarver 
1684a50f59Szoecarver #include <functional>
17f599e7a7SLouis Dionne #include <cassert>
18*c8917048SJakub Mazurkiewicz #include <concepts>
19f599e7a7SLouis Dionne #include <tuple>
20f599e7a7SLouis Dionne #include <type_traits>
21f599e7a7SLouis Dionne #include <utility>
2284a50f59Szoecarver 
2384a50f59Szoecarver #include "callable_types.h"
2484a50f59Szoecarver #include "test_macros.h"
2584a50f59Szoecarver 
2684a50f59Szoecarver struct CopyMoveInfo {
2784a50f59Szoecarver   enum { none, copy, move } copy_kind;
2884a50f59Szoecarver 
CopyMoveInfoCopyMoveInfo2984a50f59Szoecarver   constexpr CopyMoveInfo() : copy_kind(none) {}
CopyMoveInfoCopyMoveInfo3084a50f59Szoecarver   constexpr CopyMoveInfo(CopyMoveInfo const&) : copy_kind(copy) {}
CopyMoveInfoCopyMoveInfo3184a50f59Szoecarver   constexpr CopyMoveInfo(CopyMoveInfo&&) : copy_kind(move) {}
3284a50f59Szoecarver };
3384a50f59Szoecarver 
34f599e7a7SLouis Dionne template <class ...Args>
35f599e7a7SLouis Dionne struct is_bind_frontable {
36f599e7a7SLouis Dionne   template <class ...LocalArgs>
37f599e7a7SLouis Dionne   static auto test(int)
38f599e7a7SLouis Dionne       -> decltype((void)std::bind_front(std::declval<LocalArgs>()...), std::true_type());
39f599e7a7SLouis Dionne 
40f599e7a7SLouis Dionne   template <class...>
41f599e7a7SLouis Dionne   static std::false_type test(...);
42f599e7a7SLouis Dionne 
43f599e7a7SLouis Dionne   static constexpr bool value = decltype(test<Args...>(0))::value;
44f599e7a7SLouis Dionne };
45f599e7a7SLouis Dionne 
46f599e7a7SLouis Dionne struct NotCopyMove {
47f599e7a7SLouis Dionne   NotCopyMove() = delete;
48f599e7a7SLouis Dionne   NotCopyMove(const NotCopyMove&) = delete;
49f599e7a7SLouis Dionne   NotCopyMove(NotCopyMove&&) = delete;
50f599e7a7SLouis Dionne   template <class ...Args>
operator ()NotCopyMove51f599e7a7SLouis Dionne   void operator()(Args&& ...) const { }
52f599e7a7SLouis Dionne };
53f599e7a7SLouis Dionne 
54f599e7a7SLouis Dionne struct NonConstCopyConstructible {
NonConstCopyConstructibleNonConstCopyConstructible55f599e7a7SLouis Dionne   explicit NonConstCopyConstructible() {}
NonConstCopyConstructibleNonConstCopyConstructible56f599e7a7SLouis Dionne   NonConstCopyConstructible(NonConstCopyConstructible&) {}
57f599e7a7SLouis Dionne };
58f599e7a7SLouis Dionne 
59f599e7a7SLouis Dionne struct MoveConstructible {
MoveConstructibleMoveConstructible60f599e7a7SLouis Dionne   explicit MoveConstructible() {}
MoveConstructibleMoveConstructible61f599e7a7SLouis Dionne   MoveConstructible(MoveConstructible&&) {}
62f599e7a7SLouis Dionne };
63f599e7a7SLouis Dionne 
64f599e7a7SLouis Dionne struct MakeTuple {
65f599e7a7SLouis Dionne   template <class ...Args>
operator ()MakeTuple66f599e7a7SLouis Dionne   constexpr auto operator()(Args&& ...args) const {
67f599e7a7SLouis Dionne     return std::make_tuple(std::forward<Args>(args)...);
6884a50f59Szoecarver   }
69f599e7a7SLouis Dionne };
70f599e7a7SLouis Dionne 
71f599e7a7SLouis Dionne template <int X>
72f599e7a7SLouis Dionne struct Elem {
73f599e7a7SLouis Dionne   template <int Y>
operator ==Elem74f599e7a7SLouis Dionne   constexpr bool operator==(Elem<Y> const&) const
75f599e7a7SLouis Dionne   { return X == Y; }
76f599e7a7SLouis Dionne };
77f599e7a7SLouis Dionne 
test()78f599e7a7SLouis Dionne constexpr bool test() {
79f599e7a7SLouis Dionne   // Bind arguments, call without arguments
80f599e7a7SLouis Dionne   {
81f599e7a7SLouis Dionne     {
82f599e7a7SLouis Dionne       auto f = std::bind_front(MakeTuple{});
83f599e7a7SLouis Dionne       assert(f() == std::make_tuple());
84f599e7a7SLouis Dionne     }
85f599e7a7SLouis Dionne     {
86f599e7a7SLouis Dionne       auto f = std::bind_front(MakeTuple{}, Elem<1>{});
87f599e7a7SLouis Dionne       assert(f() == std::make_tuple(Elem<1>{}));
88f599e7a7SLouis Dionne     }
89f599e7a7SLouis Dionne     {
90f599e7a7SLouis Dionne       auto f = std::bind_front(MakeTuple{}, Elem<1>{}, Elem<2>{});
91f599e7a7SLouis Dionne       assert(f() == std::make_tuple(Elem<1>{}, Elem<2>{}));
92f599e7a7SLouis Dionne     }
93f599e7a7SLouis Dionne     {
94f599e7a7SLouis Dionne       auto f = std::bind_front(MakeTuple{}, Elem<1>{}, Elem<2>{}, Elem<3>{});
95f599e7a7SLouis Dionne       assert(f() == std::make_tuple(Elem<1>{}, Elem<2>{}, Elem<3>{}));
96f599e7a7SLouis Dionne     }
9784a50f59Szoecarver   }
9884a50f59Szoecarver 
99f599e7a7SLouis Dionne   // Bind no arguments, call with arguments
100f599e7a7SLouis Dionne   {
101f599e7a7SLouis Dionne     {
102f599e7a7SLouis Dionne       auto f = std::bind_front(MakeTuple{});
103f599e7a7SLouis Dionne       assert(f(Elem<1>{}) == std::make_tuple(Elem<1>{}));
104f599e7a7SLouis Dionne     }
105f599e7a7SLouis Dionne     {
106f599e7a7SLouis Dionne       auto f = std::bind_front(MakeTuple{});
107f599e7a7SLouis Dionne       assert(f(Elem<1>{}, Elem<2>{}) == std::make_tuple(Elem<1>{}, Elem<2>{}));
108f599e7a7SLouis Dionne     }
109f599e7a7SLouis Dionne     {
110f599e7a7SLouis Dionne       auto f = std::bind_front(MakeTuple{});
111f599e7a7SLouis Dionne       assert(f(Elem<1>{}, Elem<2>{}, Elem<3>{}) == std::make_tuple(Elem<1>{}, Elem<2>{}, Elem<3>{}));
112f599e7a7SLouis Dionne     }
113f599e7a7SLouis Dionne   }
114f599e7a7SLouis Dionne 
115f599e7a7SLouis Dionne   // Bind arguments, call with arguments
116f599e7a7SLouis Dionne   {
117f599e7a7SLouis Dionne     {
118f599e7a7SLouis Dionne       auto f = std::bind_front(MakeTuple{}, Elem<1>{});
119f599e7a7SLouis Dionne       assert(f(Elem<10>{}) == std::make_tuple(Elem<1>{}, Elem<10>{}));
120f599e7a7SLouis Dionne     }
121f599e7a7SLouis Dionne     {
122f599e7a7SLouis Dionne       auto f = std::bind_front(MakeTuple{}, Elem<1>{}, Elem<2>{});
123f599e7a7SLouis Dionne       assert(f(Elem<10>{}) == std::make_tuple(Elem<1>{}, Elem<2>{}, Elem<10>{}));
124f599e7a7SLouis Dionne     }
125f599e7a7SLouis Dionne     {
126f599e7a7SLouis Dionne       auto f = std::bind_front(MakeTuple{}, Elem<1>{}, Elem<2>{}, Elem<3>{});
127f599e7a7SLouis Dionne       assert(f(Elem<10>{}) == std::make_tuple(Elem<1>{}, Elem<2>{}, Elem<3>{}, Elem<10>{}));
128f599e7a7SLouis Dionne     }
129f599e7a7SLouis Dionne 
130f599e7a7SLouis Dionne     {
131f599e7a7SLouis Dionne       auto f = std::bind_front(MakeTuple{}, Elem<1>{});
132f599e7a7SLouis Dionne       assert(f(Elem<10>{}, Elem<11>{}) == std::make_tuple(Elem<1>{}, Elem<10>{}, Elem<11>{}));
133f599e7a7SLouis Dionne     }
134f599e7a7SLouis Dionne     {
135f599e7a7SLouis Dionne       auto f = std::bind_front(MakeTuple{}, Elem<1>{}, Elem<2>{});
136f599e7a7SLouis Dionne       assert(f(Elem<10>{}, Elem<11>{}) == std::make_tuple(Elem<1>{}, Elem<2>{}, Elem<10>{}, Elem<11>{}));
137f599e7a7SLouis Dionne     }
138f599e7a7SLouis Dionne     {
139f599e7a7SLouis Dionne       auto f = std::bind_front(MakeTuple{}, Elem<1>{}, Elem<2>{}, Elem<3>{});
140f599e7a7SLouis Dionne       assert(f(Elem<10>{}, Elem<11>{}) == std::make_tuple(Elem<1>{}, Elem<2>{}, Elem<3>{}, Elem<10>{}, Elem<11>{}));
141f599e7a7SLouis Dionne     }
142f599e7a7SLouis Dionne   }
143f599e7a7SLouis Dionne 
144f599e7a7SLouis Dionne   // Basic tests with fundamental types
145f599e7a7SLouis Dionne   {
14684a50f59Szoecarver     int n         = 2;
14784a50f59Szoecarver     int m         = 1;
148*c8917048SJakub Mazurkiewicz     int sum       = 0;
149f599e7a7SLouis Dionne     auto add      = [](int x, int y) { return x + y; };
150*c8917048SJakub Mazurkiewicz     auto addN     = [](int a, int b, int c, int d, int e, int f) { return a + b + c + d + e + f; };
151*c8917048SJakub Mazurkiewicz     auto add_ref  = [&](int x, int y) -> int& { return sum = x + y; };
152*c8917048SJakub Mazurkiewicz     auto add_rref = [&](int x, int y) -> int&& { return std::move(sum = x + y); };
15384a50f59Szoecarver 
15484a50f59Szoecarver     auto a = std::bind_front(add, m, n);
15584a50f59Szoecarver     assert(a() == 3);
15684a50f59Szoecarver 
157f599e7a7SLouis Dionne     auto b = std::bind_front(addN, m, n, m, m, m, m);
15884a50f59Szoecarver     assert(b() == 7);
15984a50f59Szoecarver 
160f599e7a7SLouis Dionne     auto c = std::bind_front(addN, n, m);
16184a50f59Szoecarver     assert(c(1, 1, 1, 1) == 7);
16284a50f59Szoecarver 
163*c8917048SJakub Mazurkiewicz     auto d = std::bind_front(add_ref, n, m);
164*c8917048SJakub Mazurkiewicz     std::same_as<int&> decltype(auto) dresult(d());
165*c8917048SJakub Mazurkiewicz     assert(dresult == 3);
166*c8917048SJakub Mazurkiewicz 
167*c8917048SJakub Mazurkiewicz     auto e = std::bind_front(add_rref, n, m);
168*c8917048SJakub Mazurkiewicz     std::same_as<int&&> decltype(auto) eresult(e());
169*c8917048SJakub Mazurkiewicz     assert(eresult == 3);
170*c8917048SJakub Mazurkiewicz 
17184a50f59Szoecarver     auto f = std::bind_front(add, n);
17284a50f59Szoecarver     assert(f(3) == 5);
17384a50f59Szoecarver 
17484a50f59Szoecarver     auto g = std::bind_front(add, n, 1);
17584a50f59Szoecarver     assert(g() == 3);
17684a50f59Szoecarver 
177f599e7a7SLouis Dionne     auto h = std::bind_front(addN, 1, 1, 1);
17884a50f59Szoecarver     assert(h(2, 2, 2) == 9);
179*c8917048SJakub Mazurkiewicz 
180*c8917048SJakub Mazurkiewicz     auto i = std::bind_front(add_ref, n);
181*c8917048SJakub Mazurkiewicz     std::same_as<int&> decltype(auto) iresult(i(5));
182*c8917048SJakub Mazurkiewicz     assert(iresult == 7);
183*c8917048SJakub Mazurkiewicz 
184*c8917048SJakub Mazurkiewicz     auto j = std::bind_front(add_rref, m);
185*c8917048SJakub Mazurkiewicz     std::same_as<int&&> decltype(auto) jresult(j(4));
186*c8917048SJakub Mazurkiewicz     assert(jresult == 5);
18784a50f59Szoecarver   }
18884a50f59Szoecarver 
189f599e7a7SLouis Dionne   // Make sure we don't treat std::reference_wrapper specially.
190f599e7a7SLouis Dionne   {
191f599e7a7SLouis Dionne     auto add = [](std::reference_wrapper<int> a, std::reference_wrapper<int> b) {
192f599e7a7SLouis Dionne       return a.get() + b.get();
19384a50f59Szoecarver     };
194f599e7a7SLouis Dionne     int i = 1, j = 2;
195f599e7a7SLouis Dionne     auto f = std::bind_front(add, std::ref(i));
196f599e7a7SLouis Dionne     assert(f(std::ref(j)) == 3);
19784a50f59Szoecarver   }
19884a50f59Szoecarver 
199f599e7a7SLouis Dionne   // Make sure we can call a function that's a pointer to a member function.
200f599e7a7SLouis Dionne   {
201f599e7a7SLouis Dionne     struct MemberFunction {
202f599e7a7SLouis Dionne       constexpr bool foo(int, int) { return true; }
20384a50f59Szoecarver     };
204f599e7a7SLouis Dionne     MemberFunction value;
205f599e7a7SLouis Dionne     auto fn = std::bind_front(&MemberFunction::foo, value, 0);
20684a50f59Szoecarver     assert(fn(0));
20784a50f59Szoecarver   }
20884a50f59Szoecarver 
209f599e7a7SLouis Dionne   // Make sure that we copy the bound arguments into the unspecified-type.
210f599e7a7SLouis Dionne   {
211f599e7a7SLouis Dionne     auto add = [](int x, int y) { return x + y; };
212f599e7a7SLouis Dionne     int n = 2;
213f599e7a7SLouis Dionne     auto i = std::bind_front(add, n, 1);
214f599e7a7SLouis Dionne     n = 100;
215f599e7a7SLouis Dionne     assert(i() == 3);
21684a50f59Szoecarver   }
21784a50f59Szoecarver 
218f599e7a7SLouis Dionne   // Make sure we pass the bound arguments to the function object
219f599e7a7SLouis Dionne   // with the right value category.
220f599e7a7SLouis Dionne   {
221f599e7a7SLouis Dionne     {
222f599e7a7SLouis Dionne       auto wasCopied = [](CopyMoveInfo info) {
223f599e7a7SLouis Dionne         return info.copy_kind == CopyMoveInfo::copy;
224f599e7a7SLouis Dionne       };
225f599e7a7SLouis Dionne       CopyMoveInfo info;
226f599e7a7SLouis Dionne       auto copied = std::bind_front(wasCopied, info);
227f599e7a7SLouis Dionne       assert(copied());
228f599e7a7SLouis Dionne     }
22984a50f59Szoecarver 
230f599e7a7SLouis Dionne     {
231f599e7a7SLouis Dionne       auto wasMoved = [](CopyMoveInfo info) {
232f599e7a7SLouis Dionne         return info.copy_kind == CopyMoveInfo::move;
233f599e7a7SLouis Dionne       };
234f599e7a7SLouis Dionne       CopyMoveInfo info;
235f599e7a7SLouis Dionne       auto moved = std::bind_front(wasMoved, info);
236f599e7a7SLouis Dionne       assert(std::move(moved)());
237f599e7a7SLouis Dionne     }
238f599e7a7SLouis Dionne   }
239f599e7a7SLouis Dionne 
240f599e7a7SLouis Dionne   // Make sure we call the correctly cv-ref qualified operator() based on the
241f599e7a7SLouis Dionne   // value category of the bind_front unspecified-type.
242f599e7a7SLouis Dionne   {
243f599e7a7SLouis Dionne     struct F {
244f599e7a7SLouis Dionne       constexpr int operator()() & { return 1; }
245f599e7a7SLouis Dionne       constexpr int operator()() const& { return 2; }
246f599e7a7SLouis Dionne       constexpr int operator()() && { return 3; }
247f599e7a7SLouis Dionne       constexpr int operator()() const&& { return 4; }
248f599e7a7SLouis Dionne     };
249f599e7a7SLouis Dionne     auto x = std::bind_front(F{});
250f599e7a7SLouis Dionne     using X = decltype(x);
251f599e7a7SLouis Dionne     assert(static_cast<X&>(x)() == 1);
252f599e7a7SLouis Dionne     assert(static_cast<X const&>(x)() == 2);
253f599e7a7SLouis Dionne     assert(static_cast<X&&>(x)() == 3);
254f599e7a7SLouis Dionne     assert(static_cast<X const&&>(x)() == 4);
255f599e7a7SLouis Dionne   }
256f599e7a7SLouis Dionne 
257f599e7a7SLouis Dionne   // Make sure the bind_front unspecified-type is NOT invocable when the call would select a
258f599e7a7SLouis Dionne   // differently-qualified operator().
259f599e7a7SLouis Dionne   //
260f599e7a7SLouis Dionne   // For example, if the call to `operator()() &` is ill-formed, the call to the unspecified-type
261f599e7a7SLouis Dionne   // should be ill-formed and not fall back to the `operator()() const&` overload.
262f599e7a7SLouis Dionne   {
263f599e7a7SLouis Dionne     // Make sure we delete the & overload when the underlying call isn't valid
264f599e7a7SLouis Dionne     {
265f599e7a7SLouis Dionne       struct F {
266f599e7a7SLouis Dionne         void operator()() & = delete;
267f599e7a7SLouis Dionne         void operator()() const&;
268f599e7a7SLouis Dionne         void operator()() &&;
269f599e7a7SLouis Dionne         void operator()() const&&;
270f599e7a7SLouis Dionne       };
271f599e7a7SLouis Dionne       using X = decltype(std::bind_front(F{}));
272f599e7a7SLouis Dionne       static_assert(!std::is_invocable_v<X&>);
273f599e7a7SLouis Dionne       static_assert( std::is_invocable_v<X const&>);
274f599e7a7SLouis Dionne       static_assert( std::is_invocable_v<X>);
275f599e7a7SLouis Dionne       static_assert( std::is_invocable_v<X const>);
276f599e7a7SLouis Dionne     }
277f599e7a7SLouis Dionne 
278f599e7a7SLouis Dionne     // There's no way to make sure we delete the const& overload when the underlying call isn't valid,
279f599e7a7SLouis Dionne     // so we can't check this one.
280f599e7a7SLouis Dionne 
281f599e7a7SLouis Dionne     // Make sure we delete the && overload when the underlying call isn't valid
282f599e7a7SLouis Dionne     {
283f599e7a7SLouis Dionne       struct F {
284f599e7a7SLouis Dionne         void operator()() &;
285f599e7a7SLouis Dionne         void operator()() const&;
286f599e7a7SLouis Dionne         void operator()() && = delete;
287f599e7a7SLouis Dionne         void operator()() const&&;
288f599e7a7SLouis Dionne       };
289f599e7a7SLouis Dionne       using X = decltype(std::bind_front(F{}));
290f599e7a7SLouis Dionne       static_assert( std::is_invocable_v<X&>);
291f599e7a7SLouis Dionne       static_assert( std::is_invocable_v<X const&>);
292f599e7a7SLouis Dionne       static_assert(!std::is_invocable_v<X>);
293f599e7a7SLouis Dionne       static_assert( std::is_invocable_v<X const>);
294f599e7a7SLouis Dionne     }
295f599e7a7SLouis Dionne 
296f599e7a7SLouis Dionne     // Make sure we delete the const&& overload when the underlying call isn't valid
297f599e7a7SLouis Dionne     {
298f599e7a7SLouis Dionne       struct F {
299f599e7a7SLouis Dionne         void operator()() &;
300f599e7a7SLouis Dionne         void operator()() const&;
301f599e7a7SLouis Dionne         void operator()() &&;
302f599e7a7SLouis Dionne         void operator()() const&& = delete;
303f599e7a7SLouis Dionne       };
304f599e7a7SLouis Dionne       using X = decltype(std::bind_front(F{}));
305f599e7a7SLouis Dionne       static_assert( std::is_invocable_v<X&>);
306f599e7a7SLouis Dionne       static_assert( std::is_invocable_v<X const&>);
307f599e7a7SLouis Dionne       static_assert( std::is_invocable_v<X>);
308f599e7a7SLouis Dionne       static_assert(!std::is_invocable_v<X const>);
309f599e7a7SLouis Dionne     }
310f599e7a7SLouis Dionne   }
311f599e7a7SLouis Dionne 
312f599e7a7SLouis Dionne   // Some examples by Tim Song
313f599e7a7SLouis Dionne   {
314f599e7a7SLouis Dionne     {
315f599e7a7SLouis Dionne       struct T { };
316f599e7a7SLouis Dionne       struct F {
317f599e7a7SLouis Dionne         void operator()(T&&) const &;
318f599e7a7SLouis Dionne         void operator()(T&&) && = delete;
319f599e7a7SLouis Dionne       };
320f599e7a7SLouis Dionne       using X = decltype(std::bind_front(F{}));
321f599e7a7SLouis Dionne       static_assert(!std::is_invocable_v<X, T>);
322f599e7a7SLouis Dionne     }
323f599e7a7SLouis Dionne 
324f599e7a7SLouis Dionne     {
325f599e7a7SLouis Dionne       struct T { };
326f599e7a7SLouis Dionne       struct F {
327f599e7a7SLouis Dionne         void operator()(T const&) const;
328f599e7a7SLouis Dionne         void operator()(T&&) const = delete;
329f599e7a7SLouis Dionne       };
330f599e7a7SLouis Dionne       using X = decltype(std::bind_front(F{}, T{}));
331f599e7a7SLouis Dionne       static_assert(!std::is_invocable_v<X>);
332f599e7a7SLouis Dionne     }
333f599e7a7SLouis Dionne   }
334f599e7a7SLouis Dionne 
335f599e7a7SLouis Dionne   // Test properties of the constructor of the unspecified-type returned by bind_front.
336f599e7a7SLouis Dionne   {
33784a50f59Szoecarver     {
338c2df7076SLouis Dionne       MoveOnlyCallable<bool> value(true);
33984a50f59Szoecarver       auto ret = std::bind_front(std::move(value), 1);
34084a50f59Szoecarver       assert(ret());
34184a50f59Szoecarver       assert(ret(1, 2, 3));
34284a50f59Szoecarver 
34384a50f59Szoecarver       auto ret1 = std::move(ret);
34484a50f59Szoecarver       assert(!ret());
34584a50f59Szoecarver       assert(ret1());
34684a50f59Szoecarver       assert(ret1(1, 2, 3));
347ce96d81cSLouis Dionne 
348ce96d81cSLouis Dionne       using RetT = decltype(ret);
349ce96d81cSLouis Dionne       static_assert( std::is_move_constructible<RetT>::value);
350ce96d81cSLouis Dionne       static_assert(!std::is_copy_constructible<RetT>::value);
351ce96d81cSLouis Dionne       static_assert(!std::is_move_assignable<RetT>::value);
352ce96d81cSLouis Dionne       static_assert(!std::is_copy_assignable<RetT>::value);
35384a50f59Szoecarver     }
35484a50f59Szoecarver     {
355c2df7076SLouis Dionne       CopyCallable<bool> value(true);
35684a50f59Szoecarver       auto ret = std::bind_front(value, 1);
35784a50f59Szoecarver       assert(ret());
35884a50f59Szoecarver       assert(ret(1, 2, 3));
35984a50f59Szoecarver 
36084a50f59Szoecarver       auto ret1 = std::move(ret);
36184a50f59Szoecarver       assert(ret1());
36284a50f59Szoecarver       assert(ret1(1, 2, 3));
36384a50f59Szoecarver 
36484a50f59Szoecarver       auto ret2 = std::bind_front(std::move(value), 1);
36584a50f59Szoecarver       assert(!ret());
36684a50f59Szoecarver       assert(ret2());
36784a50f59Szoecarver       assert(ret2(1, 2, 3));
368ce96d81cSLouis Dionne 
369ce96d81cSLouis Dionne       using RetT = decltype(ret);
370ce96d81cSLouis Dionne       static_assert( std::is_move_constructible<RetT>::value);
371ce96d81cSLouis Dionne       static_assert( std::is_copy_constructible<RetT>::value);
372ce96d81cSLouis Dionne       static_assert(!std::is_move_assignable<RetT>::value);
373ce96d81cSLouis Dionne       static_assert(!std::is_copy_assignable<RetT>::value);
37484a50f59Szoecarver     }
37584a50f59Szoecarver     {
37684a50f59Szoecarver       CopyAssignableWrapper value(true);
37784a50f59Szoecarver       using RetT = decltype(std::bind_front(value, 1));
37884a50f59Szoecarver 
37984a50f59Szoecarver       static_assert(std::is_move_constructible<RetT>::value);
38084a50f59Szoecarver       static_assert(std::is_copy_constructible<RetT>::value);
38184a50f59Szoecarver       static_assert(std::is_move_assignable<RetT>::value);
38284a50f59Szoecarver       static_assert(std::is_copy_assignable<RetT>::value);
38384a50f59Szoecarver     }
38484a50f59Szoecarver     {
38584a50f59Szoecarver       MoveAssignableWrapper value(true);
38684a50f59Szoecarver       using RetT = decltype(std::bind_front(std::move(value), 1));
38784a50f59Szoecarver 
38884a50f59Szoecarver       static_assert( std::is_move_constructible<RetT>::value);
38984a50f59Szoecarver       static_assert(!std::is_copy_constructible<RetT>::value);
39084a50f59Szoecarver       static_assert( std::is_move_assignable<RetT>::value);
39184a50f59Szoecarver       static_assert(!std::is_copy_assignable<RetT>::value);
39284a50f59Szoecarver     }
39384a50f59Szoecarver   }
39484a50f59Szoecarver 
395f599e7a7SLouis Dionne   // Make sure bind_front is SFINAE friendly
396f599e7a7SLouis Dionne   {
397f599e7a7SLouis Dionne     static_assert(!std::is_constructible_v<NotCopyMove, NotCopyMove&>);
39884a50f59Szoecarver     static_assert(!std::is_move_constructible_v<NotCopyMove>);
39984a50f59Szoecarver     static_assert(!is_bind_frontable<NotCopyMove>::value);
40084a50f59Szoecarver     static_assert(!is_bind_frontable<NotCopyMove&>::value);
40184a50f59Szoecarver 
402f599e7a7SLouis Dionne     auto takeAnything = [](auto&& ...) { };
403f599e7a7SLouis Dionne     static_assert(!std::is_constructible_v<MoveConstructible, MoveConstructible&>);
40484a50f59Szoecarver     static_assert( std::is_move_constructible_v<MoveConstructible>);
405f599e7a7SLouis Dionne     static_assert( is_bind_frontable<decltype(takeAnything), MoveConstructible>::value);
406f599e7a7SLouis Dionne     static_assert(!is_bind_frontable<decltype(takeAnything), MoveConstructible&>::value);
40784a50f59Szoecarver 
408f599e7a7SLouis Dionne     static_assert( std::is_constructible_v<NonConstCopyConstructible, NonConstCopyConstructible&>);
40984a50f59Szoecarver     static_assert(!std::is_move_constructible_v<NonConstCopyConstructible>);
410f599e7a7SLouis Dionne     static_assert(!is_bind_frontable<decltype(takeAnything), NonConstCopyConstructible&>::value);
411f599e7a7SLouis Dionne     static_assert(!is_bind_frontable<decltype(takeAnything), NonConstCopyConstructible>::value);
41284a50f59Szoecarver   }
41384a50f59Szoecarver 
41489a7bdb1SLouis Dionne   // Make sure bind_front's unspecified type's operator() is SFINAE-friendly
41589a7bdb1SLouis Dionne   {
41689a7bdb1SLouis Dionne     using T = decltype(std::bind_front(std::declval<int(*)(int, int)>(), 1));
41789a7bdb1SLouis Dionne     static_assert(!std::is_invocable<T>::value);
41889a7bdb1SLouis Dionne     static_assert( std::is_invocable<T, int>::value);
41989a7bdb1SLouis Dionne     static_assert(!std::is_invocable<T, void*>::value);
42089a7bdb1SLouis Dionne     static_assert(!std::is_invocable<T, int, int>::value);
42189a7bdb1SLouis Dionne   }
42289a7bdb1SLouis Dionne 
42384a50f59Szoecarver   return true;
42484a50f59Szoecarver }
42584a50f59Szoecarver 
main(int,char **)42684a50f59Szoecarver int main(int, char**) {
42784a50f59Szoecarver   test();
42884a50f59Szoecarver   static_assert(test());
42984a50f59Szoecarver 
43084a50f59Szoecarver   return 0;
43184a50f59Szoecarver }
432