Sane C++ Libraries
C++ Platform Abstraction Libraries
Segment.h
1// Copyright (c) Stefano Cristiano
2// SPDX-License-Identifier: MIT
3#pragma once
4#include "../Foundation/Assert.h"
5#include "../Foundation/Span.h"
6
7namespace SC
8{
11enum class SegmentAllocator
12{
13 Global = 0,
14 ThreadLocal = 1,
15};
16
17namespace detail
18{
19struct SC_COMPILER_EXPORT SegmentHeader;
20struct alignas(uint64_t) SegmentHeader
21{
22 static constexpr uint32_t MaxCapacity = (~static_cast<uint32_t>(0)) >> 1;
23
24 SegmentHeader(uint32_t capacity = 0, SegmentAllocator allocator = SegmentAllocator::Global)
25 {
26 sizeBytes = 0;
27 allocatorType = static_cast<uint32_t>(allocator);
28 capacityBytes = capacity;
29 hasInlineData = capacity > 0;
30 }
31 uint32_t sizeBytes : sizeof(uint32_t) * 8 - 1;
32 uint32_t allocatorType : 1;
33 uint32_t capacityBytes : sizeof(uint32_t) * 8 - 1;
34 uint32_t hasInlineData : 1;
35};
36
37struct SegmentHeaderOffset
38{
39 using PtrOffset = size_t;
40 SegmentHeader header;
41 PtrOffset offset = 0; // memory offset representing relative pointer to data from "this"
42};
43
44template <typename T>
45struct SegmentSelfRelativePointer : protected SegmentHeaderOffset
46{
47 // clang-format off
48 SC_COMPILER_FORCE_INLINE T* data() noexcept { return offset == 0 ? nullptr : toPtr(toOffset(this) + offset); }
49 SC_COMPILER_FORCE_INLINE const T* data() const noexcept { return offset == 0 ? nullptr : toPtr(toOffset(this) + offset); }
50 SC_COMPILER_FORCE_INLINE bool isInline() const noexcept { return (offset == sizeof(SegmentHeaderOffset) + sizeof(uint64_t)) and header.hasInlineData; }
51
52 protected:
53 struct InlineData : public SegmentHeaderOffset // Data layout corresponds to SmallBuffer, SmallVector etc.
54 {
55 uint64_t capacity; // Could use uint32_t but we need to align data to 64 bit anyway
56 ~InlineData() {}
57 union
58 {
59 T data[1]; // Accessing the whole class through volatile cast anyway so array size can be whatever
60 };
61 };
62 SC_COMPILER_FORCE_INLINE static auto toOffset(const volatile void* src) noexcept { return reinterpret_cast<PtrOffset>(src); }
63 SC_COMPILER_FORCE_INLINE static T* toPtr(PtrOffset src) noexcept { return reinterpret_cast<T*>(src); }
64 SC_COMPILER_FORCE_INLINE void setData(T* mem) noexcept { offset = mem == nullptr ? 0 : toOffset(mem) - toOffset(this); }
65 SC_COMPILER_FORCE_INLINE T* getInlineData() noexcept { return (T*)reinterpret_cast<volatile InlineData*>(this)->data; }
66 SC_COMPILER_FORCE_INLINE auto getInlineCapacity() noexcept { return reinterpret_cast<volatile InlineData*>(this)->capacity; }
67 // clang-format on
68};
69
71template <typename T>
72struct SegmentTrivial
73{
74 using Type = T;
75 inline static void destruct(Span<T> data) noexcept;
76 // clang-format off
77 template <typename U> inline static void copyConstructAs(Span<T> data, Span<const T> value) noexcept;
78 template <typename U> inline static void copyConstruct(Span<T> data, const T* src) noexcept;
79 template <typename U> inline static void copyAssign(Span<T> data, const T* src) noexcept;
80 template <typename U> inline static void copyInsert(Span<T> data, Span<const T> values) noexcept;
81 template <typename U> inline static void moveConstruct(Span<T> data, T* src) noexcept;
82 template <typename U> inline static void moveAssign(Span<T> data, T* src) noexcept;
83 // clang-format on
84 inline static void remove(Span<T> data, size_t numElements) noexcept;
85};
86
88template <typename ParentSegment, typename CommonParent, int N = 0,
89 SegmentAllocator Allocator = SegmentAllocator::ThreadLocal>
90struct SegmentCustom : public ParentSegment
91{
92 SegmentCustom() : ParentSegment(N, Allocator) {}
93 SegmentCustom(const CommonParent& other) : SegmentCustom() { CommonParent::operator=(other); }
94 SegmentCustom(CommonParent&& other) : SegmentCustom() { CommonParent::operator=(move(other)); }
95
96 SegmentCustom(const SegmentCustom& other) : SegmentCustom() { ParentSegment::operator=(other); }
97 SegmentCustom(SegmentCustom&& other) : SegmentCustom() { ParentSegment::operator=(move(other)); }
98 // clang-format off
99 SegmentCustom& operator=(const SegmentCustom& other) { ParentSegment::operator=(other); return *this; }
100 SegmentCustom& operator=(SegmentCustom&& other) { ParentSegment::operator=(move(other)); return *this; }
101 // clang-format on
102};
103} // namespace detail
104
111template <typename VTable>
112struct Segment : public VTable
113{
114 using VTable::data;
115 using T = typename VTable::Type;
116 Segment(uint32_t capacityInBytes, SegmentAllocator allocator = SegmentAllocator::Global) noexcept;
117
118 Segment() noexcept;
119 ~Segment() noexcept;
120 Segment(Segment&& other) noexcept;
121 Segment(const Segment& other) noexcept;
122 Segment& operator=(Segment&& other) noexcept;
123 Segment& operator=(const Segment& other) noexcept;
124
125 // clang-format off
126 template <typename U = T> Segment(Span<const U> span) noexcept : Segment() { SC_ASSERT_RELEASE(assign(span)); }
127 Segment(std::initializer_list<T> list) noexcept : Segment() { SC_ASSERT_RELEASE(assign<T>({list.begin(), list.size()})); }
128 // clang-format on
129
132 [[nodiscard]] bool resizeWithoutInitializing(size_t newSize) noexcept;
133
135 [[nodiscard]] bool resize(size_t newSize, const T& value = T()) noexcept;
136
138 [[nodiscard]] bool reserve(size_t capacity) noexcept;
139
141 template <typename U = T>
142 [[nodiscard]] bool append(Span<const U> span) noexcept;
143
145 template <typename VTable2>
146 [[nodiscard]] bool appendMove(Segment<VTable2>&& other) noexcept;
147
150 [[nodiscard]] bool shrink_to_fit() noexcept;
151
153 void clear() noexcept;
154
157 template <typename U = T>
158 [[nodiscard]] bool assign(Span<const U> span) noexcept;
159
162 template <typename VTable2>
163 [[nodiscard]] bool assignMove(Segment<VTable2>&& other) noexcept;
164
166 [[nodiscard]] bool push_back(const T& value) noexcept { return resize(size() + 1, value); }
167
169 [[nodiscard]] bool push_back(T&& value) noexcept;
170
172 [[nodiscard]] bool push_front(const T& value) noexcept { return insert(0, value); }
173
177 [[nodiscard]] bool pop_back(T* removedValue = nullptr) noexcept;
178
182 [[nodiscard]] bool pop_front(T* removedValue = nullptr) noexcept;
183
184 // clang-format off
185 [[nodiscard]] T* begin() noexcept SC_LANGUAGE_LIFETIME_BOUND { return data(); }
186 [[nodiscard]] const T* begin() const noexcept SC_LANGUAGE_LIFETIME_BOUND { return data(); }
187 [[nodiscard]] T* end() noexcept SC_LANGUAGE_LIFETIME_BOUND { return data() + size(); }
188 [[nodiscard]] const T* end() const noexcept SC_LANGUAGE_LIFETIME_BOUND { return data() + size(); }
189
190 [[nodiscard]] T& back() noexcept SC_LANGUAGE_LIFETIME_BOUND { SC_ASSERT_RELEASE(not isEmpty()); return *(data() + size() - 1);}
191 [[nodiscard]] T& front() noexcept SC_LANGUAGE_LIFETIME_BOUND { SC_ASSERT_RELEASE(not isEmpty()); return *data();}
192 [[nodiscard]] const T& back() const noexcept SC_LANGUAGE_LIFETIME_BOUND { SC_ASSERT_RELEASE(not isEmpty()); return *(data() + size() - 1);}
193 [[nodiscard]] const T& front() const noexcept SC_LANGUAGE_LIFETIME_BOUND { SC_ASSERT_RELEASE(not isEmpty()); return *data();}
194
195 [[nodiscard]] T& operator[](size_t idx) noexcept SC_LANGUAGE_LIFETIME_BOUND { SC_ASSERT_DEBUG(idx < size()); return *(data() + idx);}
196 [[nodiscard]] const T& operator[](size_t idx) const noexcept SC_LANGUAGE_LIFETIME_BOUND { SC_ASSERT_DEBUG(idx < size()); return *(data() + idx);}
197 // clang-format on
198
200 [[nodiscard]] bool isEmpty() const noexcept { return VTable::header.sizeBytes == 0; }
201
203 [[nodiscard]] Span<T> toSpan() noexcept SC_LANGUAGE_LIFETIME_BOUND { return {data(), size()}; }
204
206 [[nodiscard]] Span<const T> toSpanConst() const noexcept SC_LANGUAGE_LIFETIME_BOUND { return {data(), size()}; }
207
209 [[nodiscard]] size_t size() const noexcept { return VTable::header.sizeBytes / sizeof(T); }
210
212 [[nodiscard]] size_t capacity() const noexcept { return VTable::header.capacityBytes / sizeof(T); }
213
218 [[nodiscard]] bool removeRange(size_t start, size_t length) noexcept;
219
223 [[nodiscard]] bool removeAt(size_t index) noexcept { return removeRange(index, 1); }
224
229 [[nodiscard]] bool insert(size_t index, Span<const T> data) noexcept;
230
231 protected:
232 template <typename VTable2>
233 friend struct Segment;
234 struct Internal;
235};
236
238} // namespace SC
#define SC_ASSERT_DEBUG(e)
Assert expression e to be true.
Definition: Assert.h:82
#define SC_COMPILER_EXPORT
Macro for symbol visibility in non-MSVC compilers.
Definition: Compiler.h:78
#define SC_ASSERT_RELEASE(e)
Assert expression e to be true.
Definition: Assert.h:66
#define SC_COMPILER_FORCE_INLINE
Macro for forcing inline functions.
Definition: Compiler.h:46
constexpr T && move(T &value)
Converts an lvalue to an rvalue reference.
Definition: Compiler.h:269
unsigned long long uint64_t
Platform independent (8) bytes unsigned int.
Definition: PrimitiveTypes.h:42
unsigned long size_t
Platform independent unsigned size type.
Definition: PrimitiveTypes.h:56
unsigned int uint32_t
Platform independent (4) bytes unsigned int.
Definition: PrimitiveTypes.h:38
A slice of contiguous memory, prefixed by and header containing size and capacity.
Definition: Segment.h:113
size_t size() const noexcept
Returns current size.
Definition: Segment.h:209
bool resize(size_t newSize, const T &value=T()) noexcept
Re-allocates to the requested new size, preserving its contents and setting new items to value.
size_t capacity() const noexcept
Returns current capacity (always >= of size())
Definition: Segment.h:212
Span< T > toSpan() noexcept SC_LANGUAGE_LIFETIME_BOUND
Obtains a Span of internal contents.
Definition: Segment.h:203
bool isEmpty() const noexcept
Check if is empty (size() == 0)
Definition: Segment.h:200
bool removeRange(size_t start, size_t length) noexcept
Removes the range [start, start + length] from the segment.
bool push_front(const T &value) noexcept
Appends a single element to the start of the segment.
Definition: Segment.h:172
bool removeAt(size_t index) noexcept
Removes the element at index.
Definition: Segment.h:223
bool pop_back(T *removedValue=nullptr) noexcept
Removes the last element of the segment.
bool push_back(T &&value) noexcept
Moves a single element to the end of the segment.
bool insert(size_t index, Span< const T > data) noexcept
Insert a span at the given index.
bool resizeWithoutInitializing(size_t newSize) noexcept
Re-allocates to the requested new size, preserving its contents.
Span< const T > toSpanConst() const noexcept SC_LANGUAGE_LIFETIME_BOUND
Obtains a Span of internal contents.
Definition: Segment.h:206
View over a contiguous sequence of items (pointer + size in elements).
Definition: Span.h:32