diff --git a/Redcraft.Utility/Source/Public/Containers/Array.h b/Redcraft.Utility/Source/Public/Containers/Array.h index a8132d0..26418b0 100644 --- a/Redcraft.Utility/Source/Public/Containers/Array.h +++ b/Redcraft.Utility/Source/Public/Containers/Array.h @@ -16,96 +16,15 @@ NAMESPACE_REDCRAFT_BEGIN NAMESPACE_MODULE_BEGIN(Redcraft) NAMESPACE_MODULE_BEGIN(Utility) -NAMESPACE_PRIVATE_BEGIN - -template -class TArrayIterator -{ -public: - - using ElementType = T; - - FORCEINLINE TArrayIterator() = default; - -# if DO_CHECK - FORCEINLINE TArrayIterator(const TArrayIterator>& InValue) requires (CConst) - : Owner(InValue.Owner), Pointer(InValue.Pointer) - { } -# else - FORCEINLINE TArrayIterator(const TArrayIterator>& InValue) requires (CConst) - : Pointer(InValue.Pointer) - { } -# endif - - FORCEINLINE TArrayIterator(const TArrayIterator&) = default; - FORCEINLINE TArrayIterator(TArrayIterator&&) = default; - FORCEINLINE TArrayIterator& operator=(const TArrayIterator&) = default; - FORCEINLINE TArrayIterator& operator=(TArrayIterator&&) = default; - - NODISCARD friend FORCEINLINE bool operator==(const TArrayIterator& LHS, const TArrayIterator& RHS) { return LHS.Pointer == RHS.Pointer; } - - NODISCARD friend FORCEINLINE strong_ordering operator<=>(const TArrayIterator & LHS, const TArrayIterator & RHS) { return LHS.Pointer <=> RHS.Pointer; } - - NODISCARD FORCEINLINE ElementType& operator*() const { CheckThis(true); return *Pointer; } - NODISCARD FORCEINLINE ElementType* operator->() const { CheckThis(true); return Pointer; } - - NODISCARD FORCEINLINE ElementType& operator[](ptrdiff Index) const { TArrayIterator Temp = *this + Index; return *Temp; } - - FORCEINLINE TArrayIterator& operator++() { ++Pointer; CheckThis(); return *this; } - FORCEINLINE TArrayIterator& operator--() { --Pointer; CheckThis(); return *this; } - - FORCEINLINE TArrayIterator operator++(int) { TArrayIterator Temp = *this; ++*this; return Temp; } - FORCEINLINE TArrayIterator operator--(int) { TArrayIterator Temp = *this; --*this; return Temp; } - - FORCEINLINE TArrayIterator& operator+=(ptrdiff Offset) { Pointer += Offset; CheckThis(); return *this; } - FORCEINLINE TArrayIterator& operator-=(ptrdiff Offset) { Pointer -= Offset; CheckThis(); return *this; } - - NODISCARD friend FORCEINLINE TArrayIterator operator+(TArrayIterator Iter, ptrdiff Offset) { TArrayIterator Temp = Iter; Temp += Offset; return Temp; } - NODISCARD friend FORCEINLINE TArrayIterator operator+(ptrdiff Offset, TArrayIterator Iter) { TArrayIterator Temp = Iter; Temp += Offset; return Temp; } - - NODISCARD FORCEINLINE TArrayIterator operator-(ptrdiff Offset) const { TArrayIterator Temp = *this; Temp -= Offset; return Temp; } - - NODISCARD friend FORCEINLINE ptrdiff operator-(const TArrayIterator& LHS, const TArrayIterator& RHS) { LHS.CheckThis(); RHS.CheckThis(); return LHS.Pointer - RHS.Pointer; } - - NODISCARD FORCEINLINE explicit operator TObserverPtr() const { CheckThis(); return TObserverPtr(Pointer); } - -private: - -# if DO_CHECK - const ArrayType* Owner = nullptr; -# endif - - ElementType* Pointer = nullptr; - -# if DO_CHECK - FORCEINLINE TArrayIterator(const ArrayType* InContainer, ElementType* InPointer) - : Owner(InContainer), Pointer(InPointer) - { } -# else - FORCEINLINE TArrayIterator(const ArrayType* InContainer, ElementType* InPointer) - : Pointer(InPointer) - { } -# endif - - FORCEINLINE void CheckThis(bool bExceptEnd = false) const - { - checkf(Owner && Owner->IsValidIterator(*this), TEXT("Read access violation. Please check IsValidIterator().")); - checkf(!(bExceptEnd && Owner->End() == *this), TEXT("Read access violation. Please check IsValidIterator().")); - } - - friend ArrayType; - - template - friend class TArrayIterator; - -}; - -NAMESPACE_PRIVATE_END - /** Dynamic array. The elements are stored contiguously, which means that elements can be accessed not only through iterators, but also using offsets to regular pointers to elements. */ template requires (!CConst) class TArray final { +private: + + template + class IteratorImpl; + public: using ElementType = T; @@ -114,8 +33,8 @@ public: using Reference = T&; using ConstReference = const T&; - using Iterator = NAMESPACE_PRIVATE::TArrayIterator; - using ConstIterator = NAMESPACE_PRIVATE::TArrayIterator; + using Iterator = IteratorImpl; + using ConstIterator = IteratorImpl; using ReverseIterator = TReverseIterator< Iterator>; using ConstReverseIterator = TReverseIterator; @@ -155,7 +74,7 @@ public: { if constexpr (CForwardIterator) { - if (CSizedSentinelFor) checkf(First <= Last, TEXT("Illegal range iterator. Please check First <= Last.")); + if (CSizedSentinelFor) { checkf(First <= Last, TEXT("Illegal range iterator. Please check First <= Last.")); } const size_t Count = Iteration::Distance(First, Last); @@ -615,7 +534,7 @@ public: if constexpr (CForwardIterator) { - if (CSizedSentinelFor) checkf(First <= Last, TEXT("Illegal range iterator. Please check First <= Last.")); + if (CSizedSentinelFor) { checkf(First <= Last, TEXT("Illegal range iterator. Please check First <= Last.")); } const size_t InsertIndex = Iter - Begin(); const size_t Count = Iteration::Distance(First, Last); @@ -1142,6 +1061,89 @@ private: } ALLOCATOR_WRAPPER_END(AllocatorType, ElementType, Impl) +private: + + template + class IteratorImpl + { + public: + + using ElementType = TConditional; + + FORCEINLINE IteratorImpl() = default; + +# if DO_CHECK + FORCEINLINE IteratorImpl(const IteratorImpl& InValue) requires (bConst) + : Owner(InValue.Owner), Pointer(InValue.Pointer) + { } +# else + FORCEINLINE IteratorImpl(const IteratorImpl& InValue) requires (bConst) + : Pointer(InValue.Pointer) + { } +# endif + + FORCEINLINE IteratorImpl(const IteratorImpl&) = default; + FORCEINLINE IteratorImpl(IteratorImpl&&) = default; + FORCEINLINE IteratorImpl& operator=(const IteratorImpl&) = default; + FORCEINLINE IteratorImpl& operator=(IteratorImpl&&) = default; + + NODISCARD friend FORCEINLINE bool operator==(const IteratorImpl& LHS, const IteratorImpl& RHS) { return LHS.Pointer == RHS.Pointer; } + + NODISCARD friend FORCEINLINE strong_ordering operator<=>(const IteratorImpl& LHS, const IteratorImpl& RHS) { return LHS.Pointer <=> RHS.Pointer; } + + NODISCARD FORCEINLINE ElementType& operator*() const { CheckThis(true); return *Pointer; } + NODISCARD FORCEINLINE ElementType* operator->() const { CheckThis(true); return Pointer; } + + NODISCARD FORCEINLINE ElementType& operator[](ptrdiff Index) const { IteratorImpl Temp = *this + Index; return *Temp; } + + FORCEINLINE IteratorImpl& operator++() { ++Pointer; CheckThis(); return *this; } + FORCEINLINE IteratorImpl& operator--() { --Pointer; CheckThis(); return *this; } + + FORCEINLINE IteratorImpl operator++(int) { IteratorImpl Temp = *this; ++*this; return Temp; } + FORCEINLINE IteratorImpl operator--(int) { IteratorImpl Temp = *this; --*this; return Temp; } + + FORCEINLINE IteratorImpl& operator+=(ptrdiff Offset) { Pointer += Offset; CheckThis(); return *this; } + FORCEINLINE IteratorImpl& operator-=(ptrdiff Offset) { Pointer -= Offset; CheckThis(); return *this; } + + NODISCARD friend FORCEINLINE IteratorImpl operator+(IteratorImpl Iter, ptrdiff Offset) { IteratorImpl Temp = Iter; Temp += Offset; return Temp; } + NODISCARD friend FORCEINLINE IteratorImpl operator+(ptrdiff Offset, IteratorImpl Iter) { IteratorImpl Temp = Iter; Temp += Offset; return Temp; } + + NODISCARD FORCEINLINE IteratorImpl operator-(ptrdiff Offset) const { IteratorImpl Temp = *this; Temp -= Offset; return Temp; } + + NODISCARD friend FORCEINLINE ptrdiff operator-(const IteratorImpl& LHS, const IteratorImpl& RHS) { LHS.CheckThis(); RHS.CheckThis(); return LHS.Pointer - RHS.Pointer; } + + NODISCARD FORCEINLINE explicit operator TObserverPtr() const { CheckThis(); return TObserverPtr(Pointer); } + + private: + +# if DO_CHECK + const TArray* Owner = nullptr; +# endif + + ElementType* Pointer = nullptr; + +# if DO_CHECK + FORCEINLINE IteratorImpl(const TArray* InContainer, ElementType* InPointer) + : Owner(InContainer), Pointer(InPointer) + { } +# else + FORCEINLINE IteratorImpl(const TArray* InContainer, ElementType* InPointer) + : Pointer(InPointer) + { } +# endif + + FORCEINLINE void CheckThis(bool bExceptEnd = false) const + { + checkf(Owner && Owner->IsValidIterator(*this), TEXT("Read access violation. Please check IsValidIterator().")); + checkf(!(bExceptEnd && Owner->End() == *this), TEXT("Read access violation. Please check IsValidIterator().")); + } + + template friend class IteratorImpl; + + friend TArray; + + }; + }; template diff --git a/Redcraft.Utility/Source/Public/Containers/ArrayView.h b/Redcraft.Utility/Source/Public/Containers/ArrayView.h index beeb0da..5806511 100644 --- a/Redcraft.Utility/Source/Public/Containers/ArrayView.h +++ b/Redcraft.Utility/Source/Public/Containers/ArrayView.h @@ -14,108 +14,6 @@ NAMESPACE_REDCRAFT_BEGIN NAMESPACE_MODULE_BEGIN(Redcraft) NAMESPACE_MODULE_BEGIN(Utility) -template -class TArrayView; - -NAMESPACE_PRIVATE_BEGIN - -template -class TArrayViewIterator -{ -public: - - using ElementType = T; - - FORCEINLINE constexpr TArrayViewIterator() = default; - -# if DO_CHECK - FORCEINLINE constexpr TArrayViewIterator(const TArrayViewIterator>& InValue) requires (CConst) - : Pointer(InValue.Pointer), FirstSentinel(InValue.FirstSentinel), EndSentinel(InValue.EndSentinel) - { } -# else - FORCEINLINE constexpr TArrayViewIterator(const TArrayViewIterator>& InValue) requires (CConst) - : Pointer(InValue.Pointer) - { } -# endif - - FORCEINLINE constexpr TArrayViewIterator(const TArrayViewIterator&) = default; - FORCEINLINE constexpr TArrayViewIterator(TArrayViewIterator&&) = default; - FORCEINLINE constexpr TArrayViewIterator& operator=(const TArrayViewIterator&) = default; - FORCEINLINE constexpr TArrayViewIterator& operator=(TArrayViewIterator&&) = default; - - NODISCARD friend FORCEINLINE constexpr bool operator==(const TArrayViewIterator& LHS, const TArrayViewIterator& RHS) { return LHS.Pointer == RHS.Pointer; } - - NODISCARD friend FORCEINLINE constexpr strong_ordering operator<=>(const TArrayViewIterator & LHS, const TArrayViewIterator & RHS) { return LHS.Pointer <=> RHS.Pointer; } - - NODISCARD FORCEINLINE constexpr ElementType& operator*() const { CheckThis(true); return *Pointer; } - NODISCARD FORCEINLINE constexpr ElementType* operator->() const { CheckThis(true); return Pointer; } - - NODISCARD FORCEINLINE constexpr ElementType& operator[](ptrdiff Index) const { TArrayViewIterator Temp = *this + Index; return *Temp; } - - FORCEINLINE constexpr TArrayViewIterator& operator++() { ++Pointer; CheckThis(); return *this; } - FORCEINLINE constexpr TArrayViewIterator& operator--() { --Pointer; CheckThis(); return *this; } - - FORCEINLINE constexpr TArrayViewIterator operator++(int) { TArrayViewIterator Temp = *this; ++*this; return Temp; } - FORCEINLINE constexpr TArrayViewIterator operator--(int) { TArrayViewIterator Temp = *this; --*this; return Temp; } - - FORCEINLINE constexpr TArrayViewIterator& operator+=(ptrdiff Offset) { Pointer += Offset; CheckThis(); return *this; } - FORCEINLINE constexpr TArrayViewIterator& operator-=(ptrdiff Offset) { Pointer -= Offset; CheckThis(); return *this; } - - NODISCARD friend FORCEINLINE constexpr TArrayViewIterator operator+(TArrayViewIterator Iter, ptrdiff Offset) { TArrayViewIterator Temp = Iter; Temp += Offset; return Temp; } - NODISCARD friend FORCEINLINE constexpr TArrayViewIterator operator+(ptrdiff Offset, TArrayViewIterator Iter) { TArrayViewIterator Temp = Iter; Temp += Offset; return Temp; } - - NODISCARD FORCEINLINE constexpr TArrayViewIterator operator-(ptrdiff Offset) const { TArrayViewIterator Temp = *this; Temp -= Offset; return Temp; } - - NODISCARD friend FORCEINLINE constexpr ptrdiff operator-(const TArrayViewIterator& LHS, const TArrayViewIterator& RHS) { LHS.CheckThis(); RHS.CheckThis(); return LHS.Pointer - RHS.Pointer; } - - NODISCARD FORCEINLINE constexpr explicit operator TObserverPtr() const { CheckThis(); return TObserverPtr(Pointer); } - -private: - - ElementType* Pointer = nullptr; - -# if DO_CHECK - ElementType* FirstSentinel = nullptr; - ElementType* EndSentinel = nullptr; -# endif - -# if DO_CHECK - FORCEINLINE constexpr TArrayViewIterator(ElementType* InPointer, ElementType* InFirstSentinel, ElementType* InEndSentinel) - : Pointer(InPointer), FirstSentinel(InFirstSentinel), EndSentinel(InEndSentinel) - { } -# else - FORCEINLINE constexpr TArrayViewIterator(ElementType* InPointer, ElementType* InFirstSentinel, ElementType* InEndSentinel) - : Pointer(InPointer) - { } -# endif - - FORCEINLINE constexpr void CheckThis(bool bExceptEnd = false) const - { - check_code - ({ - const bool bInLegalRange = FirstSentinel && EndSentinel && FirstSentinel <= Pointer && Pointer <= EndSentinel; - const bool bIsDereferenceable = Pointer != EndSentinel; - - checkf(bInLegalRange && (!bExceptEnd || bIsDereferenceable), TEXT("Read access violation. Please check IsValidIterator().")); - }); - } - - template - friend class TArrayViewIterator; - - template - friend class NAMESPACE_REDCRAFT::TArrayView; - -}; - -template -struct TEnableArrayNum { size_t ArrayNum; }; - -template <> -struct TEnableArrayNum { size_t ArrayNum; }; - -NAMESPACE_PRIVATE_END - template struct TStaticArray; @@ -130,19 +28,15 @@ inline constexpr size_t DynamicExtent = INDEX_NONE; * is known at compile-time and encoded in the type, or a dynamic extent. */ template -class TArrayView final : private NAMESPACE_PRIVATE::TEnableArrayNum +class TArrayView final { -private: - - using Impl = NAMESPACE_PRIVATE::TEnableArrayNum; - public: using ElementType = T; using Reference = T&; - using Iterator = NAMESPACE_PRIVATE::TArrayViewIterator; + class Iterator; using ReverseIterator = TReverseIterator; @@ -155,35 +49,41 @@ public: /** Constructs an array view that is a view over the range ['InFirst', 'InFirst' + 'Count'). */ template requires (CConvertibleTo(*)[], ElementType(*)[]>) - FORCEINLINE constexpr explicit (Extent != DynamicExtent) TArrayView(I InFirst, size_t InCount) : Pointer(static_cast[]>>(InFirst)) + FORCEINLINE constexpr explicit (Extent != DynamicExtent) TArrayView(I InFirst, size_t InCount) { checkf(Extent == DynamicExtent || Extent == InCount, TEXT("Illegal range count. Please check InCount.")); + Impl.Pointer = AddressOf(*InFirst); + if constexpr (Extent == DynamicExtent) { - Impl::ArrayNum = InCount; + Impl.ArrayNum = InCount; } } /** Constructs an array view that is a view over the range ['InFirst', 'InLast'). */ template S> requires (CConvertibleTo(*)[], ElementType(*)[]>) - FORCEINLINE constexpr explicit (Extent != DynamicExtent) TArrayView(I InFirst, S InLast) : Pointer(static_cast[]>>(InFirst)) + FORCEINLINE constexpr explicit (Extent != DynamicExtent) TArrayView(I InFirst, S InLast) { checkf(Extent == DynamicExtent || Extent == InLast - InFirst, TEXT("Illegal range iterator. Please check InLast - InFirst.")); + Impl.Pointer = AddressOf(*InFirst); + if constexpr (Extent == DynamicExtent) { - Impl::ArrayNum = InLast - InFirst; + Impl.ArrayNum = InLast - InFirst; } } /** Constructs an array view that is a view over the array 'InArray'. */ template requires (Extent == DynamicExtent || N == Extent) - FORCEINLINE constexpr TArrayView(ElementType(&InArray)[N]) : Pointer(InArray) + FORCEINLINE constexpr TArrayView(ElementType(&InArray)[N]) { + Impl.Pointer = AddressOf(InArray[0]); + if constexpr (Extent == DynamicExtent) { - Impl::ArrayNum = N; + Impl.ArrayNum = N; } } @@ -211,13 +111,15 @@ public: /** Converting constructor from another array view 'InValue'. */ template requires ((Extent == DynamicExtent || N == DynamicExtent || N == Extent) && CConvertibleTo) - FORCEINLINE constexpr explicit (Extent != DynamicExtent && N == DynamicExtent) TArrayView(TArrayView InValue) : Pointer(InValue.GetData()) + FORCEINLINE constexpr explicit (Extent != DynamicExtent && N == DynamicExtent) TArrayView(TArrayView InValue) { checkf(Extent == DynamicExtent || Extent == InValue.Num(), TEXT("Illegal view extent. Please check InValue.Num().")); + Impl.Pointer = AddressOf(InValue[0]); + if constexpr (Extent == DynamicExtent) { - Impl::ArrayNum = InValue.Num(); + Impl.ArrayNum = InValue.Num(); } } @@ -365,18 +267,18 @@ public: } /** @return The pointer to the underlying element storage. */ - NODISCARD FORCEINLINE constexpr TObserverPtr GetData() const { return Pointer; } + NODISCARD FORCEINLINE constexpr TObserverPtr GetData() const { return TObserverPtr(Impl.Pointer); } /** @return The iterator to the first or end element. */ - NODISCARD FORCEINLINE constexpr Iterator Begin() const { return Iterator(Pointer.Get() , Pointer.Get(), Pointer.Get() + Num()); } - NODISCARD FORCEINLINE constexpr Iterator End() const { return Iterator(Pointer.Get() + Num(), Pointer.Get(), Pointer.Get() + Num()); } + NODISCARD FORCEINLINE constexpr Iterator Begin() const { return Iterator(this, Impl.Pointer); } + NODISCARD FORCEINLINE constexpr Iterator End() const { return Iterator(this, Impl.Pointer + Num()); } /** @return The reverse iterator to the first or end element. */ NODISCARD FORCEINLINE constexpr ReverseIterator RBegin() const { return ReverseIterator(End()); } NODISCARD FORCEINLINE constexpr ReverseIterator REnd() const { return ReverseIterator(Begin()); } /** @return The number of elements in the container. */ - NODISCARD FORCEINLINE constexpr size_t Num() const { if constexpr (Extent == DynamicExtent) return Impl::ArrayNum; return Extent; } + NODISCARD FORCEINLINE constexpr size_t Num() const { if constexpr (Extent == DynamicExtent) { return Impl.ArrayNum; } return Extent; } /** @return The number of bytes in the container. */ NODISCARD FORCEINLINE constexpr size_t NumBytes() const { return Num() * sizeof(ElementType); } @@ -388,7 +290,7 @@ public: NODISCARD FORCEINLINE constexpr bool IsValidIterator(Iterator Iter) const { return Begin() <= Iter && Iter <= End(); } /** @return The reference to the requested element. */ - NODISCARD FORCEINLINE constexpr ElementType& operator[](size_t Index) const { checkf(Index < Num(), TEXT("Read access violation. Please check IsValidIterator().")); return Pointer[Index]; } + NODISCARD FORCEINLINE constexpr ElementType& operator[](size_t Index) const { checkf(Index < Num(), TEXT("Read access violation. Please check IsValidIterator().")); return Impl.Pointer[Index]; } /** @return The reference to the first or last element. */ NODISCARD FORCEINLINE constexpr ElementType& Front() const { return *Begin(); } @@ -411,7 +313,80 @@ public: private: - TObserverPtr Pointer; + struct FImplWithoutNum { ElementType* Pointer; }; + + struct FImplWithNum : FImplWithoutNum { size_t ArrayNum; }; + + TConditional Impl; + +public: + + class Iterator + { + public: + + using ElementType = T; + + FORCEINLINE constexpr Iterator() = default; + FORCEINLINE constexpr Iterator(const Iterator&) = default; + FORCEINLINE constexpr Iterator(Iterator&&) = default; + FORCEINLINE constexpr Iterator& operator=(const Iterator&) = default; + FORCEINLINE constexpr Iterator& operator=(Iterator&&) = default; + + NODISCARD friend FORCEINLINE constexpr bool operator==(const Iterator& LHS, const Iterator& RHS) { return LHS.Pointer == RHS.Pointer; } + + NODISCARD friend FORCEINLINE constexpr strong_ordering operator<=>(const Iterator& LHS, const Iterator& RHS) { return LHS.Pointer <=> RHS.Pointer; } + + NODISCARD FORCEINLINE constexpr ElementType& operator*() const { CheckThis(true); return *Pointer; } + NODISCARD FORCEINLINE constexpr ElementType* operator->() const { CheckThis(true); return Pointer; } + + NODISCARD FORCEINLINE constexpr ElementType& operator[](ptrdiff Index) const { Iterator Temp = *this + Index; return *Temp; } + + FORCEINLINE constexpr Iterator& operator++() { ++Pointer; CheckThis(); return *this; } + FORCEINLINE constexpr Iterator& operator--() { --Pointer; CheckThis(); return *this; } + + FORCEINLINE constexpr Iterator operator++(int) { Iterator Temp = *this; ++*this; return Temp; } + FORCEINLINE constexpr Iterator operator--(int) { Iterator Temp = *this; --*this; return Temp; } + + FORCEINLINE constexpr Iterator& operator+=(ptrdiff Offset) { Pointer += Offset; CheckThis(); return *this; } + FORCEINLINE constexpr Iterator& operator-=(ptrdiff Offset) { Pointer -= Offset; CheckThis(); return *this; } + + NODISCARD friend FORCEINLINE constexpr Iterator operator+(Iterator Iter, ptrdiff Offset) { Iterator Temp = Iter; Temp += Offset; return Temp; } + NODISCARD friend FORCEINLINE constexpr Iterator operator+(ptrdiff Offset, Iterator Iter) { Iterator Temp = Iter; Temp += Offset; return Temp; } + + NODISCARD FORCEINLINE constexpr Iterator operator-(ptrdiff Offset) const { Iterator Temp = *this; Temp -= Offset; return Temp; } + + NODISCARD friend FORCEINLINE constexpr ptrdiff operator-(const Iterator& LHS, const Iterator& RHS) { LHS.CheckThis(); RHS.CheckThis(); return LHS.Pointer - RHS.Pointer; } + + NODISCARD FORCEINLINE constexpr explicit operator TObserverPtr() const { CheckThis(); return TObserverPtr(Pointer); } + + private: + +# if DO_CHECK + const TArrayView* Owner = nullptr; +# endif + + ElementType* Pointer = nullptr; + +# if DO_CHECK + FORCEINLINE constexpr Iterator(const TArrayView* InContainer, ElementType* InPointer) + : Owner(InContainer), Pointer(InPointer) + { } +# else + FORCEINLINE constexpr Iterator(const TArrayView* InContainer, ElementType* InPointer) + : Pointer(InPointer) + { } +# endif + + FORCEINLINE constexpr void CheckThis(bool bExceptEnd = false) const + { + checkf(Owner && Owner->IsValidIterator(*this), TEXT("Read access violation. Please check IsValidIterator().")); + checkf(!(bExceptEnd && Owner->End() == *this), TEXT("Read access violation. Please check IsValidIterator().")); + } + + friend TArrayView; + + }; }; diff --git a/Redcraft.Utility/Source/Public/Containers/StaticArray.h b/Redcraft.Utility/Source/Public/Containers/StaticArray.h index f7adf18..90daa86 100644 --- a/Redcraft.Utility/Source/Public/Containers/StaticArray.h +++ b/Redcraft.Utility/Source/Public/Containers/StaticArray.h @@ -20,14 +20,20 @@ NAMESPACE_MODULE_BEGIN(Utility) template struct TStaticArray final { +private: + + template + class IteratorImpl; + +public: using ElementType = T; using Reference = T&; using ConstReference = const T&; - using Iterator = TArrayView< T, N>::Iterator; - using ConstIterator = TArrayView::Iterator; + using Iterator = IteratorImpl; + using ConstIterator = IteratorImpl; using ReverseIterator = TReverseIterator< Iterator>; using ConstReverseIterator = TReverseIterator; @@ -87,10 +93,10 @@ struct TStaticArray final NODISCARD FORCEINLINE constexpr TObserverPtr GetData() const { return TObserverPtr(_); } /** @return The iterator to the first or end element. */ - NODISCARD FORCEINLINE constexpr Iterator Begin() { return TArrayView< T, N>(*this).Begin(); } - NODISCARD FORCEINLINE constexpr ConstIterator Begin() const { return TArrayView(*this).Begin(); } - NODISCARD FORCEINLINE constexpr Iterator End() { return TArrayView< T, N>(*this).End(); } - NODISCARD FORCEINLINE constexpr ConstIterator End() const { return TArrayView(*this).End(); } + NODISCARD FORCEINLINE constexpr Iterator Begin() { return Iterator(this, _); } + NODISCARD FORCEINLINE constexpr ConstIterator Begin() const { return ConstIterator(this, _); } + NODISCARD FORCEINLINE constexpr Iterator End() { return Iterator(this, _ + Num()); } + NODISCARD FORCEINLINE constexpr ConstIterator End() const { return ConstIterator(this, _ + Num()); } /** @return The reverse iterator to the first or end element. */ NODISCARD FORCEINLINE constexpr ReverseIterator RBegin() { return ReverseIterator(End()); } @@ -137,6 +143,89 @@ struct TStaticArray final T _[N]; +private: + + template + class IteratorImpl + { + public: + + using ElementType = TConditional; + + FORCEINLINE constexpr IteratorImpl() = default; + +# if DO_CHECK + FORCEINLINE IteratorImpl(const IteratorImpl& InValue) requires (bConst) + : Owner(InValue.Owner), Pointer(InValue.Pointer) + { } +# else + FORCEINLINE IteratorImpl(const IteratorImpl& InValue) requires (bConst) + : Pointer(InValue.Pointer) + { } +# endif + + FORCEINLINE constexpr IteratorImpl(const IteratorImpl&) = default; + FORCEINLINE constexpr IteratorImpl(IteratorImpl&&) = default; + FORCEINLINE constexpr IteratorImpl& operator=(const IteratorImpl&) = default; + FORCEINLINE constexpr IteratorImpl& operator=(IteratorImpl&&) = default; + + NODISCARD friend FORCEINLINE constexpr bool operator==(const IteratorImpl& LHS, const IteratorImpl& RHS) { return LHS.Pointer == RHS.Pointer; } + + NODISCARD friend FORCEINLINE constexpr strong_ordering operator<=>(const IteratorImpl& LHS, const IteratorImpl& RHS) { return LHS.Pointer <=> RHS.Pointer; } + + NODISCARD FORCEINLINE constexpr ElementType& operator*() const { CheckThis(true); return *Pointer; } + NODISCARD FORCEINLINE constexpr ElementType* operator->() const { CheckThis(true); return Pointer; } + + NODISCARD FORCEINLINE constexpr ElementType& operator[](ptrdiff Index) const { IteratorImpl Temp = *this + Index; return *Temp; } + + FORCEINLINE constexpr IteratorImpl& operator++() { ++Pointer; CheckThis(); return *this; } + FORCEINLINE constexpr IteratorImpl& operator--() { --Pointer; CheckThis(); return *this; } + + FORCEINLINE constexpr IteratorImpl operator++(int) { IteratorImpl Temp = *this; ++*this; return Temp; } + FORCEINLINE constexpr IteratorImpl operator--(int) { IteratorImpl Temp = *this; --*this; return Temp; } + + FORCEINLINE constexpr IteratorImpl& operator+=(ptrdiff Offset) { Pointer += Offset; CheckThis(); return *this; } + FORCEINLINE constexpr IteratorImpl& operator-=(ptrdiff Offset) { Pointer -= Offset; CheckThis(); return *this; } + + NODISCARD friend FORCEINLINE constexpr IteratorImpl operator+(IteratorImpl Iter, ptrdiff Offset) { IteratorImpl Temp = Iter; Temp += Offset; return Temp; } + NODISCARD friend FORCEINLINE constexpr IteratorImpl operator+(ptrdiff Offset, IteratorImpl Iter) { IteratorImpl Temp = Iter; Temp += Offset; return Temp; } + + NODISCARD FORCEINLINE constexpr IteratorImpl operator-(ptrdiff Offset) const { IteratorImpl Temp = *this; Temp -= Offset; return Temp; } + + NODISCARD friend FORCEINLINE constexpr ptrdiff operator-(const IteratorImpl& LHS, const IteratorImpl& RHS) { LHS.CheckThis(); RHS.CheckThis(); return LHS.Pointer - RHS.Pointer; } + + NODISCARD FORCEINLINE constexpr explicit operator TObserverPtr() const { CheckThis(); return TObserverPtr(Pointer); } + + private: + +# if DO_CHECK + const TStaticArray* Owner = nullptr; +# endif + + ElementType* Pointer = nullptr; + +# if DO_CHECK + FORCEINLINE constexpr IteratorImpl(const TStaticArray* InContainer, ElementType* InPointer) + : Owner(InContainer), Pointer(InPointer) + { } +# else + FORCEINLINE constexpr IteratorImpl(const TStaticArray* InContainer, ElementType* InPointer) + : Pointer(InPointer) + { } +# endif + + FORCEINLINE constexpr void CheckThis(bool bExceptEnd = false) const + { + checkf(Owner && Owner->IsValidIterator(*this), TEXT("Read access violation. Please check IsValidIterator().")); + checkf(!(bExceptEnd && Owner->End() == *this), TEXT("Read access violation. Please check IsValidIterator().")); + } + + template friend class IteratorImpl; + + friend TStaticArray; + + }; + }; template requires (true && ... && CSameAs) diff --git a/Redcraft.Utility/Source/Public/Memory/Allocator.h b/Redcraft.Utility/Source/Public/Memory/Allocator.h index 21457c8..19cc606 100644 --- a/Redcraft.Utility/Source/Public/Memory/Allocator.h +++ b/Redcraft.Utility/Source/Public/Memory/Allocator.h @@ -12,8 +12,17 @@ NAMESPACE_MODULE_BEGIN(Utility) struct FAllocatorInterface; -template -concept CInstantiableAllocator = CDerivedFrom && !CSameAs; +template +concept CInstantiableAllocator = !CSameAs + && requires (typename A::template ForElementType& Allocator, T* InPtr, size_t Num, size_t NumAllocated) + { + { Allocator.Allocate(Num) } -> CSameAs; + { Allocator.Deallocate(InPtr) } -> CSameAs; + { AsConst(Allocator).IsTransferable(InPtr) } -> CBooleanTestable; + { AsConst(Allocator).CalculateSlackGrow(Num, NumAllocated) } -> CSameAs; + { AsConst(Allocator).CalculateSlackShrink(Num, NumAllocated) } -> CSameAs; + { AsConst(Allocator).CalculateSlackReserve(Num) } -> CSameAs; + }; /** * This is the allocator interface, the allocator does not use virtual, this contains the default of @@ -24,10 +33,16 @@ concept CInstantiableAllocator = CDerivedFrom && !CSameA struct FAllocatorInterface { template - class ForElementType : private FSingleton + class ForElementType /*: private FSingleton*/ { public: + ForElementType() = default; + ForElementType(const ForElementType&) = delete; + ForElementType(ForElementType&&) = delete; + ForElementType& operator=(const ForElementType&) = delete; + ForElementType& operator=(ForElementType&&) = delete; + /** * Allocates uninitialized storage. * Should be allocated according to the results given by the CalculateSlackReserve() family, @@ -57,7 +72,7 @@ struct FAllocatorInterface #define ALLOCATOR_WRAPPER_BEGIN(Allocator, Type, Name) \ \ - struct PREPROCESSOR_JOIN(F, Name) : private FSingleton + struct PREPROCESSOR_JOIN(F, Name) /*: private FSingleton*/ #define ALLOCATOR_WRAPPER_END(Allocator, Type, Name) ; \ \ @@ -90,13 +105,19 @@ struct FAllocatorInterface PREPROCESSOR_JOIN(T, Name)> Name; /** This is heap allocator that calls Memory::Malloc() directly for memory allocation. */ -struct FHeapAllocator : public FAllocatorInterface +struct FHeapAllocator { template - class ForElementType : public FAllocatorInterface::ForElementType + class ForElementType { public: + ForElementType() = default; + ForElementType(const ForElementType&) = delete; + ForElementType(ForElementType&&) = delete; + ForElementType& operator=(const ForElementType&) = delete; + ForElementType& operator=(ForElementType&&) = delete; + NODISCARD FORCEINLINE T* Allocate(size_t InNum) { return InNum != 0 ? static_cast(Memory::Malloc(Memory::QuantizeSize(InNum * sizeof(T)), alignof(T))) : nullptr; @@ -107,6 +128,8 @@ struct FHeapAllocator : public FAllocatorInterface Memory::Free(InPtr); } + NODISCARD FORCEINLINE bool IsTransferable(T* InPtr) const { return true; } + NODISCARD FORCEINLINE size_t CalculateSlackGrow(size_t Num, size_t NumAllocated) const { const size_t FirstGrow = 4; @@ -159,13 +182,19 @@ struct FHeapAllocator : public FAllocatorInterface * Any allocation needed beyond that causes all data to be moved into an indirect allocation. */ template -struct TInlineAllocator : public FAllocatorInterface +struct TInlineAllocator { template - class ForElementType : public FAllocatorInterface::ForElementType + class ForElementType { public: + ForElementType() = default; + ForElementType(const ForElementType&) = delete; + ForElementType(ForElementType&&) = delete; + ForElementType& operator=(const ForElementType&) = delete; + ForElementType& operator=(ForElementType&&) = delete; + NODISCARD FORCEINLINE T* Allocate(size_t InNum) { if (InNum == 0) return nullptr; @@ -233,13 +262,19 @@ struct TInlineAllocator : public FAllocatorInterface }; /** This is a null allocator for which all operations are illegal. */ -struct FNullAllocator : public FAllocatorInterface +struct FNullAllocator { template - class ForElementType : public FAllocatorInterface::ForElementType + class ForElementType { public: + ForElementType() = default; + ForElementType(const ForElementType&) = delete; + ForElementType(ForElementType&&) = delete; + ForElementType& operator=(const ForElementType&) = delete; + ForElementType& operator=(ForElementType&&) = delete; + NODISCARD FORCEINLINE T* Allocate(size_t InNum) { check_no_entry(); return nullptr; } FORCEINLINE void Deallocate(T* InPtr) { check_no_entry(); } diff --git a/Redcraft.Utility/Source/Public/Templates/Noncopyable.h b/Redcraft.Utility/Source/Public/Templates/Noncopyable.h index 53da1a2..88a2c55 100644 --- a/Redcraft.Utility/Source/Public/Templates/Noncopyable.h +++ b/Redcraft.Utility/Source/Public/Templates/Noncopyable.h @@ -6,6 +6,8 @@ NAMESPACE_REDCRAFT_BEGIN NAMESPACE_MODULE_BEGIN(Redcraft) NAMESPACE_MODULE_BEGIN(Utility) +// WARNING: Using these helper classes as base classes may raise potential EBO issues and is not recommended for objects with strict size constraints. + /** A class indicates that a derived class cannot be copied. */ struct FNoncopyable {