|  | // -*- C++ -*- | 
|  | //===----------------------------------------------------------------------===// | 
|  | // | 
|  | //                     The LLVM Compiler Infrastructure | 
|  | // | 
|  | // This file is dual licensed under the MIT and the University of Illinois Open | 
|  | // Source Licenses. See LICENSE.TXT for details. | 
|  | // | 
|  | //===----------------------------------------------------------------------===// | 
|  |  | 
|  | #ifndef _LIBCPP_FUNCTIONAL_BASE_03 | 
|  | #define _LIBCPP_FUNCTIONAL_BASE_03 | 
|  |  | 
|  | // manual variadic expansion for <functional> | 
|  |  | 
|  | // __invoke | 
|  |  | 
|  | template <class _Ret, class _T1, bool _IsFunc, bool _IsBase> | 
|  | struct __enable_invoke_imp; | 
|  |  | 
|  | template <class _Ret, class _T1> | 
|  | struct __enable_invoke_imp<_Ret, _T1, true, true> { | 
|  | typedef _Ret _Bullet1; | 
|  | typedef _Bullet1 type; | 
|  | }; | 
|  |  | 
|  | template <class _Ret, class _T1> | 
|  | struct __enable_invoke_imp<_Ret, _T1, true, false>  { | 
|  | typedef _Ret _Bullet2; | 
|  | typedef _Bullet2 type; | 
|  | }; | 
|  |  | 
|  | template <class _Ret, class _T1> | 
|  | struct __enable_invoke_imp<_Ret, _T1, false, true>  { | 
|  | typedef typename add_lvalue_reference< | 
|  | typename __apply_cv<_T1, _Ret>::type | 
|  | >::type _Bullet3; | 
|  | typedef _Bullet3 type; | 
|  | }; | 
|  |  | 
|  | template <class _Ret, class _T1> | 
|  | struct __enable_invoke_imp<_Ret, _T1, false, false>  { | 
|  | typedef typename add_lvalue_reference< | 
|  | typename __apply_cv<decltype(*_VSTD::declval<_T1>()), _Ret>::type | 
|  | >::type _Bullet4; | 
|  | typedef _Bullet4 type; | 
|  | }; | 
|  |  | 
|  | template <class _Ret, class _T1> | 
|  | struct __enable_invoke_imp<_Ret, _T1*, false, false>  { | 
|  | typedef typename add_lvalue_reference< | 
|  | typename __apply_cv<_T1, _Ret>::type | 
|  | >::type _Bullet4; | 
|  | typedef _Bullet4  type; | 
|  | }; | 
|  |  | 
|  | template <class _Fn, class _T1, | 
|  | class _Traits = __member_pointer_traits<_Fn>, | 
|  | class _Ret = typename _Traits::_ReturnType, | 
|  | class _Class = typename _Traits::_ClassType> | 
|  | struct __enable_invoke : __enable_invoke_imp< | 
|  | _Ret, _T1, | 
|  | is_member_function_pointer<_Fn>::value, | 
|  | is_base_of<_Class, typename remove_reference<_T1>::type>::value> | 
|  | { | 
|  | }; | 
|  |  | 
|  | __nat __invoke(__any, ...); | 
|  |  | 
|  | // first bullet | 
|  |  | 
|  | template <class _Fn, class _T1> | 
|  | inline _LIBCPP_INLINE_VISIBILITY | 
|  | typename __enable_invoke<_Fn, _T1>::_Bullet1 | 
|  | __invoke(_Fn __f, _T1& __t1) { | 
|  | return (__t1.*__f)(); | 
|  | } | 
|  |  | 
|  | template <class _Fn, class _T1, class _A0> | 
|  | inline _LIBCPP_INLINE_VISIBILITY | 
|  | typename __enable_invoke<_Fn, _T1>::_Bullet1 | 
|  | __invoke(_Fn __f, _T1& __t1, _A0& __a0) { | 
|  | return (__t1.*__f)(__a0); | 
|  | } | 
|  |  | 
|  | template <class _Fn, class _T1, class _A0, class _A1> | 
|  | inline _LIBCPP_INLINE_VISIBILITY | 
|  | typename __enable_invoke<_Fn, _T1>::_Bullet1 | 
|  | __invoke(_Fn __f, _T1& __t1, _A0& __a0, _A1& __a1) { | 
|  | return (__t1.*__f)(__a0, __a1); | 
|  | } | 
|  |  | 
|  | template <class _Fn, class _T1, class _A0, class _A1, class _A2> | 
|  | inline _LIBCPP_INLINE_VISIBILITY | 
|  | typename __enable_invoke<_Fn, _T1>::_Bullet1 | 
|  | __invoke(_Fn __f, _T1& __t1, _A0& __a0, _A1& __a1, _A2& __a2) { | 
|  | return (__t1.*__f)(__a0, __a1, __a2); | 
|  | } | 
|  |  | 
|  | template <class _Fn, class _T1> | 
|  | inline _LIBCPP_INLINE_VISIBILITY | 
|  | typename __enable_invoke<_Fn, _T1>::_Bullet2 | 
|  | __invoke(_Fn __f, _T1& __t1) { | 
|  | return ((*__t1).*__f)(); | 
|  | } | 
|  |  | 
|  | template <class _Fn, class _T1, class _A0> | 
|  | inline _LIBCPP_INLINE_VISIBILITY | 
|  | typename __enable_invoke<_Fn, _T1>::_Bullet2 | 
|  | __invoke(_Fn __f, _T1& __t1, _A0& __a0) { | 
|  | return ((*__t1).*__f)(__a0); | 
|  | } | 
|  |  | 
|  | template <class _Fn, class _T1, class _A0, class _A1> | 
|  | inline _LIBCPP_INLINE_VISIBILITY | 
|  | typename __enable_invoke<_Fn, _T1>::_Bullet2 | 
|  | __invoke(_Fn __f, _T1& __t1, _A0& __a0, _A1& __a1) { | 
|  | return ((*__t1).*__f)(__a0, __a1); | 
|  | } | 
|  |  | 
|  | template <class _Fn, class _T1, class _A0, class _A1, class _A2> | 
|  | inline _LIBCPP_INLINE_VISIBILITY | 
|  | typename __enable_invoke<_Fn, _T1>::_Bullet2 | 
|  | __invoke(_Fn __f, _T1& __t1, _A0& __a0, _A1& __a1, _A2& __a2) { | 
|  | return ((*__t1).*__f)(__a0, __a1, __a2); | 
|  | } | 
|  |  | 
|  | template <class _Fn, class _T1> | 
|  | inline _LIBCPP_INLINE_VISIBILITY | 
|  | typename __enable_invoke<_Fn, _T1>::_Bullet3 | 
|  | __invoke(_Fn __f, _T1& __t1) { | 
|  | return __t1.*__f; | 
|  | } | 
|  |  | 
|  | template <class _Fn, class _T1> | 
|  | inline _LIBCPP_INLINE_VISIBILITY | 
|  | typename __enable_invoke<_Fn, _T1>::_Bullet4 | 
|  | __invoke(_Fn __f, _T1& __t1) { | 
|  | return (*__t1).*__f; | 
|  | } | 
|  |  | 
|  | // fifth bullet | 
|  |  | 
|  | template <class _Fp> | 
|  | inline _LIBCPP_INLINE_VISIBILITY | 
|  | decltype(_VSTD::declval<_Fp&>()()) | 
|  | __invoke(_Fp& __f) | 
|  | { | 
|  | return __f(); | 
|  | } | 
|  |  | 
|  | template <class _Fp, class _A0> | 
|  | inline _LIBCPP_INLINE_VISIBILITY | 
|  | decltype(_VSTD::declval<_Fp&>()(_VSTD::declval<_A0&>())) | 
|  | __invoke(_Fp& __f, _A0& __a0) | 
|  | { | 
|  | return __f(__a0); | 
|  | } | 
|  |  | 
|  | template <class _Fp, class _A0, class _A1> | 
|  | inline _LIBCPP_INLINE_VISIBILITY | 
|  | decltype(_VSTD::declval<_Fp&>()(_VSTD::declval<_A0&>(), _VSTD::declval<_A1&>())) | 
|  | __invoke(_Fp& __f, _A0& __a0, _A1& __a1) | 
|  | { | 
|  | return __f(__a0, __a1); | 
|  | } | 
|  |  | 
|  | template <class _Fp, class _A0, class _A1, class _A2> | 
|  | inline _LIBCPP_INLINE_VISIBILITY | 
|  | decltype(_VSTD::declval<_Fp&>()(_VSTD::declval<_A0&>(), _VSTD::declval<_A1&>(), _VSTD::declval<_A2&>())) | 
|  | __invoke(_Fp& __f, _A0& __a0, _A1& __a1, _A2& __a2) | 
|  | { | 
|  | return __f(__a0, __a1, __a2); | 
|  | } | 
|  |  | 
|  | template <class _Fp, bool = __has_result_type<__weak_result_type<_Fp> >::value> | 
|  | struct __invoke_return | 
|  | { | 
|  | typedef typename __weak_result_type<_Fp>::result_type type; | 
|  | }; | 
|  |  | 
|  | template <class _Fp> | 
|  | struct __invoke_return<_Fp, false> | 
|  | { | 
|  | typedef decltype(__invoke(_VSTD::declval<_Fp&>())) type; | 
|  | }; | 
|  |  | 
|  | template <class _Tp, class _A0> | 
|  | struct __invoke_return0 | 
|  | { | 
|  | typedef decltype(__invoke(_VSTD::declval<_Tp&>(), _VSTD::declval<_A0&>())) type; | 
|  | }; | 
|  |  | 
|  | template <class _Rp, class _Tp, class _A0> | 
|  | struct __invoke_return0<_Rp _Tp::*, _A0> | 
|  | { | 
|  | typedef typename __enable_invoke<_Rp _Tp::*, _A0>::type type; | 
|  | }; | 
|  |  | 
|  | template <class _Tp, class _A0, class _A1> | 
|  | struct __invoke_return1 | 
|  | { | 
|  | typedef decltype(__invoke(_VSTD::declval<_Tp&>(), _VSTD::declval<_A0&>(), | 
|  | _VSTD::declval<_A1&>())) type; | 
|  | }; | 
|  |  | 
|  | template <class _Rp, class _Class, class _A0, class _A1> | 
|  | struct __invoke_return1<_Rp _Class::*, _A0, _A1> { | 
|  | typedef typename __enable_invoke<_Rp _Class::*, _A0>::type type; | 
|  | }; | 
|  |  | 
|  | template <class _Tp, class _A0, class _A1, class _A2> | 
|  | struct __invoke_return2 | 
|  | { | 
|  | typedef decltype(__invoke(_VSTD::declval<_Tp&>(), _VSTD::declval<_A0&>(), | 
|  | _VSTD::declval<_A1&>(), | 
|  | _VSTD::declval<_A2&>())) type; | 
|  | }; | 
|  |  | 
|  | template <class _Ret, class _Class, class _A0, class _A1, class _A2> | 
|  | struct __invoke_return2<_Ret _Class::*, _A0, _A1, _A2> { | 
|  | typedef typename __enable_invoke<_Ret _Class::*, _A0>::type type; | 
|  | }; | 
|  | #endif  // _LIBCPP_FUNCTIONAL_BASE_03 |