2022-04-07 15:57:02 +08:00
|
|
|
#pragma once
|
|
|
|
|
|
|
|
#include "CoreTypes.h"
|
|
|
|
#include "Templates/Any.h"
|
|
|
|
#include "Templates/Tuple.h"
|
|
|
|
#include "Templates/Invoke.h"
|
2022-04-27 22:50:56 +08:00
|
|
|
#include "Memory/Alignment.h"
|
2022-04-08 11:44:36 +08:00
|
|
|
#include "Templates/Utility.h"
|
2022-04-22 22:28:44 +08:00
|
|
|
#include "Templates/TypeHash.h"
|
2022-04-07 15:57:02 +08:00
|
|
|
#include "TypeTraits/TypeTraits.h"
|
|
|
|
#include "Miscellaneous/TypeInfo.h"
|
2022-04-08 11:44:36 +08:00
|
|
|
#include "Concepts/BooleanTestable.h"
|
2022-04-07 15:57:02 +08:00
|
|
|
#include "Miscellaneous/AssertionMacros.h"
|
|
|
|
|
2022-04-30 23:38:09 +08:00
|
|
|
// NOTE: Disable alignment limit warning
|
|
|
|
#pragma warning(disable : 4359)
|
|
|
|
|
2022-04-07 15:57:02 +08:00
|
|
|
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,
|
|
|
|
};
|
|
|
|
|
2022-04-27 22:50:56 +08:00
|
|
|
template <typename F, size_t InlineSize, size_t InlineAlignment, EFunctionSpecifiers Specifiers, EFunctionType FunctionType> requires (Memory::IsValidAlignment(InlineAlignment))
|
2022-04-07 15:57:02 +08:00
|
|
|
struct TFunctionImpl;
|
|
|
|
|
|
|
|
template <typename T> struct TIsTFunctionImpl : FFalse { };
|
|
|
|
template <typename F, size_t I, size_t J, EFunctionSpecifiers S, EFunctionType E> struct TIsTFunctionImpl<TFunctionImpl<F, I, J, S, E>> : FTrue { };
|
|
|
|
|
|
|
|
template <typename T> struct TIsTFunctionRef : FFalse { };
|
|
|
|
template <typename F, size_t I, size_t J, EFunctionSpecifiers S> struct TIsTFunctionRef<TFunctionImpl<F, I, J, S, EFunctionType::Reference>> : FTrue { };
|
|
|
|
|
|
|
|
template <typename T> struct TIsTFunction : FFalse { };
|
|
|
|
template <typename F, size_t I, size_t J, EFunctionSpecifiers S> struct TIsTFunction<TFunctionImpl<F, I, J, S, EFunctionType::Object>> : FTrue { };
|
|
|
|
|
|
|
|
template <typename T> struct TIsTUniqueFunction : FFalse { };
|
|
|
|
template <typename F, size_t I, size_t J, EFunctionSpecifiers S> struct TIsTUniqueFunction<TFunctionImpl<F, I, J, S, EFunctionType::Unique>> : FTrue { };
|
|
|
|
|
2022-04-24 22:42:40 +08:00
|
|
|
struct FFunctionIsBound
|
2022-04-07 15:57:02 +08:00
|
|
|
{
|
|
|
|
template <typename T>
|
|
|
|
static constexpr bool F(const T& Func)
|
|
|
|
{
|
|
|
|
if constexpr (TIsPointer<T>::Value || TIsMemberPointer<T>::Value || TIsTFunctionImpl<T>::Value)
|
|
|
|
{
|
|
|
|
return !!Func;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
template <EFunctionSpecifiers Specifiers, typename R, typename F, typename... Types>
|
|
|
|
struct TIsInvocableResultWithSpecifiers : FFalse { };
|
|
|
|
|
|
|
|
template <typename R, typename F, typename... Types>
|
|
|
|
struct TIsInvocableResultWithSpecifiers<EFunctionSpecifiers::None, R, F, Types...>
|
|
|
|
: TBoolConstant<TIsInvocableResult<R, F, Types...>::Value && TIsInvocableResult<R, F&, Types...>::Value>
|
|
|
|
{ };
|
|
|
|
|
|
|
|
template <typename R, typename F, typename... Types>
|
|
|
|
struct TIsInvocableResultWithSpecifiers<EFunctionSpecifiers::LValue, R, F, Types...> : TIsInvocableResult<R, F&, Types...> { };
|
|
|
|
|
|
|
|
template <typename R, typename F, typename... Types>
|
|
|
|
struct TIsInvocableResultWithSpecifiers<EFunctionSpecifiers::RValue, R, F, Types...> : TIsInvocableResult<R, F, Types...> { };
|
|
|
|
|
|
|
|
template <typename R, typename F, typename... Types>
|
|
|
|
struct TIsInvocableResultWithSpecifiers<EFunctionSpecifiers::Const, R, F, Types...>
|
|
|
|
: TBoolConstant<TIsInvocableResult<R, const F, Types...>::Value && TIsInvocableResult<R, const F&, Types...>::Value>
|
|
|
|
{ };
|
|
|
|
|
|
|
|
template <typename R, typename F, typename... Types>
|
|
|
|
struct TIsInvocableResultWithSpecifiers<EFunctionSpecifiers::ConstLValue, R, F, Types...> : TIsInvocableResult<R, const F&, Types...> { };
|
|
|
|
|
|
|
|
template <typename R, typename F, typename... Types>
|
|
|
|
struct TIsInvocableResultWithSpecifiers<EFunctionSpecifiers::ConstRValue, R, F, Types...> : TIsInvocableResult<R, const F, Types...> { };
|
|
|
|
|
2022-04-24 22:42:40 +08:00
|
|
|
template <typename T, EFunctionSpecifiers Specifiers> struct TFunctionCallSpecifiers;
|
|
|
|
template <typename T> struct TFunctionCallSpecifiers<T, EFunctionSpecifiers::None> { using Type = T& ; };
|
|
|
|
template <typename T> struct TFunctionCallSpecifiers<T, EFunctionSpecifiers::LValue> { using Type = T& ; };
|
|
|
|
template <typename T> struct TFunctionCallSpecifiers<T, EFunctionSpecifiers::RValue> { using Type = T&&; };
|
|
|
|
template <typename T> struct TFunctionCallSpecifiers<T, EFunctionSpecifiers::Const> { using Type = const T& ; };
|
|
|
|
template <typename T> struct TFunctionCallSpecifiers<T, EFunctionSpecifiers::ConstLValue> { using Type = const T& ; };
|
|
|
|
template <typename T> struct TFunctionCallSpecifiers<T, EFunctionSpecifiers::ConstRValue> { using Type = const T&&; };
|
2022-04-07 15:57:02 +08:00
|
|
|
|
|
|
|
template <typename R, typename... Types, size_t InlineSize, size_t InlineAlignment, EFunctionSpecifiers Specifiers, EFunctionType FunctionType>
|
2022-04-30 23:38:09 +08:00
|
|
|
struct alignas(InlineAlignment) TFunctionImpl<R(Types...), InlineSize, InlineAlignment, Specifiers, FunctionType>
|
2022-04-07 15:57:02 +08:00
|
|
|
{
|
|
|
|
public:
|
|
|
|
|
|
|
|
using ResultType = R;
|
|
|
|
using ArgumentType = TTuple<Types...>;
|
|
|
|
|
2022-04-24 22:42:40 +08:00
|
|
|
constexpr TFunctionImpl(nullptr_t = nullptr) requires (FunctionType != EFunctionType::Reference) : Callable(nullptr) { }
|
2022-04-07 15:57:02 +08:00
|
|
|
|
|
|
|
TFunctionImpl(const TFunctionImpl& InValue) requires (FunctionType != EFunctionType::Unique)
|
2022-04-24 22:42:40 +08:00
|
|
|
: Callable(InValue.Callable), Storage(InValue.Storage)
|
2022-04-07 15:57:02 +08:00
|
|
|
{ }
|
|
|
|
|
|
|
|
TFunctionImpl(TFunctionImpl&& InValue)
|
2022-04-24 22:42:40 +08:00
|
|
|
: Callable(InValue.Callable), Storage(MoveTemp(InValue.Storage))
|
2022-04-07 15:57:02 +08:00
|
|
|
{ if constexpr (FunctionType != EFunctionType::Reference) InValue.Reset(); }
|
|
|
|
|
|
|
|
template <typename T> requires (!TIsTFunctionImpl<typename TDecay<T>::Type>::Value) && (!TIsTInPlaceType<typename TDecay<T>::Type>::Value)
|
|
|
|
&& TIsInvocableResultWithSpecifiers<Specifiers, ResultType, typename TDecay<T>::Type, Types...>::Value
|
|
|
|
&& (FunctionType == EFunctionType::Reference || TIsConstructible<typename TDecay<T>::Type, T&&>::Value)
|
|
|
|
&& ((FunctionType == EFunctionType::Object && TIsCopyConstructible<typename TDecay<T>::Type>::Value)
|
|
|
|
|| (FunctionType == EFunctionType::Unique && TIsMoveConstructible<typename TDecay<T>::Type>::Value)
|
|
|
|
|| FunctionType == EFunctionType::Reference)
|
|
|
|
FORCEINLINE TFunctionImpl(T&& InValue)
|
|
|
|
{
|
|
|
|
using DecayedFunctorType = typename TDecay<T>::Type;
|
|
|
|
|
|
|
|
if constexpr (FunctionType == EFunctionType::Reference)
|
|
|
|
{
|
2022-04-24 22:42:40 +08:00
|
|
|
checkf(FFunctionIsBound::F(InValue), TEXT("Cannot bind a null/unbound callable to a TFunctionRef"));
|
2022-04-07 15:57:02 +08:00
|
|
|
}
|
|
|
|
|
2022-04-24 22:42:40 +08:00
|
|
|
if (!FFunctionIsBound::F(InValue)) Callable = nullptr;
|
2022-04-07 15:57:02 +08:00
|
|
|
else EmplaceImpl<DecayedFunctorType>(Forward<T>(InValue));
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename T, typename... ArgTypes> requires (FunctionType != EFunctionType::Reference)
|
|
|
|
&& TIsInvocableResultWithSpecifiers<Specifiers, ResultType, typename TDecay<T>::Type, Types...>::Value && TIsConstructible<typename TDecay<T>::Type, ArgTypes...>::Value
|
|
|
|
&& ((FunctionType == EFunctionType::Object && TIsCopyConstructible<typename TDecay<T>::Type>::Value)
|
|
|
|
|| (FunctionType == EFunctionType::Unique && TIsMoveConstructible<typename TDecay<T>::Type>::Value))
|
|
|
|
FORCEINLINE TFunctionImpl(TInPlaceType<T>, ArgTypes&&... Args)
|
|
|
|
{
|
|
|
|
using DecayedFunctorType = typename TDecay<T>::Type;
|
|
|
|
EmplaceImpl<DecayedFunctorType>(Forward<ArgTypes>(Args)...);
|
|
|
|
}
|
|
|
|
|
|
|
|
template <size_t OtherInlineSize, size_t OtherInlineAlignment, EFunctionType OtherFunctionType>
|
|
|
|
requires (FunctionType == EFunctionType::Reference) && (OtherFunctionType != EFunctionType::Reference)
|
|
|
|
FORCEINLINE TFunctionImpl(const TFunctionImpl<R(Types...), OtherInlineSize, OtherInlineAlignment, Specifiers, OtherFunctionType>& InValue)
|
|
|
|
{
|
2022-04-24 22:42:40 +08:00
|
|
|
checkf(FFunctionIsBound::F(InValue), TEXT("Cannot bind a null/unbound callable to a TFunctionRef"));
|
2022-04-07 15:57:02 +08:00
|
|
|
EmplaceImpl<TFunctionImpl<R(Types...), OtherInlineSize, OtherInlineAlignment, Specifiers, OtherFunctionType>>(InValue);
|
|
|
|
}
|
|
|
|
|
|
|
|
FORCEINLINE TFunctionImpl(const TFunctionImpl<R(Types...), InlineSize, InlineAlignment, Specifiers, EFunctionType::Object>& InValue) requires (FunctionType == EFunctionType::Unique)
|
2022-04-24 22:42:40 +08:00
|
|
|
: Callable((*reinterpret_cast<const TFunctionImpl*>(&InValue)).Callable), Storage((*reinterpret_cast<const TFunctionImpl*>(&InValue)).Storage)
|
2022-04-07 15:57:02 +08:00
|
|
|
{ }
|
|
|
|
|
|
|
|
FORCEINLINE TFunctionImpl(TFunctionImpl<R(Types...), InlineSize, InlineAlignment, Specifiers, EFunctionType::Object>&& InValue) requires (FunctionType == EFunctionType::Unique)
|
2022-04-24 22:42:40 +08:00
|
|
|
: Callable((*reinterpret_cast<TFunctionImpl*>(&InValue)).Callable), Storage(MoveTemp((*reinterpret_cast<TFunctionImpl*>(&InValue)).Storage))
|
2022-04-07 15:57:02 +08:00
|
|
|
{ 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<R(Types...), InlineSize, InlineAlignment, Specifiers, EFunctionType::Object>& InValue) requires (FunctionType == EFunctionType::Unique)
|
|
|
|
{
|
|
|
|
AssignImpl(*reinterpret_cast<const TFunctionImpl*>(&InValue));
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
FORCEINLINE TFunctionImpl& operator=(TFunctionImpl<R(Types...), InlineSize, InlineAlignment, Specifiers, EFunctionType::Object>&& InValue) requires (FunctionType == EFunctionType::Unique)
|
|
|
|
{
|
|
|
|
AssignImpl(MoveTemp(*reinterpret_cast<TFunctionImpl*>(&InValue)));
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
constexpr TFunctionImpl& operator=(nullptr_t) requires (FunctionType != EFunctionType::Reference) { Reset(); return *this; }
|
|
|
|
|
|
|
|
template <typename T> requires (FunctionType != EFunctionType::Reference) && (!TIsTFunctionImpl<typename TDecay<T>::Type>::Value)
|
|
|
|
&& TIsInvocableResultWithSpecifiers<Specifiers, ResultType, typename TDecay<T>::Type, Types...>::Value
|
|
|
|
&& TIsConstructible<typename TDecay<T>::Type, T&&>::Value
|
|
|
|
&& ((FunctionType == EFunctionType::Object && TIsCopyConstructible<typename TDecay<T>::Type>::Value)
|
|
|
|
|| (FunctionType == EFunctionType::Unique && TIsMoveConstructible<typename TDecay<T>::Type>::Value))
|
|
|
|
FORCEINLINE TFunctionImpl& operator=(T&& InValue)
|
|
|
|
{
|
|
|
|
using DecayedFunctorType = typename TDecay<T>::Type;
|
|
|
|
|
2022-04-24 22:42:40 +08:00
|
|
|
if (!FFunctionIsBound::F(InValue)) Reset();
|
2022-04-07 15:57:02 +08:00
|
|
|
else EmplaceImpl<DecayedFunctorType>(Forward<T>(InValue));
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename T, typename... ArgTypes> requires (FunctionType != EFunctionType::Reference)
|
|
|
|
&& TIsInvocableResultWithSpecifiers<Specifiers, ResultType, typename TDecay<T>::Type, Types...>::Value
|
|
|
|
&& TIsConstructible<typename TDecay<T>::Type, ArgTypes...>::Value
|
|
|
|
&& ((FunctionType == EFunctionType::Object && TIsCopyConstructible<typename TDecay<T>::Type>::Value)
|
|
|
|
|| (FunctionType == EFunctionType::Unique && TIsMoveConstructible<typename TDecay<T>::Type>::Value))
|
|
|
|
FORCEINLINE typename TDecay<T>::Type& Emplace(ArgTypes&&... Args)
|
|
|
|
{
|
|
|
|
using DecayedFunctorType = typename TDecay<T>::Type;
|
|
|
|
EmplaceImpl<DecayedFunctorType>(Forward<ArgTypes>(Args)...);
|
2022-04-08 16:52:17 +08:00
|
|
|
return Target<DecayedFunctorType>();
|
2022-04-07 15:57:02 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
FORCEINLINE ResultType operator()(Types... Args) requires (Specifiers == EFunctionSpecifiers::None ) { return CallImpl(Forward<Types>(Args)...); }
|
|
|
|
FORCEINLINE ResultType operator()(Types... Args) & requires (Specifiers == EFunctionSpecifiers::LValue ) { return CallImpl(Forward<Types>(Args)...); }
|
|
|
|
FORCEINLINE ResultType operator()(Types... Args) && requires (Specifiers == EFunctionSpecifiers::RValue ) { return CallImpl(Forward<Types>(Args)...); }
|
|
|
|
FORCEINLINE ResultType operator()(Types... Args) const requires (Specifiers == EFunctionSpecifiers::Const ) { return CallImpl(Forward<Types>(Args)...); }
|
|
|
|
FORCEINLINE ResultType operator()(Types... Args) const& requires (Specifiers == EFunctionSpecifiers::ConstLValue) { return CallImpl(Forward<Types>(Args)...); }
|
|
|
|
FORCEINLINE ResultType operator()(Types... Args) const&& requires (Specifiers == EFunctionSpecifiers::ConstRValue) { return CallImpl(Forward<Types>(Args)...); }
|
|
|
|
|
2022-04-24 22:42:40 +08:00
|
|
|
constexpr bool IsValid() const { return Callable != nullptr; }
|
|
|
|
constexpr explicit operator bool() const { return Callable != nullptr; }
|
2022-04-08 16:52:17 +08:00
|
|
|
|
2022-04-07 15:57:02 +08:00
|
|
|
FORCEINLINE const FTypeInfo& TargetType() const requires (FunctionType != EFunctionType::Reference) { return IsValid() ? Storage.GetTypeInfo() : Typeid(void); };
|
|
|
|
|
2022-04-08 16:52:17 +08:00
|
|
|
template <typename T> FORCEINLINE T& Target() & requires (FunctionType != EFunctionType::Reference) && TIsSame<T, typename TDecay<T>::Type>::Value && TIsObject<typename TDecay<T>::Type>::Value && (!TIsArray<typename TDecay<T>::Type>::Value) && TIsDestructible<typename TDecay<T>::Type>::Value { return static_cast< StorageType& >(Storage).template GetValue<T>(); }
|
|
|
|
template <typename T> FORCEINLINE T&& Target() && requires (FunctionType != EFunctionType::Reference) && TIsSame<T, typename TDecay<T>::Type>::Value && TIsObject<typename TDecay<T>::Type>::Value && (!TIsArray<typename TDecay<T>::Type>::Value) && TIsDestructible<typename TDecay<T>::Type>::Value { return static_cast< StorageType&&>(Storage).template GetValue<T>(); }
|
|
|
|
template <typename T> FORCEINLINE const T& Target() const& requires (FunctionType != EFunctionType::Reference) && TIsSame<T, typename TDecay<T>::Type>::Value && TIsObject<typename TDecay<T>::Type>::Value && (!TIsArray<typename TDecay<T>::Type>::Value) && TIsDestructible<typename TDecay<T>::Type>::Value { return static_cast<const StorageType& >(Storage).template GetValue<T>(); }
|
|
|
|
template <typename T> FORCEINLINE const T&& Target() const&& requires (FunctionType != EFunctionType::Reference) && TIsSame<T, typename TDecay<T>::Type>::Value && TIsObject<typename TDecay<T>::Type>::Value && (!TIsArray<typename TDecay<T>::Type>::Value) && TIsDestructible<typename TDecay<T>::Type>::Value { return static_cast<const StorageType&&>(Storage).template GetValue<T>(); }
|
2022-04-07 15:57:02 +08:00
|
|
|
|
2022-04-24 22:42:40 +08:00
|
|
|
constexpr void Reset() requires (FunctionType != EFunctionType::Reference) { Callable = nullptr; }
|
2022-04-07 15:57:02 +08:00
|
|
|
|
2022-04-14 22:41:22 +08:00
|
|
|
constexpr void Swap(TFunctionImpl& InValue) requires (FunctionType != EFunctionType::Reference)
|
|
|
|
{
|
|
|
|
if (!IsValid() && !InValue.IsValid()) return;
|
|
|
|
|
|
|
|
if (IsValid() && !InValue.IsValid())
|
|
|
|
{
|
|
|
|
InValue = MoveTemp(*this);
|
|
|
|
Reset();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (InValue.IsValid() && !IsValid())
|
|
|
|
{
|
|
|
|
*this = MoveTemp(InValue);
|
|
|
|
InValue.Reset();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2022-04-24 22:42:40 +08:00
|
|
|
NAMESPACE_REDCRAFT::Swap(Callable, InValue.Callable);
|
2022-04-14 22:41:22 +08:00
|
|
|
NAMESPACE_REDCRAFT::Swap(Storage, InValue.Storage);
|
|
|
|
}
|
|
|
|
|
2022-04-07 15:57:02 +08:00
|
|
|
private:
|
|
|
|
|
2022-04-30 23:38:09 +08:00
|
|
|
using StorageType = typename TConditional<FunctionType == EFunctionType::Reference, void*, TAny<InlineSize, 1>>::Type;
|
2022-04-24 22:42:40 +08:00
|
|
|
using StorageRef = typename TConditional<FunctionType == EFunctionType::Reference, void*, typename TFunctionCallSpecifiers<StorageType, Specifiers>::Type&>::Type;
|
|
|
|
using CallFunc = ResultType(*)(StorageRef, Types&&...);
|
2022-04-07 15:57:02 +08:00
|
|
|
|
|
|
|
StorageType Storage;
|
2022-04-24 22:42:40 +08:00
|
|
|
CallFunc Callable;
|
2022-04-07 15:57:02 +08:00
|
|
|
|
|
|
|
template <typename SelectedType, typename... ArgTypes>
|
|
|
|
FORCEINLINE void EmplaceImpl(ArgTypes&&... Args)
|
|
|
|
{
|
|
|
|
if constexpr (FunctionType == EFunctionType::Reference) Storage = ((void*)&Args, ...);
|
|
|
|
else Storage.template Emplace<SelectedType>(Forward<ArgTypes>(Args)...);
|
2022-04-24 22:42:40 +08:00
|
|
|
|
|
|
|
Callable = [](StorageRef Storage, Types&&... Args) -> ResultType
|
|
|
|
{
|
|
|
|
const auto GetFunc = [&Storage]() -> decltype(auto)
|
|
|
|
{
|
|
|
|
if constexpr (FunctionType == EFunctionType::Reference) return *reinterpret_cast<SelectedType*>(Storage);
|
|
|
|
else return Storage.template GetValue<SelectedType>();
|
|
|
|
};
|
|
|
|
|
|
|
|
return InvokeResult<R>(Forward<typename TFunctionCallSpecifiers<SelectedType, Specifiers>::Type>(GetFunc()), Forward<Types>(Args)...);
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
FORCEINLINE ResultType CallImpl(Types&&... Args)
|
|
|
|
{
|
|
|
|
checkf(IsValid(), TEXT("Attempting to call an unbound TFunction!"));
|
|
|
|
return Callable(Storage, Forward<Types>(Args)...);
|
2022-04-07 15:57:02 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
FORCEINLINE ResultType CallImpl(Types&&... Args) const
|
|
|
|
{
|
|
|
|
checkf(IsValid(), TEXT("Attempting to call an unbound TFunction!"));
|
2022-04-24 22:42:40 +08:00
|
|
|
return Callable(Storage, Forward<Types>(Args)...);
|
2022-04-07 15:57:02 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
FORCEINLINE void AssignImpl(const TFunctionImpl& InValue)
|
|
|
|
{
|
|
|
|
if (InValue.IsValid())
|
|
|
|
{
|
2022-04-24 22:42:40 +08:00
|
|
|
Callable = InValue.Callable;
|
2022-04-07 15:57:02 +08:00
|
|
|
Storage = InValue.Storage;
|
|
|
|
}
|
|
|
|
else Reset();
|
|
|
|
}
|
|
|
|
|
|
|
|
FORCEINLINE void AssignImpl(TFunctionImpl&& InValue)
|
|
|
|
{
|
|
|
|
if (InValue.IsValid())
|
|
|
|
{
|
2022-04-24 22:42:40 +08:00
|
|
|
Callable = InValue.Callable;
|
2022-04-07 15:57:02 +08:00
|
|
|
Storage = MoveTemp(InValue.Storage);
|
|
|
|
InValue.Reset();
|
|
|
|
}
|
|
|
|
else Reset();
|
|
|
|
}
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
template <typename F, size_t InlineSize, size_t InlineAlignment, EFunctionType FunctionType>
|
|
|
|
struct TFunctionSelect;
|
|
|
|
|
|
|
|
template <typename R, typename... Types, size_t InlineSize, size_t InlineAlignment, EFunctionType FunctionType>
|
|
|
|
struct TFunctionSelect<R(Types...) , InlineSize, InlineAlignment, FunctionType>
|
|
|
|
{
|
2022-04-08 16:52:17 +08:00
|
|
|
using Type = TFunctionImpl<R(Types...), InlineSize, InlineAlignment, EFunctionSpecifiers::None, FunctionType>;
|
2022-04-07 15:57:02 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
template <typename R, typename... Types, size_t InlineSize, size_t InlineAlignment, EFunctionType FunctionType>
|
|
|
|
struct TFunctionSelect<R(Types...) & , InlineSize, InlineAlignment, FunctionType>
|
|
|
|
{
|
2022-04-08 16:52:17 +08:00
|
|
|
using Type = TFunctionImpl<R(Types...), InlineSize, InlineAlignment, EFunctionSpecifiers::LValue, FunctionType>;
|
2022-04-07 15:57:02 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
template <typename R, typename... Types, size_t InlineSize, size_t InlineAlignment, EFunctionType FunctionType>
|
|
|
|
struct TFunctionSelect<R(Types...) && , InlineSize, InlineAlignment, FunctionType>
|
|
|
|
{
|
2022-04-08 16:52:17 +08:00
|
|
|
using Type = TFunctionImpl<R(Types...), InlineSize, InlineAlignment, EFunctionSpecifiers::RValue, FunctionType>;
|
2022-04-07 15:57:02 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
template <typename R, typename... Types, size_t InlineSize, size_t InlineAlignment, EFunctionType FunctionType>
|
|
|
|
struct TFunctionSelect<R(Types...) const , InlineSize, InlineAlignment, FunctionType>
|
|
|
|
{
|
2022-04-08 16:52:17 +08:00
|
|
|
using Type = TFunctionImpl<R(Types...), InlineSize, InlineAlignment, EFunctionSpecifiers::Const, FunctionType>;
|
2022-04-07 15:57:02 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
template <typename R, typename... Types, size_t InlineSize, size_t InlineAlignment, EFunctionType FunctionType>
|
|
|
|
struct TFunctionSelect<R(Types...) const& , InlineSize, InlineAlignment, FunctionType>
|
|
|
|
{
|
2022-04-08 16:52:17 +08:00
|
|
|
using Type = TFunctionImpl<R(Types...), InlineSize, InlineAlignment, EFunctionSpecifiers::ConstLValue, FunctionType>;
|
2022-04-07 15:57:02 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
template <typename R, typename... Types, size_t InlineSize, size_t InlineAlignment, EFunctionType FunctionType>
|
|
|
|
struct TFunctionSelect<R(Types...) const&&, InlineSize, InlineAlignment, FunctionType>
|
|
|
|
{
|
2022-04-08 16:52:17 +08:00
|
|
|
using Type = TFunctionImpl<R(Types...), InlineSize, InlineAlignment, EFunctionSpecifiers::ConstRValue, FunctionType>;
|
2022-04-07 15:57:02 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
NAMESPACE_PRIVATE_END
|
|
|
|
|
2022-04-30 23:38:09 +08:00
|
|
|
inline constexpr size_t FUNCTION_DEFAULT_INLINE_SIZE = ANY_DEFAULT_INLINE_SIZE - sizeof(uintptr);
|
|
|
|
inline constexpr size_t FUNCTION_DEFAULT_INLINE_ALIGNMENT = ANY_DEFAULT_INLINE_ALIGNMENT;
|
2022-04-07 15:57:02 +08:00
|
|
|
|
|
|
|
template <typename F>
|
2022-04-27 22:50:56 +08:00
|
|
|
using TFunctionRef = typename NAMESPACE_PRIVATE::TFunctionSelect<F, 0, 0, NAMESPACE_PRIVATE::EFunctionType::Reference>::Type;
|
2022-04-07 15:57:02 +08:00
|
|
|
|
|
|
|
template <typename F, size_t InlineSize = FUNCTION_DEFAULT_INLINE_SIZE, size_t InlineAlignment = FUNCTION_DEFAULT_INLINE_ALIGNMENT>
|
|
|
|
using TFunction = typename NAMESPACE_PRIVATE::TFunctionSelect<F, InlineSize, InlineAlignment, NAMESPACE_PRIVATE::EFunctionType::Object>::Type;
|
|
|
|
|
|
|
|
template <typename F, size_t InlineSize = FUNCTION_DEFAULT_INLINE_SIZE, size_t InlineAlignment = FUNCTION_DEFAULT_INLINE_ALIGNMENT>
|
|
|
|
using TUniqueFunction = typename NAMESPACE_PRIVATE::TFunctionSelect<F, InlineSize, InlineAlignment, NAMESPACE_PRIVATE::EFunctionType::Unique>::Type;
|
|
|
|
|
|
|
|
template <typename T> struct TIsTFunctionRef : NAMESPACE_PRIVATE::TIsTFunctionRef<T> { };
|
|
|
|
template <typename T> struct TIsTFunction : NAMESPACE_PRIVATE::TIsTFunction<T> { };
|
|
|
|
template <typename T> struct TIsTUniqueFunction : NAMESPACE_PRIVATE::TIsTUniqueFunction<T> { };
|
|
|
|
|
|
|
|
template <typename F>
|
|
|
|
constexpr bool operator==(const TFunctionRef<F>& LHS, nullptr_t)
|
|
|
|
{
|
|
|
|
return !LHS;
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename F>
|
|
|
|
constexpr bool operator==(const TFunction<F>& LHS, nullptr_t)
|
|
|
|
{
|
|
|
|
return !LHS;
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename F>
|
|
|
|
constexpr bool operator==(const TUniqueFunction<F>& LHS, nullptr_t)
|
|
|
|
{
|
|
|
|
return !LHS;
|
|
|
|
}
|
|
|
|
|
|
|
|
static_assert(sizeof(TFunction<void()>) == 64, "The byte size of TFunction is unexpected");
|
|
|
|
static_assert(sizeof(TUniqueFunction<void()>) == 64, "The byte size of TUniqueFunction is unexpected");
|
|
|
|
|
2022-04-08 11:44:36 +08:00
|
|
|
NAMESPACE_PRIVATE_BEGIN
|
|
|
|
|
|
|
|
template <typename F>
|
|
|
|
struct NotFunctionType
|
|
|
|
{
|
|
|
|
F Func;
|
|
|
|
|
|
|
|
NotFunctionType(const NotFunctionType&) = default;
|
|
|
|
NotFunctionType(NotFunctionType&&) = default;
|
|
|
|
|
|
|
|
template <typename InF>
|
|
|
|
constexpr NotFunctionType(InF&& InFunc) : Func(Forward<InF>(InFunc)) { }
|
|
|
|
|
|
|
|
template <typename... Types> requires TIsInvocable<F&, Types&&...>::Value
|
|
|
|
constexpr auto operator()(Types&&... Args) &
|
|
|
|
-> decltype(!Invoke(Func, Forward<Types>(Args)...))
|
|
|
|
{
|
|
|
|
return !Invoke(Func, Forward<Types>(Args)...);
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename... Types> requires TIsInvocable<F&&, Types&&...>::Value
|
|
|
|
constexpr auto operator()(Types&&... Args) &&
|
|
|
|
-> decltype(!Invoke(MoveTemp(Func), Forward<Types>(Args)...))
|
|
|
|
{
|
|
|
|
return !Invoke(MoveTemp(Func), Forward<Types>(Args)...);
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename... Types> requires TIsInvocable<const F&, Types&&...>::Value
|
|
|
|
constexpr auto operator()(Types&&... Args) const&
|
|
|
|
-> decltype(!Invoke(Func, Forward<Types>(Args)...))
|
|
|
|
{
|
|
|
|
return !Invoke(Func, Forward<Types>(Args)...);
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename... Types> requires TIsInvocable<const F&&, Types&&...>::Value
|
|
|
|
constexpr auto operator()(Types&&... Args) const&&
|
|
|
|
-> decltype(!Invoke(MoveTemp(Func), Forward<Types>(Args)...))
|
|
|
|
{
|
|
|
|
return !Invoke(MoveTemp(Func), Forward<Types>(Args)...);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
NAMESPACE_PRIVATE_END
|
|
|
|
|
|
|
|
template <typename F>
|
|
|
|
constexpr NAMESPACE_PRIVATE::NotFunctionType<typename TDecay<F>::Type> NotFn(F&& Func)
|
|
|
|
{
|
|
|
|
return NAMESPACE_PRIVATE::NotFunctionType<typename TDecay<F>::Type>(Forward<F>(Func));
|
|
|
|
}
|
|
|
|
|
2022-04-07 15:57:02 +08:00
|
|
|
NAMESPACE_MODULE_END(Utility)
|
|
|
|
NAMESPACE_MODULE_END(Redcraft)
|
|
|
|
NAMESPACE_REDCRAFT_END
|