1 | //===----------------------------------------------------------------------===// |
2 | // |
3 | // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. |
4 | // See https://llvm.org/LICENSE.txt for license information. |
5 | // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception |
6 | // |
7 | //===----------------------------------------------------------------------===// |
8 | |
9 | // <functional> |
10 | |
11 | // reference_wrapper |
12 | |
13 | // has weak result type |
14 | |
15 | // REQUIRES: c++03 || c++11 || c++14 || c++17 |
16 | |
17 | // ADDITIONAL_COMPILE_FLAGS: -D_LIBCPP_DISABLE_DEPRECATION_WARNINGS |
18 | |
19 | #include <functional> |
20 | #include <type_traits> |
21 | |
22 | #include "test_macros.h" |
23 | |
24 | template <class Arg, class Result> |
25 | struct my_unary_function |
26 | { // std::unary_function was removed in C++17 |
27 | typedef Arg argument_type; |
28 | typedef Result result_type; |
29 | }; |
30 | |
31 | template <class Arg1, class Arg2, class Result> |
32 | struct my_binary_function |
33 | { // std::binary_function was removed in C++17 |
34 | typedef Arg1 first_argument_type; |
35 | typedef Arg2 second_argument_type; |
36 | typedef Result result_type; |
37 | }; |
38 | |
39 | class functor1 |
40 | : public my_unary_function<int, char> |
41 | { |
42 | }; |
43 | |
44 | class functor2 |
45 | : public my_binary_function<char, int, double> |
46 | { |
47 | }; |
48 | |
49 | class functor3 |
50 | : public my_unary_function<char, int>, |
51 | public my_binary_function<char, int, double> |
52 | { |
53 | public: |
54 | typedef float result_type; |
55 | }; |
56 | |
57 | class functor4 |
58 | : public my_unary_function<char, int>, |
59 | public my_binary_function<char, int, double> |
60 | { |
61 | public: |
62 | }; |
63 | |
64 | class C {}; |
65 | |
66 | template <class T> |
67 | struct has_result_type |
68 | { |
69 | private: |
70 | struct two {char _; char __;}; |
71 | template <class U> static two test(...); |
72 | template <class U> static char test(typename U::result_type* = 0); |
73 | public: |
74 | static const bool value = sizeof(test<T>(0)) == 1; |
75 | }; |
76 | |
77 | int main(int, char**) |
78 | { |
79 | static_assert((std::is_same<std::reference_wrapper<functor1>::result_type, |
80 | char>::value), "" ); |
81 | static_assert((std::is_same<std::reference_wrapper<functor2>::result_type, |
82 | double>::value), "" ); |
83 | static_assert((std::is_same<std::reference_wrapper<functor3>::result_type, |
84 | float>::value), "" ); |
85 | static_assert((std::is_same<std::reference_wrapper<void()>::result_type, |
86 | void>::value), "" ); |
87 | static_assert((std::is_same<std::reference_wrapper<int*(double*)>::result_type, |
88 | int*>::value), "" ); |
89 | static_assert((std::is_same<std::reference_wrapper<void(*)()>::result_type, |
90 | void>::value), "" ); |
91 | static_assert((std::is_same<std::reference_wrapper<int*(*)(double*)>::result_type, |
92 | int*>::value), "" ); |
93 | static_assert((std::is_same<std::reference_wrapper<int*(C::*)(double*)>::result_type, |
94 | int*>::value), "" ); |
95 | static_assert((std::is_same<std::reference_wrapper<int (C::*)(double*) const volatile>::result_type, |
96 | int>::value), "" ); |
97 | static_assert((std::is_same<std::reference_wrapper<C()>::result_type, |
98 | C>::value), "" ); |
99 | static_assert(has_result_type<std::reference_wrapper<functor3> >::value, "" ); |
100 | static_assert(!has_result_type<std::reference_wrapper<functor4> >::value, "" ); |
101 | static_assert(!has_result_type<std::reference_wrapper<C> >::value, "" ); |
102 | |
103 | return 0; |
104 | } |
105 | |