1*c8dee2aaSAndroid Build Coastguard Worker /* 2*c8dee2aaSAndroid Build Coastguard Worker * Copyright 2018 Google Inc. 3*c8dee2aaSAndroid Build Coastguard Worker * 4*c8dee2aaSAndroid Build Coastguard Worker * Use of this source code is governed by a BSD-style license that can be 5*c8dee2aaSAndroid Build Coastguard Worker * found in the LICENSE file. 6*c8dee2aaSAndroid Build Coastguard Worker */ 7*c8dee2aaSAndroid Build Coastguard Worker 8*c8dee2aaSAndroid Build Coastguard Worker #ifndef SkCallableTraits_DEFINED 9*c8dee2aaSAndroid Build Coastguard Worker #define SkCallableTraits_DEFINED 10*c8dee2aaSAndroid Build Coastguard Worker 11*c8dee2aaSAndroid Build Coastguard Worker #include <type_traits> 12*c8dee2aaSAndroid Build Coastguard Worker #include <tuple> 13*c8dee2aaSAndroid Build Coastguard Worker 14*c8dee2aaSAndroid Build Coastguard Worker template <typename R, typename... Args> struct sk_base_callable_traits { 15*c8dee2aaSAndroid Build Coastguard Worker using return_type = R; 16*c8dee2aaSAndroid Build Coastguard Worker static constexpr std::size_t arity = sizeof...(Args); 17*c8dee2aaSAndroid Build Coastguard Worker template <std::size_t N> struct argument { 18*c8dee2aaSAndroid Build Coastguard Worker static_assert(N < arity, ""); 19*c8dee2aaSAndroid Build Coastguard Worker using type = typename std::tuple_element<N, std::tuple<Args...>>::type; 20*c8dee2aaSAndroid Build Coastguard Worker }; 21*c8dee2aaSAndroid Build Coastguard Worker }; 22*c8dee2aaSAndroid Build Coastguard Worker 23*c8dee2aaSAndroid Build Coastguard Worker #define SK_CALLABLE_TRAITS__COMMA , 24*c8dee2aaSAndroid Build Coastguard Worker 25*c8dee2aaSAndroid Build Coastguard Worker #define SK_CALLABLE_TRAITS__VARARGS(quals, _) \ 26*c8dee2aaSAndroid Build Coastguard Worker SK_CALLABLE_TRAITS__INSTANCE(quals,) \ 27*c8dee2aaSAndroid Build Coastguard Worker SK_CALLABLE_TRAITS__INSTANCE(quals, SK_CALLABLE_TRAITS__COMMA ...) 28*c8dee2aaSAndroid Build Coastguard Worker 29*c8dee2aaSAndroid Build Coastguard Worker #ifdef __cpp_noexcept_function_type 30*c8dee2aaSAndroid Build Coastguard Worker #define SK_CALLABLE_TRAITS__NE_VARARGS(quals, _) \ 31*c8dee2aaSAndroid Build Coastguard Worker SK_CALLABLE_TRAITS__VARARGS(quals,) \ 32*c8dee2aaSAndroid Build Coastguard Worker SK_CALLABLE_TRAITS__VARARGS(quals noexcept,) 33*c8dee2aaSAndroid Build Coastguard Worker #else 34*c8dee2aaSAndroid Build Coastguard Worker #define SK_CALLABLE_TRAITS__NE_VARARGS(quals, _) \ 35*c8dee2aaSAndroid Build Coastguard Worker SK_CALLABLE_TRAITS__VARARGS(quals,) 36*c8dee2aaSAndroid Build Coastguard Worker #endif 37*c8dee2aaSAndroid Build Coastguard Worker 38*c8dee2aaSAndroid Build Coastguard Worker #define SK_CALLABLE_TRAITS__REF_NE_VARARGS(quals, _) \ 39*c8dee2aaSAndroid Build Coastguard Worker SK_CALLABLE_TRAITS__NE_VARARGS(quals,) \ 40*c8dee2aaSAndroid Build Coastguard Worker SK_CALLABLE_TRAITS__NE_VARARGS(quals &,) \ 41*c8dee2aaSAndroid Build Coastguard Worker SK_CALLABLE_TRAITS__NE_VARARGS(quals &&,) 42*c8dee2aaSAndroid Build Coastguard Worker 43*c8dee2aaSAndroid Build Coastguard Worker #define SK_CALLABLE_TRAITS__CV_REF_NE_VARARGS() \ 44*c8dee2aaSAndroid Build Coastguard Worker SK_CALLABLE_TRAITS__REF_NE_VARARGS(,) \ 45*c8dee2aaSAndroid Build Coastguard Worker SK_CALLABLE_TRAITS__REF_NE_VARARGS(const,) \ 46*c8dee2aaSAndroid Build Coastguard Worker SK_CALLABLE_TRAITS__REF_NE_VARARGS(volatile,) \ 47*c8dee2aaSAndroid Build Coastguard Worker SK_CALLABLE_TRAITS__REF_NE_VARARGS(const volatile,) 48*c8dee2aaSAndroid Build Coastguard Worker 49*c8dee2aaSAndroid Build Coastguard Worker /** Infer the return_type and argument<N> of a callable type T. */ 50*c8dee2aaSAndroid Build Coastguard Worker template <typename T> struct SkCallableTraits : SkCallableTraits<decltype(&T::operator())> {}; 51*c8dee2aaSAndroid Build Coastguard Worker 52*c8dee2aaSAndroid Build Coastguard Worker // function (..., (const, volatile), (&, &&), noexcept) 53*c8dee2aaSAndroid Build Coastguard Worker #define SK_CALLABLE_TRAITS__INSTANCE(quals, varargs) \ 54*c8dee2aaSAndroid Build Coastguard Worker template <typename R, typename... Args> \ 55*c8dee2aaSAndroid Build Coastguard Worker struct SkCallableTraits<R(Args... varargs) quals> : sk_base_callable_traits<R, Args...> {}; 56*c8dee2aaSAndroid Build Coastguard Worker 57*c8dee2aaSAndroid Build Coastguard Worker SK_CALLABLE_TRAITS__CV_REF_NE_VARARGS() 58*c8dee2aaSAndroid Build Coastguard Worker #undef SK_CALLABLE_TRAITS__INSTANCE 59*c8dee2aaSAndroid Build Coastguard Worker 60*c8dee2aaSAndroid Build Coastguard Worker // pointer to function (..., noexcept) 61*c8dee2aaSAndroid Build Coastguard Worker #define SK_CALLABLE_TRAITS__INSTANCE(quals, varargs) \ 62*c8dee2aaSAndroid Build Coastguard Worker template <typename R, typename... Args> \ 63*c8dee2aaSAndroid Build Coastguard Worker struct SkCallableTraits<R(*)(Args... varargs) quals> : sk_base_callable_traits<R, Args...> {}; 64*c8dee2aaSAndroid Build Coastguard Worker 65*c8dee2aaSAndroid Build Coastguard Worker SK_CALLABLE_TRAITS__NE_VARARGS(,) 66*c8dee2aaSAndroid Build Coastguard Worker #undef SK_CALLABLE_TRAITS__INSTANCE 67*c8dee2aaSAndroid Build Coastguard Worker 68*c8dee2aaSAndroid Build Coastguard Worker // pointer to method (..., (const, volatile), (&, &&), noexcept) 69*c8dee2aaSAndroid Build Coastguard Worker #define SK_CALLABLE_TRAITS__INSTANCE(quals, varargs) \ 70*c8dee2aaSAndroid Build Coastguard Worker template <typename T, typename R, typename... Args> \ 71*c8dee2aaSAndroid Build Coastguard Worker struct SkCallableTraits<R(T::*)(Args... varargs) quals> : sk_base_callable_traits<R, Args...> {}; 72*c8dee2aaSAndroid Build Coastguard Worker 73*c8dee2aaSAndroid Build Coastguard Worker SK_CALLABLE_TRAITS__CV_REF_NE_VARARGS() 74*c8dee2aaSAndroid Build Coastguard Worker #undef SK_CALLABLE_TRAITS__INSTANCE 75*c8dee2aaSAndroid Build Coastguard Worker 76*c8dee2aaSAndroid Build Coastguard Worker // pointer to field 77*c8dee2aaSAndroid Build Coastguard Worker template <typename T, typename R> 78*c8dee2aaSAndroid Build Coastguard Worker struct SkCallableTraits<R T::*> : sk_base_callable_traits<typename std::add_lvalue_reference<R>::type> {}; 79*c8dee2aaSAndroid Build Coastguard Worker 80*c8dee2aaSAndroid Build Coastguard Worker #undef SK_CALLABLE_TRAITS__CV_REF_NE_VARARGS 81*c8dee2aaSAndroid Build Coastguard Worker #undef SK_CALLABLE_TRAITS__REF_NE_VARARGS 82*c8dee2aaSAndroid Build Coastguard Worker #undef SK_CALLABLE_TRAITS__NE_VARARGS 83*c8dee2aaSAndroid Build Coastguard Worker #undef SK_CALLABLE_TRAITS__VARARGS 84*c8dee2aaSAndroid Build Coastguard Worker #undef SK_CALLABLE_TRAITS__COMMA 85*c8dee2aaSAndroid Build Coastguard Worker 86*c8dee2aaSAndroid Build Coastguard Worker #endif 87