2020-04-22 21:05:25 +01:00
|
|
|
/* This file is part of the mp project.
|
|
|
|
* Copyright (c) 2017 MerryMage
|
|
|
|
* SPDX-License-Identifier: 0BSD
|
|
|
|
*/
|
|
|
|
|
|
|
|
#pragma once
|
|
|
|
|
|
|
|
#include <cstddef>
|
|
|
|
#include <tuple>
|
|
|
|
|
|
|
|
#include <mp/typelist/list.h>
|
|
|
|
|
|
|
|
namespace mp {
|
|
|
|
|
|
|
|
template<class F>
|
2020-04-22 21:06:56 +01:00
|
|
|
struct function_info : function_info<decltype(&F::operator())> {};
|
2020-04-22 21:05:25 +01:00
|
|
|
|
|
|
|
template<class R, class... As>
|
|
|
|
struct function_info<R(As...)> {
|
|
|
|
using return_type = R;
|
|
|
|
using parameter_list = list<As...>;
|
|
|
|
static constexpr std::size_t parameter_count = sizeof...(As);
|
|
|
|
|
|
|
|
using equivalent_function_type = R(As...);
|
|
|
|
|
|
|
|
template<std::size_t I>
|
|
|
|
struct parameter {
|
|
|
|
static_assert(I < parameter_count, "Non-existent parameter");
|
|
|
|
using type = std::tuple_element_t<I, std::tuple<As...>>;
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
template<class R, class... As>
|
2020-04-22 21:06:56 +01:00
|
|
|
struct function_info<R(*)(As...)> : function_info<R(As...)> {};
|
2020-04-22 21:05:25 +01:00
|
|
|
|
|
|
|
template<class C, class R, class... As>
|
2020-04-22 21:06:56 +01:00
|
|
|
struct function_info<R(C::*)(As...)> : function_info<R(As...)> {
|
2020-04-22 21:05:25 +01:00
|
|
|
using class_type = C;
|
|
|
|
};
|
|
|
|
|
|
|
|
template<class C, class R, class... As>
|
2020-04-22 21:06:56 +01:00
|
|
|
struct function_info<R(C::*)(As...) const> : function_info<R(As...)> {
|
2020-04-22 21:05:25 +01:00
|
|
|
using class_type = C;
|
|
|
|
};
|
|
|
|
|
|
|
|
template<class F>
|
|
|
|
constexpr size_t parameter_count_v = function_info<F>::parameter_count;
|
|
|
|
|
|
|
|
template<class F>
|
|
|
|
using parameter_list = typename function_info<F>::parameter_list;
|
|
|
|
|
|
|
|
template<class F, std::size_t I>
|
|
|
|
using get_parameter = typename function_info<F>::template parameter<I>::type;
|
|
|
|
|
|
|
|
template<class F>
|
|
|
|
using equivalent_function_type = typename function_info<F>::equivalent_function_type;
|
|
|
|
|
|
|
|
template<class F>
|
|
|
|
using return_type = typename function_info<F>::return_type;
|
|
|
|
|
|
|
|
template<class F>
|
|
|
|
using class_type = typename function_info<F>::class_type;
|
|
|
|
|
|
|
|
} // namespace mp
|