1
0
mirror of https://github.com/doitsujin/dxvk.git synced 2025-01-21 02:52:10 +01:00
dxvk/src/util/util_small_vector.h
Robin Kertels 80e950ac32 [util] Fix small_vector move
... again. We can only move over the ptr
if the other small_vector has MORE than N
elements, otherwise it will still use the local array.
2024-09-20 01:20:25 +02:00

202 lines
4.3 KiB
C++

#pragma once
#include <type_traits>
namespace dxvk {
template<typename T, size_t N>
class small_vector {
using storage = std::aligned_storage_t<sizeof(T), alignof(T)>;
public:
constexpr static size_t MinCapacity = N;
small_vector() { }
small_vector(const small_vector& other) {
reserve(other.m_size);
for (size_t i = 0; i < other.m_size; i++) {
*ptr(i) = *other.ptr(i);
}
m_size = other.m_size;
};
small_vector& operator = (const small_vector& other) {
for (size_t i = 0; i < m_size; i++)
ptr(i)->~T();
reserve(other.m_size);
for (size_t i = 0; i < other.m_size; i++) {
*ptr(i) = *other.ptr(i);
}
m_size = other.m_size;
};
small_vector(small_vector&& other) {
if (other.m_size <= N) {
for (size_t i = 0; i < other.m_size; i++) {
new (&u.m_data[i]) T(std::move(*other.ptr(i)));
}
} else {
u.m_ptr = other.u.m_ptr;
m_capacity = other.m_capacity;
other.u.m_ptr = nullptr;
other.m_capacity = N;
}
m_size = other.m_size;
other.m_size = 0;
}
small_vector& operator = (small_vector&& other) {
for (size_t i = 0; i < m_size; i++)
ptr(i)->~T();
if (m_capacity > N)
delete[] u.m_ptr;
if (other.m_size <= N) {
m_capacity = N;
for (size_t i = 0; i < other.m_size; i++) {
new (&u.m_data[i]) T(std::move(*other.ptr(i)));
}
} else {
u.m_ptr = other.u.m_ptr;
m_capacity = other.m_capacity;
other.u.m_ptr = nullptr;
other.m_capacity = N;
}
m_size = other.m_size;
other.m_size = 0;
}
~small_vector() {
for (size_t i = 0; i < m_size; i++)
ptr(i)->~T();
if (m_capacity > N)
delete[] u.m_ptr;
}
size_t size() const {
return m_size;
}
void reserve(size_t n) {
n = pick_capacity(n);
if (n <= m_capacity)
return;
storage* data = new storage[n];
for (size_t i = 0; i < m_size; i++) {
new (&data[i]) T(std::move(*ptr(i)));
ptr(i)->~T();
}
if (m_capacity > N)
delete[] u.m_ptr;
m_capacity = n;
u.m_ptr = data;
}
const T* data() const { return ptr(0); }
T* data() { return ptr(0); }
void resize(size_t n) {
reserve(n);
for (size_t i = n; i < m_size; i++)
ptr(i)->~T();
for (size_t i = m_size; i < n; i++)
new (ptr(i)) T();
m_size = n;
}
void push_back(const T& object) {
reserve(m_size + 1);
new (ptr(m_size++)) T(object);
}
void push_back(T&& object) {
reserve(m_size + 1);
new (ptr(m_size++)) T(std::move(object));
}
template<typename... Args>
void emplace_back(Args... args) {
reserve(m_size + 1);
new (ptr(m_size++)) T(std::forward<Args>(args)...);
}
void erase(size_t idx) {
ptr(idx)->~T();
for (size_t i = idx; i < m_size - 1; i++) {
new (ptr(i)) T(std::move(*ptr(i + 1)));
ptr(i + 1)->~T();
}
}
void pop_back() {
ptr(--m_size)->~T();
}
void clear() {
for (size_t i = 0; i < m_size; i++)
ptr(i)->~T();
m_size = 0;
}
bool empty() const {
return m_size == 0;
}
T& operator [] (size_t idx) { return *ptr(idx); }
const T& operator [] (size_t idx) const { return *ptr(idx); }
T& front() { return *ptr(0); }
const T& front() const { return *ptr(0); }
T& back() { return *ptr(m_size - 1); }
const T& back() const { return *ptr(m_size - 1); }
private:
size_t m_capacity = N;
size_t m_size = 0;
union {
storage* m_ptr;
storage m_data[N];
} u;
size_t pick_capacity(size_t n) {
size_t capacity = m_capacity;
while (capacity < n)
capacity *= 2;
return capacity;
}
T* ptr(size_t idx) {
return m_capacity == N
? reinterpret_cast<T*>(&u.m_data[idx])
: reinterpret_cast<T*>(&u.m_ptr[idx]);
}
const T* ptr(size_t idx) const {
return m_capacity == N
? reinterpret_cast<const T*>(&u.m_data[idx])
: reinterpret_cast<const T*>(&u.m_ptr[idx]);
}
};
}