#pragma once #include "CoreTypes.h" #include "Memory/Memory.h" #include "Templates/Any.h" #include "Templates/Tuple.h" #include "Templates/Invoke.h" #include "Templates/Container.h" #include "TypeTraits/TypeTraits.h" #include "Miscellaneous/TypeInfo.h" #include "Miscellaneous/AssertionMacros.h" NAMESPACE_REDCRAFT_BEGIN NAMESPACE_MODULE_BEGIN(Redcraft) NAMESPACE_MODULE_BEGIN(Utility) NAMESPACE_PRIVATE_BEGIN enum class EFunctionType { Reference, Object, Unique, }; enum class EFunctionSpecifiers { None, LValue, RValue, Const, ConstLValue, ConstRValue, }; template struct TFunctionImpl; template struct TIsTFunctionImpl : FFalse { }; template struct TIsTFunctionImpl> : FTrue { }; template struct TIsTFunctionRef : FFalse { }; template struct TIsTFunctionRef> : FTrue { }; template struct TIsTFunction : FFalse { }; template struct TIsTFunction> : FTrue { }; template struct TIsTUniqueFunction : FFalse { }; template struct TIsTUniqueFunction> : FTrue { }; struct FFunctionIsBoundImpl { template static constexpr bool F(const T& Func) { if constexpr (TIsPointer::Value || TIsMemberPointer::Value || TIsTFunctionImpl::Value) { return !!Func; } else { return true; } } }; template struct TIsInvocableResultWithSpecifiers : FFalse { }; template struct TIsInvocableResultWithSpecifiers : TBoolConstant::Value && TIsInvocableResult::Value> { }; template struct TIsInvocableResultWithSpecifiers : TIsInvocableResult { }; template struct TIsInvocableResultWithSpecifiers : TIsInvocableResult { }; template struct TIsInvocableResultWithSpecifiers : TBoolConstant::Value && TIsInvocableResult::Value> { }; template struct TIsInvocableResultWithSpecifiers : TIsInvocableResult { }; template struct TIsInvocableResultWithSpecifiers : TIsInvocableResult { }; template struct TFunctionCallImpl; template struct TFunctionCallImpl { static inline R F(void* Func, Types&&... Args) { return InvokeResult(*reinterpret_cast(Func), Forward(Args)...); } }; template struct TFunctionCallImpl { static inline R F(void* Func, Types&&... Args) { return InvokeResult(*reinterpret_cast(Func), Forward(Args)...); } }; template struct TFunctionCallImpl { static inline R F(void* Func, Types&&... Args) { return InvokeResult(MoveTemp(*reinterpret_cast(Func)), Forward(Args)...); } }; template struct TFunctionCallImpl { static inline R F(void* Func, Types&&... Args) { return InvokeResult(AsConst(*reinterpret_cast(Func)), Forward(Args)...); } }; template struct TFunctionCallImpl { static inline R F(void* Func, Types&&... Args) { return InvokeResult(AsConst(*reinterpret_cast(Func)), Forward(Args)...); } }; template struct TFunctionCallImpl { static inline R F(void* Func, Types&&... Args) { return InvokeResult(MoveTemp(AsConst(*reinterpret_cast(Func))), Forward(Args)...); } }; template struct TFunctionImpl { public: using ResultType = R; using ArgumentType = TTuple; constexpr TFunctionImpl(nullptr_t = nullptr) requires (FunctionType != EFunctionType::Reference) : Call(nullptr) { } TFunctionImpl(const TFunctionImpl& InValue) requires (FunctionType != EFunctionType::Unique) : Call(InValue.Call), Storage(InValue.Storage) { } TFunctionImpl(TFunctionImpl&& InValue) : Call(InValue.Call), Storage(MoveTemp(InValue.Storage)) { if constexpr (FunctionType != EFunctionType::Reference) InValue.Reset(); } template requires (!TIsTFunctionImpl::Type>::Value) && (!TIsTInPlaceType::Type>::Value) && TIsInvocableResultWithSpecifiers::Type, Types...>::Value && (FunctionType == EFunctionType::Reference || TIsConstructible::Type, T&&>::Value) && ((FunctionType == EFunctionType::Object && TIsCopyConstructible::Type>::Value) || (FunctionType == EFunctionType::Unique && TIsMoveConstructible::Type>::Value) || FunctionType == EFunctionType::Reference) FORCEINLINE TFunctionImpl(T&& InValue) { using DecayedFunctorType = typename TDecay::Type; if constexpr (FunctionType == EFunctionType::Reference) { checkf(FFunctionIsBoundImpl::F(InValue), TEXT("Cannot bind a null/unbound callable to a TFunctionRef")); } if (!FFunctionIsBoundImpl::F(InValue)) Call = nullptr; else EmplaceImpl(Forward(InValue)); } template requires (FunctionType != EFunctionType::Reference) && TIsInvocableResultWithSpecifiers::Type, Types...>::Value && TIsConstructible::Type, ArgTypes...>::Value && ((FunctionType == EFunctionType::Object && TIsCopyConstructible::Type>::Value) || (FunctionType == EFunctionType::Unique && TIsMoveConstructible::Type>::Value)) FORCEINLINE TFunctionImpl(TInPlaceType, ArgTypes&&... Args) { using DecayedFunctorType = typename TDecay::Type; EmplaceImpl(Forward(Args)...); } template requires (FunctionType == EFunctionType::Reference) && (OtherFunctionType != EFunctionType::Reference) FORCEINLINE TFunctionImpl(const TFunctionImpl& InValue) { checkf(FFunctionIsBoundImpl::F(InValue), TEXT("Cannot bind a null/unbound callable to a TFunctionRef")); EmplaceImpl>(InValue); } FORCEINLINE TFunctionImpl(const TFunctionImpl& InValue) requires (FunctionType == EFunctionType::Unique) : Call((*reinterpret_cast(&InValue)).Call), Storage((*reinterpret_cast(&InValue)).Storage) { } FORCEINLINE TFunctionImpl(TFunctionImpl&& InValue) requires (FunctionType == EFunctionType::Unique) : Call((*reinterpret_cast(&InValue)).Call), Storage(MoveTemp((*reinterpret_cast(&InValue)).Storage)) { InValue.Reset(); } ~TFunctionImpl() = default; FORCEINLINE TFunctionImpl& operator=(const TFunctionImpl& InValue) requires (FunctionType == EFunctionType::Object) { AssignImpl(InValue); return *this; } FORCEINLINE TFunctionImpl& operator=(TFunctionImpl&& InValue) requires (FunctionType != EFunctionType::Reference) { if (&InValue == this) return *this; AssignImpl(MoveTemp(InValue)); return *this; } FORCEINLINE TFunctionImpl& operator=(const TFunctionImpl& InValue) requires (FunctionType == EFunctionType::Unique) { AssignImpl(*reinterpret_cast(&InValue)); return *this; } FORCEINLINE TFunctionImpl& operator=(TFunctionImpl&& InValue) requires (FunctionType == EFunctionType::Unique) { AssignImpl(MoveTemp(*reinterpret_cast(&InValue))); return *this; } constexpr TFunctionImpl& operator=(nullptr_t) requires (FunctionType != EFunctionType::Reference) { Reset(); return *this; } template requires (FunctionType != EFunctionType::Reference) && (!TIsTFunctionImpl::Type>::Value) && TIsInvocableResultWithSpecifiers::Type, Types...>::Value && TIsConstructible::Type, T&&>::Value && ((FunctionType == EFunctionType::Object && TIsCopyConstructible::Type>::Value) || (FunctionType == EFunctionType::Unique && TIsMoveConstructible::Type>::Value)) FORCEINLINE TFunctionImpl& operator=(T&& InValue) { using DecayedFunctorType = typename TDecay::Type; if (!FFunctionIsBoundImpl::F(InValue)) Reset(); else EmplaceImpl(Forward(InValue)); return *this; } template requires (FunctionType != EFunctionType::Reference) && TIsInvocableResultWithSpecifiers::Type, Types...>::Value && TIsConstructible::Type, ArgTypes...>::Value && ((FunctionType == EFunctionType::Object && TIsCopyConstructible::Type>::Value) || (FunctionType == EFunctionType::Unique && TIsMoveConstructible::Type>::Value)) FORCEINLINE typename TDecay::Type& Emplace(ArgTypes&&... Args) { using DecayedFunctorType = typename TDecay::Type; EmplaceImpl(Forward(Args)...); return *reinterpret_cast::Type*>(Target()); } FORCEINLINE ResultType operator()(Types... Args) requires (Specifiers == EFunctionSpecifiers::None ) { return CallImpl(Forward(Args)...); } FORCEINLINE ResultType operator()(Types... Args) & requires (Specifiers == EFunctionSpecifiers::LValue ) { return CallImpl(Forward(Args)...); } FORCEINLINE ResultType operator()(Types... Args) && requires (Specifiers == EFunctionSpecifiers::RValue ) { return CallImpl(Forward(Args)...); } FORCEINLINE ResultType operator()(Types... Args) const requires (Specifiers == EFunctionSpecifiers::Const ) { return CallImpl(Forward(Args)...); } FORCEINLINE ResultType operator()(Types... Args) const& requires (Specifiers == EFunctionSpecifiers::ConstLValue) { return CallImpl(Forward(Args)...); } FORCEINLINE ResultType operator()(Types... Args) const&& requires (Specifiers == EFunctionSpecifiers::ConstRValue) { return CallImpl(Forward(Args)...); } constexpr bool IsValid() const { return Call != nullptr; } constexpr explicit operator bool() const { return Call != nullptr; } FORCEINLINE const FTypeInfo& TargetType() const requires (FunctionType != EFunctionType::Reference) { return IsValid() ? Storage.GetTypeInfo() : Typeid(void); }; FORCEINLINE void* Target() { if constexpr (FunctionType != EFunctionType::Reference) return Storage.GetData(); else return Storage; } FORCEINLINE const void* Target() const { if constexpr (FunctionType != EFunctionType::Reference) return Storage.GetData(); else return Storage; } constexpr void Reset() requires (FunctionType != EFunctionType::Reference) { Call = nullptr; } private: using CallFunc = ResultType(*)(void*, Types&&...); using StorageType = typename TConditional>::Type; CallFunc Call; StorageType Storage; template FORCEINLINE void EmplaceImpl(ArgTypes&&... Args) { if constexpr (FunctionType == EFunctionType::Reference) Storage = ((void*)&Args, ...); else Storage.template Emplace(Forward(Args)...); Call = &TFunctionCallImpl::F; } FORCEINLINE ResultType CallImpl(Types&&... Args) const { checkf(IsValid(), TEXT("Attempting to call an unbound TFunction!")); return Call(const_cast(*this).Target(), Forward(Args)...); } FORCEINLINE void AssignImpl(const TFunctionImpl& InValue) { if (InValue.IsValid()) { Call = InValue.Call; Storage = InValue.Storage; } else Reset(); } FORCEINLINE void AssignImpl(TFunctionImpl&& InValue) { if (InValue.IsValid()) { Call = InValue.Call; Storage = MoveTemp(InValue.Storage); InValue.Reset(); } else Reset(); } }; template struct TFunctionSelect; template struct TFunctionSelect { using Type = NAMESPACE_PRIVATE::TFunctionImpl; }; template struct TFunctionSelect { using Type = NAMESPACE_PRIVATE::TFunctionImpl; }; template struct TFunctionSelect { using Type = NAMESPACE_PRIVATE::TFunctionImpl; }; template struct TFunctionSelect { using Type = NAMESPACE_PRIVATE::TFunctionImpl; }; template struct TFunctionSelect { using Type = NAMESPACE_PRIVATE::TFunctionImpl; }; template struct TFunctionSelect { using Type = NAMESPACE_PRIVATE::TFunctionImpl; }; NAMESPACE_PRIVATE_END inline constexpr size_t FUNCTION_DEFAULT_INLINE_SIZE = 32; inline constexpr size_t FUNCTION_DEFAULT_INLINE_ALIGNMENT = 16; template using TFunctionRef = typename NAMESPACE_PRIVATE::TFunctionSelect::Type; template using TFunction = typename NAMESPACE_PRIVATE::TFunctionSelect::Type; template using TUniqueFunction = typename NAMESPACE_PRIVATE::TFunctionSelect::Type; template struct TIsTFunctionRef : NAMESPACE_PRIVATE::TIsTFunctionRef { }; template struct TIsTFunction : NAMESPACE_PRIVATE::TIsTFunction { }; template struct TIsTUniqueFunction : NAMESPACE_PRIVATE::TIsTUniqueFunction { }; template constexpr bool operator==(const TFunctionRef& LHS, nullptr_t) { return !LHS; } template constexpr bool operator==(const TFunction& LHS, nullptr_t) { return !LHS; } template constexpr bool operator==(const TUniqueFunction& LHS, nullptr_t) { return !LHS; } template constexpr void Swap(TFunction& A, TFunction& B) { if (!A && !B) return; if (A && !B) { B = MoveTemp(A); A = nullptr; return; } if (B && !A) { A = MoveTemp(B); B = nullptr; return; } TFunction Temp = MoveTemp(A); A = MoveTemp(B); B = MoveTemp(Temp); } template constexpr void Swap(TUniqueFunction& A, TUniqueFunction& B) { if (!A && !B) return; if (A && !B) { B = MoveTemp(A); A = nullptr; return; } if (B && !A) { A = MoveTemp(B); B = nullptr; return; } TFunction Temp = MoveTemp(A); A = MoveTemp(B); B = MoveTemp(Temp); } static_assert(sizeof(TFunctionRef) == 16, "The byte size of TFunctionRef is unexpected"); static_assert(sizeof(TFunction) == 64, "The byte size of TFunction is unexpected"); static_assert(sizeof(TUniqueFunction) == 64, "The byte size of TUniqueFunction is unexpected"); NAMESPACE_MODULE_END(Utility) NAMESPACE_MODULE_END(Redcraft) NAMESPACE_REDCRAFT_END