continuable/include/functional_unwrap.hpp
2015-08-09 19:32:57 +02:00

253 lines
8.8 KiB
C++

/**
* Copyright 2015 Denis Blank <denis.blank@outlook.com>
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/*
* The functional unwrap trait provides the possibility to extract
* various types out of c++ functional types, like std::function, functors and more.
*/
#ifndef _FUNCTIONAL_UNWRAP_HPP_
#define _FUNCTIONAL_UNWRAP_HPP_
#include <functional>
#include <type_traits>
#include <utility>
#include <cstddef>
namespace fu
{
/// Identity class which is used to carry parameter packs of types.
template<typename... Args>
struct identity { };
/// Sequence class which is used to carry parameter packs of unsigned integers.
template<std::size_t...>
struct sequence { };
/// The Sequence generator generates a sequence of ascending numbers with the given size.
template<std::size_t...>
struct sequence_generator;
template<std::size_t... Stack>
struct sequence_generator<0, Stack...>
{
typedef sequence<Stack...> type;
};
template<std::size_t Position, std::size_t... Stack>
struct sequence_generator<Position, Stack...>
: sequence_generator<Position - 1, Position - 1, Stack...> { };
/// Sequence generator alias
template<std::size_t Size>
using sequence_of_t = typename sequence_generator<Size>::type;
namespace detail
{
template<typename Function>
struct unwrap_function_impl;
template<typename _RTy, typename... _ATy>
struct unwrap_function_impl<_RTy(_ATy...)>
{
/// The return type of the function.
typedef _RTy return_type;
/// The argument types of the function as pack in fu::identity.
typedef identity<_ATy...> argument_type;
/// The function provided as std::function
typedef std::function<_RTy(_ATy...)> function_type;
/// The function provided as function_ptr
typedef _RTy(*function_ptr)(_ATy...);
};
/// Pack in fu::identity
template<typename _RTy, typename... _ATy>
struct unwrap_function_impl<identity<_RTy, _ATy...>>
: unwrap_function_impl<_RTy(_ATy...)> { };
/// STL: std::function
template<typename _RTy, typename... _ATy>
struct unwrap_function_impl<std::function<_RTy(_ATy...)>>
: unwrap_function_impl<_RTy(_ATy...)> { };
/// STL: std::tuple
template<typename _RTy, typename... _ATy>
struct unwrap_function_impl<std::tuple<_RTy, _ATy...>>
: unwrap_function_impl<_RTy(_ATy...)> { };
/// Const function pointers
template<typename _RTy, typename... _ATy>
struct unwrap_function_impl<_RTy(*const)(_ATy...)>
: unwrap_function_impl<_RTy(_ATy...)> { };
/// Mutable function pointers
template<typename _RTy, typename... _ATy>
struct unwrap_function_impl<_RTy(*)(_ATy...)>
: unwrap_function_impl<_RTy(_ATy...)> { };
/// Const class method pointers
template<typename _CTy, typename _RTy, typename... _ATy>
struct unwrap_function_impl<_RTy(_CTy::*)(_ATy...) const>
: unwrap_function_impl<_RTy(_ATy...)> { };
/// Mutable class method pointers
template<typename _CTy, typename _RTy, typename... _ATy>
struct unwrap_function_impl<_RTy(_CTy::*)(_ATy...)>
: unwrap_function_impl<_RTy(_ATy...)> { };
/// Unwrap through pointer of functor.
template<typename Function>
static auto select_best_unwrap_unary_arg(int)
-> unwrap_function_impl<decltype(&Function::operator())>;
/// Unwrap through plain type.
template<typename Function>
static auto select_best_unwrap_unary_arg(long)
-> unwrap_function_impl<Function>;
template<typename... _FTy>
struct select_best_unwrap;
/// Enable only if 1 template argument is given.
template<typename _FTy>
struct select_best_unwrap<_FTy>
{
typedef decltype(select_best_unwrap_unary_arg<_FTy>(0)) type;
};
// Enable if more then 1 template argument is given.
// (Handles lazy typing)
template<typename _RTy, typename... _ATy>
struct select_best_unwrap<_RTy, _ATy...>
{
typedef unwrap_function_impl<_RTy(_ATy...)> type;
};
template<typename>
struct to_true
: std::true_type { };
/// std::true_type if unwrappable
template<typename... Function>
static auto test_unwrappable(int)
-> to_true<typename select_best_unwrap<Function...>::type::function_type>;
/// std::false_type if not unwrappable
template<typename... Function>
static auto test_unwrappable(long)
-> std::false_type;
} // detail
/// Trait to unwrap function parameters of various types:
/// Function style definition: Result(Parameters...)
/// STL `std::function` : std::function<Result(Parameters...)>
/// STL `std::tuple` : std::tuple<Result, Parameters...>
/// C++ Function pointers: `Result(*)(Parameters...)`
/// C++ Class method pointers: `Result(Class::*)(Parameters...)`
/// Lazy typed signatures: `Result, Parameters...`
/// Also provides optimized unwrap of functors and functional objects.
template<typename... Function>
using unwrap_function =
typename detail::select_best_unwrap<Function...>::type;
/// Trait which defines the return type of the function.
template<typename... Function>
using return_type_of_t =
typename detail::select_best_unwrap<Function...>::type::return_type;
/// Trait which defines the argument types of the function packed in std::tuple.
template<typename... Function>
using argument_type_of_t =
typename detail::select_best_unwrap<Function...>::type::argument_type;
/// Trait which defines the std::function type of the function.
template<typename... Function>
using function_type_of_t =
typename detail::select_best_unwrap<Function...>::type::function_type;
/// Trait which defines the function pointer type of the function.
template<typename... Function>
using function_ptr_of_t =
typename detail::select_best_unwrap<Function...>::type::function_ptr;
/// Trait which defines if the given function is unwrappable or not.
template<typename... Function>
struct is_unwrappable
: decltype(detail::test_unwrappable<Function...>(0)) { };
/// Converts any functional type in std::function.
template<typename Functional>
auto functionfy(Functional&& functional)
-> function_type_of_t<typename std::decay<Functional>::type>
{
return function_type_of_t<typename std::decay<Functional>::type>(std::forward<Functional>(functional));
}
template<typename T>
struct requires_functional_constructible
{
static_assert(is_unwrappable<typename std::decay<T>::type>::value,
"Given type is not functional unwrappable, did you try to use a std::bind expression or a non functional type?");
typedef T type;
};
namespace detail
{
/// Implementation of invoke_from_tuple
template<typename Sequence>
struct invoker;
template<std::size_t... Sequence>
struct invoker<sequence<Sequence...>>
{
template<typename _TTy>
inline static void silencer(_TTy&&) { }
template<typename _FTy, typename _TTy>
inline static auto invoke(_FTy&& functional, _TTy&& tuple)
-> return_type_of_t<typename std::decay<_FTy>::type>
{
// Silences warnings about unreferenced formal parameter when pack is empty.
silencer(tuple);
return std::forward<_FTy>(functional)(std::get<Sequence>(std::forward<_TTy>(tuple))...);
}
};
}
/// Invokes a function type with the given tuple arguments.
template<typename _FTy, typename _TTy>
inline auto invoke_from_tuple(_FTy&& functional, _TTy&& tuple)
-> return_type_of_t<typename std::decay<_FTy>::type>
{
return detail::invoker<
sequence_of_t<
std::tuple_size<
typename std::decay<_TTy>::type
>::value
>
>::invoke(std::forward<_FTy>(functional), std::forward<_TTy>(tuple));
}
} // fu
#endif // _FUNCTIONAL_UNWRAP_HPP_