mpd/src/util/BindMethod.hxx

298 lines
8.6 KiB
C++
Raw Normal View History

/*
2021-12-02 22:12:36 +01:00
* Copyright 2016-2021 Max Kellermann <max.kellermann@gmail.com>
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
*
* - Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
*
* - Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the
* distribution.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
* ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
* FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
* FOUNDATION OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
* SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
* STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED
* OF THE POSSIBILITY OF SUCH DAMAGE.
*/
2021-12-02 22:12:36 +01:00
#pragma once
#include <type_traits>
#include <utility>
/**
* This object stores a function pointer wrapping a method, and a
* reference to an instance of the method's class. It can be used to
* wrap instance methods as callback functions.
*
* @param S the plain function signature type
*/
template<typename S=void()>
class BoundMethod;
template<typename R,
bool NoExcept,
typename... Args>
class BoundMethod<R(Args...) noexcept(NoExcept)> {
typedef R (*function_pointer)(void *instance, Args... args) noexcept(NoExcept);
void *instance_;
function_pointer function;
public:
/**
* Non-initializing trivial constructor
*/
BoundMethod() = default;
constexpr
2018-08-20 13:45:41 +02:00
BoundMethod(void *_instance, function_pointer _function) noexcept
:instance_(_instance), function(_function) {}
/**
* Construct an "undefined" object. It must not be called,
* and its "bool" operator returns false.
*/
2018-08-20 13:45:41 +02:00
BoundMethod(std::nullptr_t) noexcept:function(nullptr) {}
/**
* Was this object initialized with a valid function pointer?
*/
2018-08-20 13:45:41 +02:00
operator bool() const noexcept {
return function != nullptr;
}
R operator()(Args... args) const {
return function(instance_, std::forward<Args>(args)...);
}
};
namespace BindMethodDetail {
/**
* Helper class which converts a signature type to a method pointer
* type.
*
* @param T the wrapped class
* @param S the function signature type (plain, without instance
* pointer)
*/
template<typename T, typename S>
struct MethodWithSignature;
template<typename T,
bool NoExcept,
typename R, typename... Args>
struct MethodWithSignature<T, R(Args...) noexcept(NoExcept)> {
typedef R (T::*method_pointer)(Args...) noexcept(NoExcept);
};
/**
* Helper class which introspects a method pointer type.
*
* @param M the method pointer type
*/
template<typename M>
struct MethodSignatureHelper;
template<typename R, bool NoExcept, typename T, typename... Args>
struct MethodSignatureHelper<R (T::*)(Args...) noexcept(NoExcept)> {
/**
* The class which contains the given method (signature).
*/
typedef T class_type;
/**
* A function type which describes the "plain" function
* signature.
*/
typedef R plain_signature(Args...) noexcept(NoExcept);
};
/**
* Helper class which converts a plain function signature type to a
* wrapper function pointer type.
*/
template<typename S>
struct MethodWrapperWithSignature;
template<typename R, bool NoExcept, typename... Args>
struct MethodWrapperWithSignature<R(Args...) noexcept(NoExcept)> {
typedef R (*function_pointer)(void *instance,
Args...) noexcept(NoExcept);
};
/**
* Generate a wrapper function. Helper class for
* #BindMethodWrapperGenerator.
*
* @param T the containing class
* @param M the method pointer type
* @param method the method pointer
* @param R the return type
* @param Args the method arguments
*/
template<typename T, bool NoExcept, typename M, M method, typename R, typename... Args>
struct BindMethodWrapperGenerator2 {
static R Invoke(void *_instance, Args... args) noexcept(NoExcept) {
auto &t = *(T *)_instance;
return (t.*method)(std::forward<Args>(args)...);
}
};
/**
* Generate a wrapper function.
*
* @param T the containing class
* @param M the method pointer type
* @param method the method pointer
* @param S the plain function signature type
*/
template<typename T, typename M, M method, typename S>
struct BindMethodWrapperGenerator;
template<typename T, bool NoExcept,
typename M, M method, typename R, typename... Args>
struct BindMethodWrapperGenerator<T, M, method, R(Args...) noexcept(NoExcept)>
: BindMethodWrapperGenerator2<T, NoExcept, M, method, R, Args...> {
};
template<typename T, typename S,
typename MethodWithSignature<T, S>::method_pointer method>
typename MethodWrapperWithSignature<S>::function_pointer
2018-08-20 13:45:41 +02:00
MakeBindMethodWrapper() noexcept
{
return BindMethodWrapperGenerator<T, typename MethodWithSignature<T, S>::method_pointer, method, S>::Invoke;
}
2017-04-21 19:39:02 +02:00
/**
* 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)> {
2017-04-21 19:39:02 +02:00
/**
* A function type which describes the "plain" function
* signature.
*/
typedef R function_type(Args...) noexcept(NoExcept);
2017-04-21 19:39:02 +02:00
/**
* A function pointer type which describes the "plain"
* function signature.
*/
typedef R (*pointer)(Args...) noexcept(NoExcept);
2017-04-21 19:39:02 +02:00
};
/**
* Generate a wrapper function for a plain function which ignores the
* instance pointer. Helper class for
* #BindFunctionWrapperGenerator.
*
* @param F the function pointer type
* @param function the function pointer
* @param R the return type
* @param Args the function arguments
*/
template<bool NoExcept, typename F, F function, typename R, typename... Args>
2017-04-21 19:39:02 +02:00
struct BindFunctionWrapperGenerator2 {
static R Invoke(void *, Args... args) noexcept(NoExcept) {
2017-04-21 19:39:02 +02:00
return function(std::forward<Args>(args)...);
}
};
/**
* Generate a wrapper function.
*
* @param S the plain function signature type
* @param P the plain function pointer type
* @param function the function pointer
*/
template<typename S, typename P, P function>
struct BindFunctionWrapperGenerator;
template<typename P, P function, bool NoExcept, typename R, typename... Args>
struct BindFunctionWrapperGenerator<R(Args...) noexcept(NoExcept), P, function>
: BindFunctionWrapperGenerator2<NoExcept, P, function, R, Args...> {
2017-04-21 19:39:02 +02:00
};
template<typename T, typename T::pointer function>
2017-04-21 19:39:02 +02:00
typename MethodWrapperWithSignature<typename T::function_type>::function_pointer
2018-08-20 13:45:41 +02:00
MakeBindFunctionWrapper() noexcept
2017-04-21 19:39:02 +02:00
{
return BindFunctionWrapperGenerator<typename T::function_type,
typename T::pointer,
2017-04-21 19:39:02 +02:00
function>::Invoke;
}
} /* namespace BindMethodDetail */
/**
* Construct a #BoundMethod instance.
*
* @param method the method pointer
* @param instance the instance of #T to be bound
*/
2021-12-02 22:12:36 +01:00
template<auto method>
constexpr auto
BindMethod(typename BindMethodDetail::MethodSignatureHelper<decltype(method)>::class_type &instance) noexcept
{
2021-12-02 22:12:36 +01:00
using H = BindMethodDetail::MethodSignatureHelper<decltype(method)>;
using class_type = typename H::class_type;
using plain_signature = typename H::plain_signature;
return BoundMethod<plain_signature>{
&instance,
BindMethodDetail::MakeBindMethodWrapper<class_type, plain_signature, method>(),
};
}
/**
* Shortcut macro which takes an instance and a method pointer and
* constructs a #BoundMethod instance.
*/
#define BIND_METHOD(instance, method) \
2021-12-02 22:12:36 +01:00
BindMethod<method>(instance)
/**
* Shortcut wrapper for BIND_METHOD() which assumes "*this" is the
* instance to be bound.
*/
#define BIND_THIS_METHOD(method) BIND_METHOD(*this, &std::remove_reference_t<decltype(*this)>::method)
2017-04-21 19:39:02 +02:00
/**
* Construct a #BoundMethod instance for a plain function.
*
* @param T the #FunctionTraits class
* @param function the function pointer
*/
template<typename T, typename T::pointer function>
2017-04-21 19:39:02 +02:00
constexpr BoundMethod<typename T::function_type>
2018-08-20 13:45:41 +02:00
BindFunction() noexcept
2017-04-21 19:39:02 +02:00
{
return BoundMethod<typename T::function_type>(nullptr,
BindMethodDetail::MakeBindFunctionWrapper<T, function>());
}
/**
* Shortcut macro which takes a function pointer and constructs a
* #BoundMethod instance.
*/
#define BIND_FUNCTION(function) \
BindFunction<typename BindMethodDetail::FunctionTraits<decltype(function)>, &function>()