实现对类型的基础内存操作
This commit is contained in:
parent
6e3c008936
commit
83be3aa142
@ -1,4 +1,5 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include "CoreTypes.h"
|
#include "CoreTypes.h"
|
||||||
|
#include "Templates/MemoryOps.h"
|
||||||
#include "Misc/AssertionMacros.h"
|
#include "Misc/AssertionMacros.h"
|
||||||
|
@ -7,27 +7,27 @@
|
|||||||
NS_REDCRAFT_BEGIN
|
NS_REDCRAFT_BEGIN
|
||||||
NS_BEGIN(Memory)
|
NS_BEGIN(Memory)
|
||||||
|
|
||||||
void* Memmove(void* Dest, const void* Src, size_t Count)
|
FORCEINLINE void* Memmove(void* Dest, const void* Src, size_t Count)
|
||||||
{
|
{
|
||||||
return std::memmove(Dest, Src, Count);
|
return std::memmove(Dest, Src, Count);
|
||||||
}
|
}
|
||||||
|
|
||||||
int32 Memcmp(const void* Buf1, const void* Buf2, size_t Count)
|
FORCEINLINE int32 Memcmp(const void* Buf1, const void* Buf2, size_t Count)
|
||||||
{
|
{
|
||||||
return std::memcmp(Buf1, Buf2, Count);
|
return std::memcmp(Buf1, Buf2, Count);
|
||||||
}
|
}
|
||||||
|
|
||||||
void Memset(void* Dest, uint8 ValueToSet, size_t Count)
|
FORCEINLINE void Memset(void* Dest, uint8 ValueToSet, size_t Count)
|
||||||
{
|
{
|
||||||
std::memset(Dest, ValueToSet, Count);
|
std::memset(Dest, ValueToSet, Count);
|
||||||
}
|
}
|
||||||
|
|
||||||
void* Memzero(void* Dest, size_t Count)
|
FORCEINLINE void* Memzero(void* Dest, size_t Count)
|
||||||
{
|
{
|
||||||
return std::memset(Dest, 0, Count);
|
return std::memset(Dest, 0, Count);
|
||||||
}
|
}
|
||||||
|
|
||||||
void* Memcpy(void* Dest, const void* Src, size_t Count)
|
FORCEINLINE void* Memcpy(void* Dest, const void* Src, size_t Count)
|
||||||
{
|
{
|
||||||
return std::memcpy(Dest, Src, Count);
|
return std::memcpy(Dest, Src, Count);
|
||||||
}
|
}
|
||||||
@ -53,12 +53,12 @@ static FORCEINLINE void Memcpy(T& Dest, const T& Src)
|
|||||||
Memcpy(&Dest, &Src, sizeof(T));
|
Memcpy(&Dest, &Src, sizeof(T));
|
||||||
}
|
}
|
||||||
|
|
||||||
void* SystemMalloc(size_t Count)
|
FORCEINLINE void* SystemMalloc(size_t Count)
|
||||||
{
|
{
|
||||||
return std::malloc(Count);
|
return std::malloc(Count);
|
||||||
}
|
}
|
||||||
|
|
||||||
void SystemFree(void* Ptr)
|
FORCEINLINE void SystemFree(void* Ptr)
|
||||||
{
|
{
|
||||||
std::free(Ptr);
|
std::free(Ptr);
|
||||||
}
|
}
|
||||||
|
35
Redcraft.Core/Source/Public/Templates/MemoryOps.h
Normal file
35
Redcraft.Core/Source/Public/Templates/MemoryOps.h
Normal file
@ -0,0 +1,35 @@
|
|||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include "CoreTypes.h"
|
||||||
|
|
||||||
|
NS_REDCRAFT_BEGIN
|
||||||
|
NS_BEGIN(Memory)
|
||||||
|
|
||||||
|
template <typename ElementType, typename SizeType>
|
||||||
|
FORCEINLINE void DefaultConstructItems(void* Address, SizeType Count = 1);
|
||||||
|
|
||||||
|
template <typename ElementType, typename SizeType>
|
||||||
|
FORCEINLINE void DestructItems(ElementType* Element, SizeType Count = 1);
|
||||||
|
|
||||||
|
template <typename DestinationElementType, typename SourceElementType, typename SizeType>
|
||||||
|
FORCEINLINE void ConstructItems(void* Dest, const SourceElementType* Source, SizeType Count = 1);
|
||||||
|
|
||||||
|
template <typename ElementType, typename SizeType>
|
||||||
|
FORCEINLINE void CopyAssignItems(ElementType* Dest, const ElementType* Source, SizeType Count = 1);
|
||||||
|
|
||||||
|
template <typename DestinationElementType, typename SourceElementType, typename SizeType>
|
||||||
|
FORCEINLINE void RelocateConstructItems(void* Dest, const SourceElementType* Source, SizeType Count = 1);
|
||||||
|
|
||||||
|
template <typename ElementType, typename SizeType>
|
||||||
|
FORCEINLINE void MoveConstructItems(void* Dest, const ElementType* Source, SizeType Count = 1);
|
||||||
|
|
||||||
|
template <typename ElementType, typename SizeType>
|
||||||
|
FORCEINLINE void MoveAssignItems(ElementType* Dest, const ElementType* Source, SizeType Count = 1);
|
||||||
|
|
||||||
|
template <typename ElementType, typename SizeType>
|
||||||
|
FORCEINLINE bool CompareItems(const ElementType* A, const ElementType* B, SizeType Count = 1);
|
||||||
|
|
||||||
|
NS_END(Memory)
|
||||||
|
NS_REDCRAFT_END
|
||||||
|
|
||||||
|
#include "Templates/MemoryOps.inl"
|
186
Redcraft.Core/Source/Public/Templates/MemoryOps.inl
Normal file
186
Redcraft.Core/Source/Public/Templates/MemoryOps.inl
Normal file
@ -0,0 +1,186 @@
|
|||||||
|
#include "HAL/Memory.h"
|
||||||
|
#include "Templates/TypeTraits.h"
|
||||||
|
#include "MemoryOps.h"
|
||||||
|
|
||||||
|
NS_REDCRAFT_BEGIN
|
||||||
|
NS_BEGIN(Memory)
|
||||||
|
|
||||||
|
NS_PRIVATE_BEGIN
|
||||||
|
|
||||||
|
template <typename DestinationElementType, typename SourceElementType>
|
||||||
|
struct TCanBitwiseRelocate
|
||||||
|
{
|
||||||
|
enum
|
||||||
|
{
|
||||||
|
Value =
|
||||||
|
TypeTraits::TOr<
|
||||||
|
TypeTraits::TIsSame<DestinationElementType, SourceElementType>,
|
||||||
|
TypeTraits::TAnd<
|
||||||
|
TypeTraits::TIsBitwiseConstructible<DestinationElementType, SourceElementType>,
|
||||||
|
TypeTraits::TIsTriviallyDestructible<SourceElementType>
|
||||||
|
>
|
||||||
|
>::Value
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
NS_PRIVATE_END
|
||||||
|
|
||||||
|
template<typename ElementType, typename SizeType>
|
||||||
|
FORCEINLINE void DefaultConstructItems(void* Address, SizeType Count)
|
||||||
|
{
|
||||||
|
if constexpr (TypeTraits::TIsZeroConstructType<ElementType>::Value)
|
||||||
|
{
|
||||||
|
Memory::Memset(Address, 0, sizeof(ElementType) * Count);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
ElementType* Element = (ElementType*)Address;
|
||||||
|
while (Count)
|
||||||
|
{
|
||||||
|
new (Element) ElementType;
|
||||||
|
++Element;
|
||||||
|
--Count;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
template<typename ElementType, typename SizeType>
|
||||||
|
FORCEINLINE void DestructItems(ElementType* Element, SizeType Count)
|
||||||
|
{
|
||||||
|
if constexpr (!TypeTraits::TIsTriviallyDestructible<ElementType>::Value)
|
||||||
|
{
|
||||||
|
while (Count)
|
||||||
|
{
|
||||||
|
typedef ElementType DestructItemsElementTypeTypedef;
|
||||||
|
|
||||||
|
Element->DestructItemsElementTypeTypedef::~DestructItemsElementTypeTypedef();
|
||||||
|
++Element;
|
||||||
|
--Count;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
template<typename DestinationElementType, typename SourceElementType, typename SizeType>
|
||||||
|
FORCEINLINE void ConstructItems(void* Dest, const SourceElementType* Source, SizeType Count)
|
||||||
|
{
|
||||||
|
if constexpr (TypeTraits::TIsBitwiseConstructible<DestinationElementType, SourceElementType>::Value)
|
||||||
|
{
|
||||||
|
Memory::Memcpy(Dest, Source, sizeof(SourceElementType) * Count);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
while (Count)
|
||||||
|
{
|
||||||
|
new (Dest) DestinationElementType(*Source);
|
||||||
|
++(DestinationElementType*&)Dest;
|
||||||
|
++Source;
|
||||||
|
--Count;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
template<typename ElementType, typename SizeType>
|
||||||
|
FORCEINLINE void CopyAssignItems(ElementType* Dest, const ElementType* Source, SizeType Count)
|
||||||
|
{
|
||||||
|
if constexpr (TypeTraits::TIsTriviallyCopyAssignable<ElementType>::Value)
|
||||||
|
{
|
||||||
|
Memory::Memcpy(Dest, Source, sizeof(ElementType) * Count);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
while (Count)
|
||||||
|
{
|
||||||
|
*Dest = *Source;
|
||||||
|
++Dest;
|
||||||
|
++Source;
|
||||||
|
--Count;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
template<typename DestinationElementType, typename SourceElementType, typename SizeType>
|
||||||
|
FORCEINLINE void RelocateConstructItems(void* Dest, const SourceElementType* Source, SizeType Count)
|
||||||
|
{
|
||||||
|
if constexpr (NS_PRIVATE::TCanBitwiseRelocate<DestinationElementType, SourceElementType>::Value)
|
||||||
|
{
|
||||||
|
Memory::Memmove(Dest, Source, sizeof(SourceElementType) * Count);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
while (Count)
|
||||||
|
{
|
||||||
|
typedef SourceElementType RelocateConstructItemsElementTypeTypedef;
|
||||||
|
|
||||||
|
new (Dest) DestinationElementType(*Source);
|
||||||
|
++(DestinationElementType*&)Dest;
|
||||||
|
(Source++)->RelocateConstructItemsElementTypeTypedef::~RelocateConstructItemsElementTypeTypedef();
|
||||||
|
--Count;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
template<typename ElementType, typename SizeType>
|
||||||
|
FORCEINLINE void MoveConstructItems(void* Dest, const ElementType* Source, SizeType Count)
|
||||||
|
{
|
||||||
|
if constexpr (TypeTraits::TIsTriviallyCopyConstructible<ElementType>::Value)
|
||||||
|
{
|
||||||
|
Memory::Memmove(Dest, Source, sizeof(ElementType) * Count);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
while (Count)
|
||||||
|
{
|
||||||
|
new (Dest) ElementType((ElementType&&)*Source);
|
||||||
|
++(ElementType*&)Dest;
|
||||||
|
++Source;
|
||||||
|
--Count;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
template<typename ElementType, typename SizeType>
|
||||||
|
FORCEINLINE void MoveAssignItems(ElementType* Dest, const ElementType* Source, SizeType Count)
|
||||||
|
{
|
||||||
|
if constexpr (TypeTraits::TIsTriviallyCopyAssignable<ElementType>::Value)
|
||||||
|
{
|
||||||
|
Memory::Memmove(Dest, Source, sizeof(ElementType) * Count);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
while (Count)
|
||||||
|
{
|
||||||
|
*Dest = (ElementType&&)*Source;
|
||||||
|
++Dest;
|
||||||
|
++Source;
|
||||||
|
--Count;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
template<typename ElementType, typename SizeType>
|
||||||
|
FORCEINLINE bool CompareItems(const ElementType* A, const ElementType* B, SizeType Count)
|
||||||
|
{
|
||||||
|
if constexpr (TypeTraits::TCanBitwiseCompare<ElementType>::Value)
|
||||||
|
{
|
||||||
|
return !Memory::Memcmp(A, B, sizeof(ElementType) * Count);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
while (Count)
|
||||||
|
{
|
||||||
|
if (!(*A == *B))
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
++A;
|
||||||
|
++B;
|
||||||
|
--Count;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
NS_END(Memory)
|
||||||
|
NS_REDCRAFT_END
|
@ -7,7 +7,7 @@
|
|||||||
NS_REDCRAFT_BEGIN
|
NS_REDCRAFT_BEGIN
|
||||||
NS_BEGIN(TypeTraits)
|
NS_BEGIN(TypeTraits)
|
||||||
|
|
||||||
// Primary type categories
|
// Primary type categories.
|
||||||
|
|
||||||
template <typename T> struct TIsVoid { static constexpr bool Value = std::is_void_v<T>; };
|
template <typename T> struct TIsVoid { static constexpr bool Value = std::is_void_v<T>; };
|
||||||
template <typename T> struct TIsNullPointer { static constexpr bool Value = std::is_null_pointer_v<T>; };
|
template <typename T> struct TIsNullPointer { static constexpr bool Value = std::is_null_pointer_v<T>; };
|
||||||
@ -24,7 +24,7 @@ template <typename T> struct TIsRValueReference { static constexpr bool Value
|
|||||||
template <typename T> struct TIsMemberObjectPointer { static constexpr bool Value = std::is_member_object_pointer_v<T>; };
|
template <typename T> struct TIsMemberObjectPointer { static constexpr bool Value = std::is_member_object_pointer_v<T>; };
|
||||||
template <typename T> struct TIsMemberFunctionPointer { static constexpr bool Value = std::is_member_function_pointer_v<T>; };
|
template <typename T> struct TIsMemberFunctionPointer { static constexpr bool Value = std::is_member_function_pointer_v<T>; };
|
||||||
|
|
||||||
// Composite type categories
|
// Composite type categories.
|
||||||
|
|
||||||
template <typename T> struct TIsFundamental { static constexpr bool Value = std::is_fundamental_v<T>; };
|
template <typename T> struct TIsFundamental { static constexpr bool Value = std::is_fundamental_v<T>; };
|
||||||
template <typename T> struct TIsArithmetic { static constexpr bool Value = std::is_arithmetic_v<T>; };
|
template <typename T> struct TIsArithmetic { static constexpr bool Value = std::is_arithmetic_v<T>; };
|
||||||
@ -34,7 +34,7 @@ template <typename T> struct TIsCompound { static constexpr bool Value = st
|
|||||||
template <typename T> struct TIsReference { static constexpr bool Value = std::is_reference_v<T>; };
|
template <typename T> struct TIsReference { static constexpr bool Value = std::is_reference_v<T>; };
|
||||||
template <typename T> struct TIsMemberPointer { static constexpr bool Value = std::is_member_pointer_v<T>; };
|
template <typename T> struct TIsMemberPointer { static constexpr bool Value = std::is_member_pointer_v<T>; };
|
||||||
|
|
||||||
// Type properties
|
// Type properties.
|
||||||
|
|
||||||
template <typename T> struct TIsConst { static constexpr bool Value = std::is_const_v<T>; };
|
template <typename T> struct TIsConst { static constexpr bool Value = std::is_const_v<T>; };
|
||||||
template <typename T> struct TIsVolatile { static constexpr bool Value = std::is_volatile_v<T>; };
|
template <typename T> struct TIsVolatile { static constexpr bool Value = std::is_volatile_v<T>; };
|
||||||
@ -52,7 +52,7 @@ template <typename T> struct TIsUnsigned { static constexpr
|
|||||||
template <typename T> struct TIsBoundedArray { static constexpr bool Value = std::is_bounded_array_v<T>; };
|
template <typename T> struct TIsBoundedArray { static constexpr bool Value = std::is_bounded_array_v<T>; };
|
||||||
template <typename T> struct TIsUnboundedArray { static constexpr bool Value = std::is_unbounded_array_v<T>; };
|
template <typename T> struct TIsUnboundedArray { static constexpr bool Value = std::is_unbounded_array_v<T>; };
|
||||||
|
|
||||||
// Supported operations
|
// Supported operations.
|
||||||
|
|
||||||
template <typename T> struct TIsConstructible { static constexpr bool Value = std::is_constructible_v<T>; };
|
template <typename T> struct TIsConstructible { static constexpr bool Value = std::is_constructible_v<T>; };
|
||||||
template <typename T> struct TIsTriviallyConstructible { static constexpr bool Value = std::is_trivially_constructible_v<T>; };
|
template <typename T> struct TIsTriviallyConstructible { static constexpr bool Value = std::is_trivially_constructible_v<T>; };
|
||||||
@ -74,12 +74,12 @@ template <typename T> struct THasVirtualDestructor { static constexpr bool V
|
|||||||
template <typename T> struct TIsSwappableWith { static constexpr bool Value = std::is_swappable_with_v<T>; };
|
template <typename T> struct TIsSwappableWith { static constexpr bool Value = std::is_swappable_with_v<T>; };
|
||||||
template <typename T> struct TIsSwappable { static constexpr bool Value = std::is_swappable_v<T>; };
|
template <typename T> struct TIsSwappable { static constexpr bool Value = std::is_swappable_v<T>; };
|
||||||
|
|
||||||
// Property queries
|
// Property queries.
|
||||||
|
|
||||||
template <typename T> struct TRank { static constexpr size_t Value = std::rank_v<T>; };
|
template <typename T> struct TRank { static constexpr size_t Value = std::rank_v<T>; };
|
||||||
template <typename T, size_t N = 0> struct TExtent { static constexpr size_t Value = std::extent_v<T, N>; };
|
template <typename T, size_t N = 0> struct TExtent { static constexpr size_t Value = std::extent_v<T, N>; };
|
||||||
|
|
||||||
// Type relationships
|
// Type relationships.
|
||||||
|
|
||||||
template <typename T, typename U> struct TIsSame { static constexpr bool Value = std::is_same_v<T, U>; };
|
template <typename T, typename U> struct TIsSame { static constexpr bool Value = std::is_same_v<T, U>; };
|
||||||
template <typename T, typename U> struct TIsBaseOf { static constexpr bool Value = std::is_base_of_v<T, U>; };
|
template <typename T, typename U> struct TIsBaseOf { static constexpr bool Value = std::is_base_of_v<T, U>; };
|
||||||
@ -87,7 +87,7 @@ template <typename T, typename U> struct TIsConver
|
|||||||
template <typename Func, typename... ArgTypes> struct TIsInvocable { static constexpr bool Value = std::is_invocable_v<Func, ArgTypes...>; };
|
template <typename Func, typename... ArgTypes> struct TIsInvocable { static constexpr bool Value = std::is_invocable_v<Func, ArgTypes...>; };
|
||||||
template <typename Result, typename Func, typename... ArgTypes> struct TIsInvocableResult { static constexpr bool Value = std::is_invocable_r_v<Result, Func, ArgTypes...>; };
|
template <typename Result, typename Func, typename... ArgTypes> struct TIsInvocableResult { static constexpr bool Value = std::is_invocable_r_v<Result, Func, ArgTypes...>; };
|
||||||
|
|
||||||
// Const-volatility specifiers
|
// Const-volatility specifiers.
|
||||||
|
|
||||||
template <typename T> struct TRemoveCV { typedef typename std::remove_cv_t<T> Type; };
|
template <typename T> struct TRemoveCV { typedef typename std::remove_cv_t<T> Type; };
|
||||||
template <typename T> struct TRemoveConst { typedef typename std::remove_const<T> Type; };
|
template <typename T> struct TRemoveConst { typedef typename std::remove_const<T> Type; };
|
||||||
@ -96,28 +96,28 @@ template <typename T> struct TAddCV { typedef typename std::add_cv<T>
|
|||||||
template <typename T> struct TAddConst { typedef typename std::add_const<T> Type; };
|
template <typename T> struct TAddConst { typedef typename std::add_const<T> Type; };
|
||||||
template <typename T> struct TAddVolatile { typedef typename std::add_volatile<T> Type; };
|
template <typename T> struct TAddVolatile { typedef typename std::add_volatile<T> Type; };
|
||||||
|
|
||||||
// References
|
// References.
|
||||||
|
|
||||||
template <typename T> struct TRemoveReference { typedef typename std::remove_reference_t<T> Type; };
|
template <typename T> struct TRemoveReference { typedef typename std::remove_reference_t<T> Type; };
|
||||||
template <typename T> struct TAddLValueReference { typedef typename std::add_lvalue_reference_t<T> Type; };
|
template <typename T> struct TAddLValueReference { typedef typename std::add_lvalue_reference_t<T> Type; };
|
||||||
template <typename T> struct TAddRValueReference { typedef typename std::add_rvalue_reference_t<T> Type; };
|
template <typename T> struct TAddRValueReference { typedef typename std::add_rvalue_reference_t<T> Type; };
|
||||||
|
|
||||||
// Pointers
|
// Pointers.
|
||||||
|
|
||||||
template <typename T> struct TRemovePointer { typedef typename std::remove_pointer_t<T> Type; };
|
template <typename T> struct TRemovePointer { typedef typename std::remove_pointer_t<T> Type; };
|
||||||
template <typename T> struct TAddPointer { typedef typename std::add_pointer_t<T> Type; };
|
template <typename T> struct TAddPointer { typedef typename std::add_pointer_t<T> Type; };
|
||||||
|
|
||||||
// Sign modifiers
|
// Sign modifiers.
|
||||||
|
|
||||||
template <typename T> struct TMakeSigned { typedef typename std::make_signed_t<T> Type; };
|
template <typename T> struct TMakeSigned { typedef typename std::make_signed_t<T> Type; };
|
||||||
template <typename T> struct TMakeUnsigned { typedef typename std::make_unsigned_t<T> Type; };
|
template <typename T> struct TMakeUnsigned { typedef typename std::make_unsigned_t<T> Type; };
|
||||||
|
|
||||||
// Arrays
|
// Arrays.
|
||||||
|
|
||||||
template <typename T> struct TRemoveExtent { typedef typename std::remove_extent_t<T> Type; };
|
template <typename T> struct TRemoveExtent { typedef typename std::remove_extent_t<T> Type; };
|
||||||
template <typename T> struct TRemoveAllExtents { typedef typename std::remove_all_extents_t<T> Type; };
|
template <typename T> struct TRemoveAllExtents { typedef typename std::remove_all_extents_t<T> Type; };
|
||||||
|
|
||||||
// Miscellaneous transformations
|
// Miscellaneous transformations.
|
||||||
|
|
||||||
template <size_t Len, size_t Alignment> struct TAlignedStorage { typedef typename std::aligned_storage_t<Len, Alignment> Type; };
|
template <size_t Len, size_t Alignment> struct TAlignedStorage { typedef typename std::aligned_storage_t<Len, Alignment> Type; };
|
||||||
template <size_t Len, typename... Types> struct TAlignedUnion { typedef typename std::aligned_union_t<Len, Types...> Type; };
|
template <size_t Len, typename... Types> struct TAlignedUnion { typedef typename std::aligned_union_t<Len, Types...> Type; };
|
||||||
@ -128,22 +128,75 @@ template <typename... T> struct TCommonType { typedef
|
|||||||
template <typename T> struct TUnderlyingType { typedef typename std::underlying_type_t<T> Type; };
|
template <typename T> struct TUnderlyingType { typedef typename std::underlying_type_t<T> Type; };
|
||||||
template <typename Func, typename... ArgTypes> struct TInvokeResult { typedef typename std::invoke_result_t<Func, ArgTypes...> Type; };
|
template <typename Func, typename... ArgTypes> struct TInvokeResult { typedef typename std::invoke_result_t<Func, ArgTypes...> Type; };
|
||||||
|
|
||||||
// Operations on traits
|
// Operations on traits.
|
||||||
|
|
||||||
template <typename T, T InValue> struct TConstant { static constexpr T Value = InValue; };
|
template <typename T, T InValue> struct TConstant { static constexpr T Value = InValue; };
|
||||||
template <bool InValue> struct TBoolConstant : TConstant<bool, InValue> { };
|
template <bool InValue> struct TBoolConstant : TConstant<bool, InValue> { };
|
||||||
|
|
||||||
template <typename... Types> struct TAnd;
|
template <typename... Types> struct TAnd;
|
||||||
template <typename... RHS> struct TAnd<TBoolConstant<true>, RHS...> { static constexpr bool Value = TAnd<RHS...>::Value; };
|
template <typename LHS, typename... RHS> struct TAnd<LHS, RHS...> { static constexpr bool Value = LHS::Value && TAnd<RHS...>::Value; };
|
||||||
template <typename... RHS> struct TAnd<TBoolConstant<false>, RHS...> { static constexpr bool Value = false; };
|
|
||||||
template <> struct TAnd<> { static constexpr bool Value = true; };
|
template <> struct TAnd<> { static constexpr bool Value = true; };
|
||||||
|
|
||||||
template <typename... Types> struct TOr;
|
template <typename... Types> struct TOr;
|
||||||
template <typename... RHS> struct TOr<TBoolConstant<true>, RHS...> { static constexpr bool Value = true; };
|
template <typename LHS, typename... RHS> struct TOr<LHS, RHS...> { static constexpr bool Value = LHS::Value || TOr<RHS...>::Value; };
|
||||||
template <typename... RHS> struct TOr<TBoolConstant<false>, RHS...> { static constexpr bool Value = TOr<RHS...>::Value; };
|
|
||||||
template <> struct TOr<> { static constexpr bool Value = false; };
|
template <> struct TOr<> { static constexpr bool Value = false; };
|
||||||
|
|
||||||
template <typename Type> struct TNot { static constexpr bool Value = !Type::Value; };
|
template <typename Type> struct TNot { static constexpr bool Value = !Type::Value; };
|
||||||
|
|
||||||
|
// Non-STD feature.
|
||||||
|
|
||||||
|
template <typename T>
|
||||||
|
struct TIsZeroConstructType
|
||||||
|
{
|
||||||
|
static constexpr bool Value = TOr<TIsEnum<T>, TIsArithmetic<T>, TIsPointer<T>>::Value;
|
||||||
|
};
|
||||||
|
|
||||||
|
template<typename T>
|
||||||
|
struct TCanBitwiseCompare
|
||||||
|
{
|
||||||
|
static constexpr bool Value = TOr<TIsEnum<T>, TIsArithmetic<T>, TIsPointer<T>>::Value;
|
||||||
|
};
|
||||||
|
|
||||||
|
template <typename T, typename U>
|
||||||
|
struct TIsBitwiseConstructible
|
||||||
|
{
|
||||||
|
static_assert(
|
||||||
|
!TIsReference<T>::Value &&
|
||||||
|
!TIsReference<U>::Value,
|
||||||
|
"TIsBitwiseConstructible is not designed to accept reference types");
|
||||||
|
|
||||||
|
static_assert(
|
||||||
|
TIsSame<T, typename TRemoveCV<T>::Type>::Value &&
|
||||||
|
TIsSame<U, typename TRemoveCV<U>::Type>::Value,
|
||||||
|
"TIsBitwiseConstructible is not designed to accept qualified types");
|
||||||
|
|
||||||
|
static constexpr bool Value = false;
|
||||||
|
};
|
||||||
|
|
||||||
|
template <typename T>
|
||||||
|
struct TIsBitwiseConstructible<T, T>
|
||||||
|
{
|
||||||
|
static constexpr bool Value = TIsTriviallyCopyConstructible<T>::Value;
|
||||||
|
};
|
||||||
|
|
||||||
|
template <typename T, typename U>
|
||||||
|
struct TIsBitwiseConstructible<const T, U> : TIsBitwiseConstructible<T, U>
|
||||||
|
{ };
|
||||||
|
|
||||||
|
template <typename T>
|
||||||
|
struct TIsBitwiseConstructible<const T*, T*>
|
||||||
|
{
|
||||||
|
static constexpr bool Value = true;
|
||||||
|
};
|
||||||
|
|
||||||
|
template <> struct TIsBitwiseConstructible< uint8, int8> { static constexpr bool Value = true; };
|
||||||
|
template <> struct TIsBitwiseConstructible< int8, uint8> { static constexpr bool Value = true; };
|
||||||
|
template <> struct TIsBitwiseConstructible<uint16, int16> { static constexpr bool Value = true; };
|
||||||
|
template <> struct TIsBitwiseConstructible< int16, uint16> { static constexpr bool Value = true; };
|
||||||
|
template <> struct TIsBitwiseConstructible<uint32, int32> { static constexpr bool Value = true; };
|
||||||
|
template <> struct TIsBitwiseConstructible< int32, uint32> { static constexpr bool Value = true; };
|
||||||
|
template <> struct TIsBitwiseConstructible<uint64, int64> { static constexpr bool Value = true; };
|
||||||
|
template <> struct TIsBitwiseConstructible< int64, uint64> { static constexpr bool Value = true; };
|
||||||
|
|
||||||
NS_END(TypeTraits)
|
NS_END(TypeTraits)
|
||||||
NS_REDCRAFT_END
|
NS_REDCRAFT_END
|
||||||
|
@ -8,9 +8,52 @@
|
|||||||
NS_STD_USING
|
NS_STD_USING
|
||||||
NS_REDCRAFT_USING
|
NS_REDCRAFT_USING
|
||||||
|
|
||||||
|
struct FTest
|
||||||
|
{
|
||||||
|
FTest() { cout << "FTest()" << endl; }
|
||||||
|
~FTest() { cout << "~FTest()" << endl; }
|
||||||
|
FTest(int32) { cout << "FTest(int32)" << endl; }
|
||||||
|
FTest(const FTest&) { cout << "FTest(const FTest&)" << endl; }
|
||||||
|
FTest(FTest&&) { cout << "FTest(FTest&&)" << endl; }
|
||||||
|
FTest& operator =(const FTest&) { cout << "FTest& operator =(const FTest&)" << endl; return *this; }
|
||||||
|
FTest& operator =(FTest&&) { cout << "FTest& operator =(FTest&&)" << endl; return *this; }
|
||||||
|
friend bool operator ==(const FTest&, const FTest&) { cout << "bool operator ==(const FTest&, const FTest&)" << endl; return true; }
|
||||||
|
};
|
||||||
|
|
||||||
int main()
|
int main()
|
||||||
{
|
{
|
||||||
check_no_entry();
|
FTest* A = new FTest[2];
|
||||||
|
FTest* B = new FTest[2];
|
||||||
|
int32* C = new int32[2];
|
||||||
|
int32* D = new int32[2];
|
||||||
|
|
||||||
|
cout << " --- " << endl;
|
||||||
|
|
||||||
|
Memory::DefaultConstructItems<FTest>(A, 2);
|
||||||
|
Memory::DestructItems<FTest>(A, 2);
|
||||||
|
Memory::ConstructItems<FTest>(A, C, 2);
|
||||||
|
Memory::CopyAssignItems(B, A, 2);
|
||||||
|
Memory::RelocateConstructItems<FTest>(A, C, 2);
|
||||||
|
Memory::MoveConstructItems(B, A, 2);
|
||||||
|
Memory::MoveAssignItems(B, A, 2);
|
||||||
|
cout << (Memory::CompareItems(A, B, 2) ? "True" : "False") << endl;
|
||||||
|
|
||||||
|
Memory::DefaultConstructItems<int32>(C, 2);
|
||||||
|
Memory::DestructItems<int32>(C, 2);
|
||||||
|
Memory::ConstructItems<int32>(C, D, 2);
|
||||||
|
Memory::CopyAssignItems(D, C, 2);
|
||||||
|
Memory::RelocateConstructItems<int32>(D, C, 2);
|
||||||
|
Memory::MoveConstructItems(D, C, 2);
|
||||||
|
Memory::MoveAssignItems(D, C, 2);
|
||||||
|
cout << (Memory::CompareItems(C, D, 2) ? "True" : "False") << endl;
|
||||||
|
|
||||||
|
cout << " --- " << endl;
|
||||||
|
|
||||||
|
delete[] A;
|
||||||
|
delete[] B;
|
||||||
|
delete[] C;
|
||||||
|
delete[] D;
|
||||||
|
|
||||||
cout << "Done!" << endl;
|
cout << "Done!" << endl;
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user