util/BindMethod: simplify MakeBindFunctionWrapper()
This commit is contained in:
parent
7e4ba3cb72
commit
acab731fef
@ -125,14 +125,16 @@ struct MethodSignatureHelper<R (T::*)(Args...) noexcept(NoExcept)> {
|
|||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Helper class which converts a plain function signature type to a
|
* Helper class which converts a function pointer to a wrapper
|
||||||
* wrapper function pointer type.
|
* function pointer type.
|
||||||
*/
|
*/
|
||||||
template<typename S>
|
template<typename S>
|
||||||
struct MethodWrapperWithSignature;
|
struct FunctionSignatureHelper;
|
||||||
|
|
||||||
template<typename R, bool NoExcept, typename... Args>
|
template<typename R, bool NoExcept, typename... Args>
|
||||||
struct MethodWrapperWithSignature<R(Args...) noexcept(NoExcept)> {
|
struct FunctionSignatureHelper<R (*)(Args...) noexcept(NoExcept)> {
|
||||||
|
typedef R plain_signature(Args...) noexcept(NoExcept);
|
||||||
|
|
||||||
typedef R (*function_pointer)(void *instance,
|
typedef R (*function_pointer)(void *instance,
|
||||||
Args...) noexcept(NoExcept);
|
Args...) noexcept(NoExcept);
|
||||||
};
|
};
|
||||||
@ -161,52 +163,27 @@ MakeBindMethodWrapper() noexcept
|
|||||||
return BindMethodWrapperGenerator<decltype(method), method>::Invoke;
|
return BindMethodWrapperGenerator<decltype(method), method>::Invoke;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Helper class which introspects a function pointer type.
|
|
||||||
*
|
|
||||||
* @param S the function type
|
|
||||||
*/
|
|
||||||
template<typename S>
|
|
||||||
struct FunctionTraits;
|
|
||||||
|
|
||||||
template<typename R, bool NoExcept, typename... Args>
|
|
||||||
struct FunctionTraits<R(Args...) noexcept(NoExcept)> {
|
|
||||||
/**
|
|
||||||
* A function type which describes the "plain" function
|
|
||||||
* signature.
|
|
||||||
*/
|
|
||||||
typedef R function_type(Args...) noexcept(NoExcept);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* A function pointer type which describes the "plain"
|
|
||||||
* function signature.
|
|
||||||
*/
|
|
||||||
typedef R (*pointer)(Args...) noexcept(NoExcept);
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Generate a wrapper function.
|
* Generate a wrapper function.
|
||||||
*
|
*
|
||||||
* @param S the plain function signature type
|
* @param F the function pointer type
|
||||||
* @param P the plain function pointer type
|
|
||||||
* @param function the function pointer
|
* @param function the function pointer
|
||||||
*/
|
*/
|
||||||
template<typename S, auto function>
|
template<typename F, auto function>
|
||||||
struct BindFunctionWrapperGenerator;
|
struct BindFunctionWrapperGenerator;
|
||||||
|
|
||||||
template<auto function, bool NoExcept, typename R, typename... Args>
|
template<auto function, bool NoExcept, typename R, typename... Args>
|
||||||
struct BindFunctionWrapperGenerator<R(Args...) noexcept(NoExcept), function> {
|
struct BindFunctionWrapperGenerator<R (*)(Args...) noexcept(NoExcept), function> {
|
||||||
static R Invoke(void *, Args... args) noexcept(NoExcept) {
|
static R Invoke(void *, Args... args) noexcept(NoExcept) {
|
||||||
return function(std::forward<Args>(args)...);
|
return function(std::forward<Args>(args)...);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
template<typename T, typename T::pointer function>
|
template<auto function>
|
||||||
typename MethodWrapperWithSignature<typename T::function_type>::function_pointer
|
typename FunctionSignatureHelper<decltype(function)>::function_pointer
|
||||||
MakeBindFunctionWrapper() noexcept
|
MakeBindFunctionWrapper() noexcept
|
||||||
{
|
{
|
||||||
return BindFunctionWrapperGenerator<typename T::function_type,
|
return BindFunctionWrapperGenerator<decltype(function), function>::Invoke;
|
||||||
function>::Invoke;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
} /* namespace BindMethodDetail */
|
} /* namespace BindMethodDetail */
|
||||||
@ -245,15 +222,18 @@ BindMethod(typename BindMethodDetail::MethodSignatureHelper<decltype(method)>::c
|
|||||||
/**
|
/**
|
||||||
* Construct a #BoundMethod instance for a plain function.
|
* Construct a #BoundMethod instance for a plain function.
|
||||||
*
|
*
|
||||||
* @param T the #FunctionTraits class
|
|
||||||
* @param function the function pointer
|
* @param function the function pointer
|
||||||
*/
|
*/
|
||||||
template<typename T, typename T::pointer function>
|
template<auto function>
|
||||||
constexpr BoundMethod<typename T::function_type>
|
constexpr auto
|
||||||
BindFunction() noexcept
|
BindFunction() noexcept
|
||||||
{
|
{
|
||||||
return BoundMethod<typename T::function_type>(nullptr,
|
using H = BindMethodDetail::FunctionSignatureHelper<decltype(function)>;
|
||||||
BindMethodDetail::MakeBindFunctionWrapper<T, function>());
|
using plain_signature = typename H::plain_signature;
|
||||||
|
return BoundMethod<plain_signature>{
|
||||||
|
nullptr,
|
||||||
|
BindMethodDetail::MakeBindFunctionWrapper<function>(),
|
||||||
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -261,4 +241,4 @@ BindFunction() noexcept
|
|||||||
* #BoundMethod instance.
|
* #BoundMethod instance.
|
||||||
*/
|
*/
|
||||||
#define BIND_FUNCTION(function) \
|
#define BIND_FUNCTION(function) \
|
||||||
BindFunction<typename BindMethodDetail::FunctionTraits<decltype(function)>, &function>()
|
BindFunction<&function>()
|
||||||
|
Loading…
Reference in New Issue
Block a user