/* * Copyright 2016-2021 Max Kellermann * * 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. */ #pragma once #include #include /** * 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 class BoundMethod; template class BoundMethod { typedef R (*function_pointer)(void *instance, Args... args) noexcept(NoExcept); void *instance_; function_pointer function; public: /** * Non-initializing trivial constructor */ BoundMethod() = default; constexpr 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. */ BoundMethod(std::nullptr_t) noexcept:function(nullptr) {} /** * Was this object initialized with a valid function pointer? */ operator bool() const noexcept { return function != nullptr; } R operator()(Args... args) const { return function(instance_, std::forward(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 struct MethodWithSignature; template struct MethodWithSignature { typedef R (T::*method_pointer)(Args...) noexcept(NoExcept); }; /** * Helper class which introspects a method pointer type. * * @param M the method pointer type */ template struct MethodSignatureHelper; template struct MethodSignatureHelper { /** * 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 struct MethodWrapperWithSignature; template struct MethodWrapperWithSignature { typedef R (*function_pointer)(void *instance, Args...) noexcept(NoExcept); }; /** * Generate a wrapper function. * * @param T the containing class * @param method the method pointer * @param S the plain function signature type */ template struct BindMethodWrapperGenerator; template struct BindMethodWrapperGenerator { static R Invoke(void *_instance, Args... args) noexcept(NoExcept) { auto &t = *(T *)_instance; return (t.*method)(std::forward(args)...); } }; template::method_pointer method> typename MethodWrapperWithSignature::function_pointer MakeBindMethodWrapper() noexcept { return BindMethodWrapperGenerator::Invoke; } /** * Helper class which introspects a function pointer type. * * @param S the function type */ template struct FunctionTraits; template struct FunctionTraits { /** * 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. * * @param S the plain function signature type * @param P the plain function pointer type * @param function the function pointer */ template struct BindFunctionWrapperGenerator; template struct BindFunctionWrapperGenerator { static R Invoke(void *, Args... args) noexcept(NoExcept) { return function(std::forward(args)...); } }; template typename MethodWrapperWithSignature::function_pointer MakeBindFunctionWrapper() noexcept { return BindFunctionWrapperGenerator::Invoke; } } /* namespace BindMethodDetail */ /** * Construct a #BoundMethod instance. * * @param method the method pointer * @param instance the instance of #T to be bound */ template constexpr auto BindMethod(typename BindMethodDetail::MethodSignatureHelper::class_type &instance) noexcept { using H = BindMethodDetail::MethodSignatureHelper; using class_type = typename H::class_type; using plain_signature = typename H::plain_signature; return BoundMethod{ &instance, BindMethodDetail::MakeBindMethodWrapper(), }; } /** * Shortcut macro which takes an instance and a method pointer and * constructs a #BoundMethod instance. */ #define BIND_METHOD(instance, method) \ BindMethod(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::method) /** * Construct a #BoundMethod instance for a plain function. * * @param T the #FunctionTraits class * @param function the function pointer */ template constexpr BoundMethod BindFunction() noexcept { return BoundMethod(nullptr, BindMethodDetail::MakeBindFunctionWrapper()); } /** * Shortcut macro which takes a function pointer and constructs a * #BoundMethod instance. */ #define BIND_FUNCTION(function) \ BindFunction, &function>()