2022-03-19 16:05:47 +08:00
|
|
|
#pragma once
|
|
|
|
|
|
|
|
#include "CoreTypes.h"
|
2022-03-19 23:07:04 +08:00
|
|
|
#include "Memory/Memory.h"
|
2022-03-19 16:05:47 +08:00
|
|
|
#include "Templates/Utility.h"
|
|
|
|
#include "Concepts/Comparable.h"
|
|
|
|
#include "TypeTraits/TypeTraits.h"
|
|
|
|
|
|
|
|
NAMESPACE_REDCRAFT_BEGIN
|
|
|
|
NAMESPACE_MODULE_BEGIN(Redcraft)
|
|
|
|
NAMESPACE_MODULE_BEGIN(Utility)
|
|
|
|
|
|
|
|
NAMESPACE_BEGIN(Memory)
|
|
|
|
|
2022-04-30 21:33:18 +08:00
|
|
|
template <typename ElementType>
|
2022-05-12 23:36:32 +08:00
|
|
|
requires TIsDefaultConstructible<ElementType>::Value
|
2022-04-30 21:33:18 +08:00
|
|
|
FORCEINLINE void DefaultConstruct(ElementType* Address, size_t Count = 1)
|
2022-03-19 16:05:47 +08:00
|
|
|
{
|
|
|
|
if constexpr (TIsZeroConstructible<ElementType>::Value)
|
|
|
|
{
|
|
|
|
Memory::Memset(Address, 0, sizeof(ElementType) * Count);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ElementType* Element = (ElementType*)Address;
|
|
|
|
while (Count)
|
|
|
|
{
|
|
|
|
new (Element) ElementType;
|
|
|
|
++Element;
|
|
|
|
--Count;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-04-30 21:33:18 +08:00
|
|
|
template <typename DestinationElementType, typename SourceElementType>
|
2022-05-12 23:36:32 +08:00
|
|
|
requires TIsConstructible<DestinationElementType, const SourceElementType&>::Value
|
2022-04-30 21:33:18 +08:00
|
|
|
FORCEINLINE void Construct(DestinationElementType* Destination, const SourceElementType* Source, size_t Count = 1)
|
2022-03-19 16:05:47 +08:00
|
|
|
{
|
|
|
|
if constexpr (TIsBitwiseConstructible<DestinationElementType, const SourceElementType>::Value)
|
|
|
|
{
|
|
|
|
Memory::Memcpy(Destination, Source, sizeof(SourceElementType) * Count);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
while (Count)
|
|
|
|
{
|
|
|
|
new (Destination) DestinationElementType(*Source);
|
|
|
|
++(DestinationElementType*&)Destination;
|
|
|
|
++Source;
|
|
|
|
--Count;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-04-30 21:33:18 +08:00
|
|
|
template <typename ElementType>
|
2022-05-12 23:36:32 +08:00
|
|
|
requires TIsCopyConstructible<ElementType>::Value
|
2022-04-30 21:33:18 +08:00
|
|
|
FORCEINLINE void CopyConstruct(ElementType* Destination, const ElementType* Source, size_t Count = 1)
|
|
|
|
{
|
|
|
|
if constexpr (TIsTriviallyCopyConstructible<ElementType>::Value)
|
|
|
|
{
|
|
|
|
Memory::Memcpy(Destination, Source, sizeof(ElementType) * Count);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
while (Count)
|
|
|
|
{
|
|
|
|
new (Destination) ElementType(*Source);
|
|
|
|
++(ElementType*&)Destination;
|
|
|
|
++Source;
|
|
|
|
--Count;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename ElementType>
|
2022-05-12 23:36:32 +08:00
|
|
|
requires TIsMoveConstructible<ElementType>::Value
|
2022-04-30 21:33:18 +08:00
|
|
|
FORCEINLINE void MoveConstruct(ElementType* Destination, ElementType* Source, size_t Count = 1)
|
2022-03-19 16:05:47 +08:00
|
|
|
{
|
|
|
|
if constexpr (TIsTriviallyMoveConstructible<ElementType>::Value)
|
|
|
|
{
|
|
|
|
Memory::Memmove(Destination, Source, sizeof(ElementType) * Count);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
while (Count)
|
|
|
|
{
|
|
|
|
new (Destination) ElementType(MoveTemp(*Source));
|
|
|
|
++(ElementType*&)Destination;
|
|
|
|
++Source;
|
|
|
|
--Count;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-04-30 21:33:18 +08:00
|
|
|
template <typename DestinationElementType, typename SourceElementType>
|
2022-05-12 23:36:32 +08:00
|
|
|
requires TIsConstructible<DestinationElementType, SourceElementType&&>::Value && TIsDestructible<SourceElementType>::Value
|
2022-04-30 21:33:18 +08:00
|
|
|
FORCEINLINE void RelocateConstruct(DestinationElementType* Destination, SourceElementType* Source, size_t Count = 1)
|
2022-03-19 16:05:47 +08:00
|
|
|
{
|
|
|
|
if constexpr (TIsBitwiseRelocatable<DestinationElementType, SourceElementType>::Value)
|
|
|
|
{
|
|
|
|
Memory::Memmove(Destination, Source, sizeof(SourceElementType) * Count);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
while (Count)
|
|
|
|
{
|
|
|
|
typedef SourceElementType RelocateConstructItemsElementTypeTypedef;
|
|
|
|
|
|
|
|
new (Destination) DestinationElementType(MoveTemp(*Source));
|
|
|
|
++(DestinationElementType*&)Destination;
|
|
|
|
(Source++)->RelocateConstructItemsElementTypeTypedef::~RelocateConstructItemsElementTypeTypedef();
|
|
|
|
--Count;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-04-30 21:33:18 +08:00
|
|
|
template <typename ElementType>
|
2022-05-12 23:36:32 +08:00
|
|
|
requires TIsDestructible<ElementType>::Value
|
2022-04-30 21:33:18 +08:00
|
|
|
FORCEINLINE void Destruct(ElementType* Element, size_t Count = 1)
|
2022-03-19 16:05:47 +08:00
|
|
|
{
|
|
|
|
if constexpr (!TIsTriviallyDestructible<ElementType>::Value)
|
|
|
|
{
|
|
|
|
while (Count)
|
|
|
|
{
|
|
|
|
typedef ElementType DestructItemsElementTypeTypedef;
|
|
|
|
|
|
|
|
Element->DestructItemsElementTypeTypedef::~DestructItemsElementTypeTypedef();
|
|
|
|
++Element;
|
|
|
|
--Count;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-04-30 21:33:18 +08:00
|
|
|
template <typename ElementType>
|
2022-05-12 23:36:32 +08:00
|
|
|
requires TIsCopyAssignable<ElementType>::Value
|
2022-04-30 21:33:18 +08:00
|
|
|
FORCEINLINE void CopyAssign(ElementType* Destination, const ElementType* Source, size_t Count = 1)
|
2022-03-19 16:05:47 +08:00
|
|
|
{
|
|
|
|
if constexpr (TIsTriviallyCopyAssignable<ElementType>::Value)
|
|
|
|
{
|
|
|
|
Memory::Memcpy(Destination, Source, sizeof(ElementType) * Count);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
while (Count)
|
|
|
|
{
|
|
|
|
*Destination = *Source;
|
|
|
|
++Destination;
|
|
|
|
++Source;
|
|
|
|
--Count;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-04-30 21:33:18 +08:00
|
|
|
template <typename ElementType>
|
2022-05-12 23:36:32 +08:00
|
|
|
requires TIsMoveAssignable<ElementType>::Value
|
2022-04-30 21:33:18 +08:00
|
|
|
FORCEINLINE void MoveAssign(ElementType* Destination, ElementType* Source, size_t Count = 1)
|
2022-03-19 16:05:47 +08:00
|
|
|
{
|
|
|
|
if constexpr (TIsTriviallyCopyConstructible<ElementType>::Value)
|
|
|
|
{
|
|
|
|
Memory::Memmove(Destination, Source, sizeof(ElementType) * Count);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
while (Count)
|
|
|
|
{
|
|
|
|
*Destination = MoveTemp(*Source);
|
|
|
|
++(ElementType*&)Destination;
|
|
|
|
++Source;
|
|
|
|
--Count;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-04-30 21:33:18 +08:00
|
|
|
template <typename ElementType>
|
2022-05-12 23:36:32 +08:00
|
|
|
requires CEqualityComparable<ElementType>
|
2022-04-30 21:33:18 +08:00
|
|
|
FORCEINLINE bool Compare(const ElementType* LHS, const ElementType* RHS, size_t Count = 1)
|
2022-03-19 16:05:47 +08:00
|
|
|
{
|
|
|
|
if constexpr (TIsBitwiseComparable<ElementType>::Value)
|
|
|
|
{
|
|
|
|
return !Memory::Memcmp(LHS, RHS, sizeof(ElementType) * Count);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
while (Count)
|
|
|
|
{
|
2022-04-30 21:33:18 +08:00
|
|
|
if (!(*LHS == *RHS)) return false;
|
2022-03-19 16:05:47 +08:00
|
|
|
|
|
|
|
++LHS;
|
|
|
|
++RHS;
|
|
|
|
--Count;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
NAMESPACE_END(Memory)
|
|
|
|
|
|
|
|
NAMESPACE_MODULE_END(Utility)
|
|
|
|
NAMESPACE_MODULE_END(Redcraft)
|
|
|
|
NAMESPACE_REDCRAFT_END
|