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