///\file /****************************************************************************** The MIT License(MIT) Embedded Template Library. https://github.com/ETLCPP/etl https://www.etlcpp.com Copyright(c) 2019 John Wellbelove Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files(the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and / or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions : The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software. THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. ******************************************************************************/ /****************************************************************************** Copyright (C) 2017 by Sergey A Kryukov: derived work http://www.SAKryukov.org http://www.codeproject.com/Members/SAKryukov Based on original work by Sergey Ryazanov: "The Impossibly Fast C++ Delegates", 18 Jul 2005 https://www.codeproject.com/articles/11015/the-impossibly-fast-c-delegates MIT license: http://en.wikipedia.org/wiki/MIT_License Original publication: https://www.codeproject.com/Articles/1170503/The-Impossibly-Fast-Cplusplus-Delegates-Fixed ******************************************************************************/ #ifndef ETL_DELEGATE_CPP11_INCLUDED #define ETL_DELEGATE_CPP11_INCLUDED #include "../platform.h" #include "../error_handler.h" #include "../exception.h" #include "../type_traits.h" #include "../utility.h" #include "../optional.h" namespace etl { //*************************************************************************** /// The base class for delegate exceptions. //*************************************************************************** class delegate_exception : public exception { public: delegate_exception(string_type reason_, string_type file_name_, numeric_type line_number_) : exception(reason_, file_name_, line_number_) { } }; //*************************************************************************** /// The exception thrown when the delegate is uninitialised. //*************************************************************************** class delegate_uninitialised : public delegate_exception { public: delegate_uninitialised(string_type file_name_, numeric_type line_number_) : delegate_exception(ETL_ERROR_TEXT("delegate:uninitialised", ETL_DELEGATE_FILE_ID"A"), file_name_, line_number_) { } }; //************************************************************************* /// Declaration. //************************************************************************* template class delegate; //************************************************************************* /// Specialisation. //************************************************************************* template class delegate final { public: //************************************************************************* /// Default constructor. //************************************************************************* ETL_CONSTEXPR14 delegate() { } //************************************************************************* // Copy constructor. //************************************************************************* ETL_CONSTEXPR14 delegate(const delegate& other) = default; //************************************************************************* // Construct from lambda or functor. //************************************************************************* template ::value && !etl::is_same, TLambda>::value, void>> ETL_CONSTEXPR14 delegate(TLambda& instance) { assign((void*)(&instance), lambda_stub); } //************************************************************************* // Construct from const lambda or functor. //************************************************************************* template ::value && !etl::is_same, TLambda>::value, void>> ETL_CONSTEXPR14 delegate(const TLambda& instance) { assign((void*)(&instance), const_lambda_stub); } //************************************************************************* /// Create from function (Compile time). //************************************************************************* template ETL_NODISCARD static ETL_CONSTEXPR14 delegate create() { return delegate(ETL_NULLPTR, function_stub); } //************************************************************************* /// Create from Lambda or Functor. //************************************************************************* template ::value && !etl::is_same, TLambda>::value, void>> ETL_NODISCARD static ETL_CONSTEXPR14 delegate create(TLambda& instance) { return delegate((void*)(&instance), lambda_stub); } //************************************************************************* /// Create from const Lambda or Functor. //************************************************************************* template ::value && !etl::is_same, TLambda>::value, void>> ETL_NODISCARD static ETL_CONSTEXPR14 delegate create(const TLambda& instance) { return delegate((void*)(&instance), const_lambda_stub); } //************************************************************************* /// Create from instance method (Run time). //************************************************************************* template ETL_NODISCARD static ETL_CONSTEXPR14 delegate create(T& instance) { return delegate((void*)(&instance), method_stub); } //************************************************************************* /// Create from instance method (Run time). /// Deleted for rvalue references. //************************************************************************* template ETL_NODISCARD static ETL_CONSTEXPR14 delegate create(T&& instance) = delete; //************************************************************************* /// Create from const instance method (Run time). //************************************************************************* template ETL_NODISCARD static ETL_CONSTEXPR14 delegate create(const T& instance) { return delegate((void*)(&instance), const_method_stub); } //************************************************************************* /// Disable create from rvalue instance method (Run time). //************************************************************************* template static ETL_CONSTEXPR14 delegate create(T&& instance) = delete; //************************************************************************* /// Create from instance method (Compile time). //************************************************************************* template ETL_NODISCARD static ETL_CONSTEXPR14 delegate create() { return delegate(method_instance_stub); } //************************************************************************* /// Create from instance method (Compile time). /// New API //************************************************************************* template ETL_NODISCARD static ETL_CONSTEXPR14 delegate create() { return delegate(method_instance_stub); } //************************************************************************* /// Create from const instance method (Compile time). //************************************************************************* template ETL_NODISCARD static ETL_CONSTEXPR14 delegate create() { return delegate(const_method_instance_stub); } //************************************************************************* /// Create from const instance method (Compile time). /// New API //************************************************************************* template ETL_NODISCARD static ETL_CONSTEXPR14 delegate create() { return delegate(const_method_instance_stub); } #if !(defined(ETL_COMPILER_GCC) && (__GNUC__ <= 8)) //************************************************************************* /// Create from instance function operator (Compile time). /// At the time of writing, GCC appears to have trouble with this. //************************************************************************* template ETL_NODISCARD static ETL_CONSTEXPR14 delegate create() { return delegate(operator_instance_stub); } #endif //************************************************************************* /// Set from function (Compile time). //************************************************************************* template ETL_CONSTEXPR14 void set() { assign(ETL_NULLPTR, function_stub); } //************************************************************************* /// Set from Lambda or Functor. //************************************************************************* template ::value && !etl::is_same, TLambda>::value, void>> ETL_CONSTEXPR14 void set(TLambda& instance) { assign((void*)(&instance), lambda_stub); } //************************************************************************* /// Set from const Lambda or Functor. //************************************************************************* template ::value && !etl::is_same, TLambda>::value, void>> ETL_CONSTEXPR14 void set(const TLambda& instance) { assign((void*)(&instance), const_lambda_stub); } //************************************************************************* /// Set from instance method (Run time). //************************************************************************* template ETL_CONSTEXPR14 void set(T& instance) { assign((void*)(&instance), method_stub); } //************************************************************************* /// Set from const instance method (Run time). //************************************************************************* template ETL_CONSTEXPR14 void set(T& instance) { assign((void*)(&instance), const_method_stub); } //************************************************************************* /// Set from instance method (Compile time). //************************************************************************* template ETL_CONSTEXPR14 void set() { assign(ETL_NULLPTR, method_instance_stub); } //************************************************************************* /// Set from instance method (Compile time). /// New API //************************************************************************* template ETL_CONSTEXPR14 void set() { assign(ETL_NULLPTR, method_instance_stub); } //************************************************************************* /// Set from const instance method (Compile time). //************************************************************************* template ETL_CONSTEXPR14 void set() { assign(ETL_NULLPTR, const_method_instance_stub); } //************************************************************************* /// Set from const instance method (Compile time). /// New API //************************************************************************* template ETL_CONSTEXPR14 void set() { assign(ETL_NULLPTR, const_method_instance_stub); } //************************************************************************* /// Clear the delegate. //************************************************************************* ETL_CONSTEXPR14 void clear() { invocation.clear(); } //************************************************************************* /// Execute the delegate. //************************************************************************* TReturn operator()(TParams... args) const { ETL_ASSERT(is_valid(), ETL_ERROR(delegate_uninitialised)); return (*invocation.stub)(invocation.object, etl::forward(args)...); } //************************************************************************* /// Execute the delegate if valid. /// 'void' return. //************************************************************************* template ETL_CONSTEXPR14 typename etl::enable_if_t::value, bool> call_if(TParams... args) const { if (is_valid()) { (*invocation.stub)(invocation.object, etl::forward(args)...); return true; } else { return false; } } //************************************************************************* /// Execute the delegate if valid. /// Non 'void' return. //************************************************************************* template ETL_CONSTEXPR14 typename etl::enable_if_t::value, etl::optional> call_if(TParams... args) const { etl::optional result; if (is_valid()) { result = (*invocation.stub)(invocation.object, etl::forward(args)...); } return result; } //************************************************************************* /// Execute the delegate if valid or call alternative. /// Run time alternative. //************************************************************************* template TReturn call_or(TAlternative alternative, TParams... args) const { if (is_valid()) { return (*invocation.stub)(invocation.object, etl::forward(args)...); } else { return alternative(etl::forward(args)...); } } //************************************************************************* /// Execute the delegate if valid or call alternative. /// Compile time alternative. //************************************************************************* template TReturn call_or(TParams... args) const { if (is_valid()) { return (*invocation.stub)(invocation.object, etl::forward(args)...); } else { return (Method)(etl::forward(args)...); } } //************************************************************************* /// Assignment //************************************************************************* delegate& operator =(const delegate& rhs) = default; //************************************************************************* /// Create from Lambda or Functor. //************************************************************************* template ::value && !etl::is_same, TLambda>::value, void>> ETL_CONSTEXPR14 delegate& operator =(TLambda& instance) { assign((void*)(&instance), lambda_stub); return *this; } //************************************************************************* /// Create from const Lambda or Functor. //************************************************************************* template ::value && !etl::is_same, TLambda>::value, void>> ETL_CONSTEXPR14 delegate& operator =(const TLambda& instance) { assign((void*)(&instance), const_lambda_stub); return *this; } //************************************************************************* /// Checks equality. //************************************************************************* ETL_CONSTEXPR14 bool operator == (const delegate& rhs) const { return invocation == rhs.invocation; } //************************************************************************* /// Returns true if the delegate is valid. //************************************************************************* ETL_CONSTEXPR14 bool operator != (const delegate& rhs) const { return invocation != rhs.invocation; } //************************************************************************* /// Returns true if the delegate is valid. //************************************************************************* ETL_NODISCARD ETL_CONSTEXPR14 bool is_valid() const { return invocation.stub != ETL_NULLPTR; } //************************************************************************* /// Returns true if the delegate is valid. //************************************************************************* ETL_CONSTEXPR14 operator bool() const { return is_valid(); } private: using stub_type = TReturn(*)(void* object, TParams...); //************************************************************************* /// The internal invocation object. //************************************************************************* struct invocation_element { invocation_element() = default; //*********************************************************************** ETL_CONSTEXPR14 invocation_element(void* object_, stub_type stub_) : object(object_) , stub(stub_) { } //*********************************************************************** ETL_CONSTEXPR14 bool operator ==(const invocation_element& rhs) const { return (rhs.stub == stub) && (rhs.object == object); } //*********************************************************************** ETL_CONSTEXPR14 bool operator !=(const invocation_element& rhs) const { return (rhs.stub != stub) || (rhs.object != object); } //*********************************************************************** ETL_CONSTEXPR14 void clear() { object = ETL_NULLPTR; stub = ETL_NULLPTR; } //*********************************************************************** void* object = ETL_NULLPTR; stub_type stub = ETL_NULLPTR; }; //************************************************************************* /// Constructs a delegate from an object and stub. //************************************************************************* ETL_CONSTEXPR14 delegate(void* object, stub_type stub) : invocation(object, stub) { } //************************************************************************* /// Constructs a delegate from a stub. //************************************************************************* ETL_CONSTEXPR14 delegate(stub_type stub) : invocation(ETL_NULLPTR, stub) { } //************************************************************************* /// Assign from an object and stub. //************************************************************************* ETL_CONSTEXPR14 void assign(void* object, stub_type stub) { invocation.object = object; invocation.stub = stub; } //************************************************************************* /// Stub call for a member function. Run time instance. //************************************************************************* template static ETL_CONSTEXPR14 TReturn method_stub(void* object, TParams... params) { T* p = static_cast(object); return (p->*Method)(etl::forward(params)...); } //************************************************************************* /// Stub call for a const member function. Run time instance. //************************************************************************* template static ETL_CONSTEXPR14 TReturn const_method_stub(void* object, TParams... params) { T* const p = static_cast(object); return (p->*Method)(etl::forward(params)...); } //************************************************************************* /// Stub call for a member function. Compile time instance. //************************************************************************* template static ETL_CONSTEXPR14 TReturn method_instance_stub(void*, TParams... params) { return (Instance.*Method)(etl::forward(params)...); } //************************************************************************* /// Stub call for a const member function. Compile time instance. //************************************************************************* template static ETL_CONSTEXPR14 TReturn const_method_instance_stub(void*, TParams... params) { return (Instance.*Method)(etl::forward(params)...); } #if !(defined(ETL_COMPILER_GCC) && (__GNUC__ <= 8)) //************************************************************************* /// Stub call for a function operator. Compile time instance. //************************************************************************* template static ETL_CONSTEXPR14 TReturn operator_instance_stub(void*, TParams... params) { return Instance.operator()(etl::forward(params)...); } #endif //************************************************************************* /// Stub call for a free function. //************************************************************************* template static ETL_CONSTEXPR14 TReturn function_stub(void*, TParams... params) { return (Method)(etl::forward(params)...); } //************************************************************************* /// Stub call for a lambda or functor function. //************************************************************************* template static ETL_CONSTEXPR14 TReturn lambda_stub(void* object, TParams... arg) { TLambda* p = static_cast(object); return (p->operator())(etl::forward(arg)...); } //************************************************************************* /// Stub call for a const lambda or functor function. //************************************************************************* template static ETL_CONSTEXPR14 TReturn const_lambda_stub(void* object, TParams... arg) { const TLambda* p = static_cast(object); return (p->operator())(etl::forward(arg)...); } //************************************************************************* /// The invocation object. //************************************************************************* invocation_element invocation; }; } #endif