2022-12-13 10:39:57 +02:00
|
|
|
#pragma once
|
|
|
|
|
|
|
|
#include <BAN/Errors.h>
|
2022-12-30 19:52:16 +02:00
|
|
|
#include <BAN/Math.h>
|
2023-01-17 11:38:16 +02:00
|
|
|
#include <BAN/Memory.h>
|
2023-01-13 14:11:02 +02:00
|
|
|
#include <BAN/Move.h>
|
2022-12-13 10:39:57 +02:00
|
|
|
|
|
|
|
namespace BAN
|
|
|
|
{
|
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
class Queue
|
|
|
|
{
|
|
|
|
public:
|
2023-01-13 14:11:02 +02:00
|
|
|
using size_type = size_t;
|
2022-12-13 10:39:57 +02:00
|
|
|
using value_type = T;
|
|
|
|
|
|
|
|
public:
|
|
|
|
Queue() = default;
|
2023-01-13 14:11:02 +02:00
|
|
|
Queue(Queue<T>&&);
|
|
|
|
Queue(const Queue<T>&);
|
2022-12-13 10:39:57 +02:00
|
|
|
~Queue();
|
|
|
|
|
2023-01-13 14:11:02 +02:00
|
|
|
Queue<T>& operator=(Queue<T>&&);
|
|
|
|
Queue<T>& operator=(const Queue<T>&);
|
|
|
|
|
2023-02-19 20:10:30 +02:00
|
|
|
ErrorOr<void> push(T&&);
|
|
|
|
ErrorOr<void> push(const T&);
|
2023-01-17 12:02:59 +02:00
|
|
|
template<typename... Args>
|
2023-02-19 20:10:30 +02:00
|
|
|
ErrorOr<void> emplace(Args&&...);
|
2023-01-13 14:11:02 +02:00
|
|
|
|
2023-02-22 02:07:05 +02:00
|
|
|
ErrorOr<void> reserve(size_type);
|
|
|
|
ErrorOr<void> shrink_to_fit();
|
|
|
|
|
2023-02-01 21:05:44 +02:00
|
|
|
void pop();
|
|
|
|
void clear();
|
2022-12-13 10:39:57 +02:00
|
|
|
|
2023-02-01 21:05:44 +02:00
|
|
|
bool empty() const;
|
|
|
|
size_type size() const;
|
2022-12-13 10:39:57 +02:00
|
|
|
|
2023-02-01 21:05:44 +02:00
|
|
|
const T& front() const;
|
|
|
|
T& front();
|
2022-12-13 10:39:57 +02:00
|
|
|
|
|
|
|
private:
|
2023-02-19 20:10:30 +02:00
|
|
|
ErrorOr<void> ensure_capacity(size_type size);
|
2022-12-13 10:39:57 +02:00
|
|
|
|
|
|
|
private:
|
2023-02-10 00:06:58 +02:00
|
|
|
T* m_data = nullptr;
|
2022-12-13 10:39:57 +02:00
|
|
|
size_type m_capacity = 0;
|
|
|
|
size_type m_size = 0;
|
|
|
|
};
|
|
|
|
|
2023-01-13 14:11:02 +02:00
|
|
|
template<typename T>
|
|
|
|
Queue<T>::Queue(Queue<T>&& other)
|
|
|
|
{
|
|
|
|
m_data = other.m_data;
|
|
|
|
m_capacity = other.m_capacity;
|
|
|
|
m_size = other.m_size;
|
|
|
|
|
|
|
|
other.m_data = nullptr;
|
|
|
|
other.m_capacity = 0;
|
|
|
|
other.m_size = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
Queue<T>::Queue(const Queue<T>& other)
|
|
|
|
{
|
2023-02-01 21:05:44 +02:00
|
|
|
MUST(ensure_capacity(other.size()));
|
|
|
|
for (size_type i = 0; i < other.size(); i++)
|
2023-02-10 00:06:58 +02:00
|
|
|
new (m_data + i) T(other.m_data[i]);
|
2023-01-13 14:11:02 +02:00
|
|
|
m_size = other.m_size;
|
|
|
|
}
|
|
|
|
|
2022-12-13 10:39:57 +02:00
|
|
|
template<typename T>
|
|
|
|
Queue<T>::~Queue()
|
|
|
|
{
|
2023-02-01 21:05:44 +02:00
|
|
|
clear();
|
2022-12-13 10:39:57 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
template<typename T>
|
2023-01-13 14:11:02 +02:00
|
|
|
Queue<T>& Queue<T>::operator=(Queue<T>&& other)
|
|
|
|
{
|
2023-02-01 21:05:44 +02:00
|
|
|
clear();
|
2023-01-13 14:11:02 +02:00
|
|
|
|
|
|
|
m_data = other.m_data;
|
|
|
|
m_capacity = other.m_capacity;
|
|
|
|
m_size = other.m_size;
|
|
|
|
|
|
|
|
other.m_data = nullptr;
|
|
|
|
other.m_capacity = 0;
|
|
|
|
other.m_size = 0;
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
Queue<T>& Queue<T>::operator=(const Queue<T>& other)
|
|
|
|
{
|
2023-02-01 21:05:44 +02:00
|
|
|
clear();
|
|
|
|
MUST(ensure_capacity(other.size()));
|
|
|
|
for (size_type i = 0; i < other.size(); i++)
|
2023-02-10 00:06:58 +02:00
|
|
|
new (m_data + i) T(other.m_data[i]);
|
2023-01-13 14:11:02 +02:00
|
|
|
m_size = other.m_size;
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename T>
|
2023-02-01 21:05:44 +02:00
|
|
|
ErrorOr<void> Queue<T>::push(T&& value)
|
2022-12-13 10:39:57 +02:00
|
|
|
{
|
2023-02-01 21:05:44 +02:00
|
|
|
TRY(ensure_capacity(m_size + 1));
|
2023-02-10 00:06:58 +02:00
|
|
|
new (m_data + m_size) T(move(value));
|
2023-01-13 14:11:02 +02:00
|
|
|
m_size++;
|
2022-12-13 10:39:57 +02:00
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
2023-01-13 14:11:02 +02:00
|
|
|
template<typename T>
|
2023-02-01 21:05:44 +02:00
|
|
|
ErrorOr<void> Queue<T>::push(const T& value)
|
2023-01-13 14:11:02 +02:00
|
|
|
{
|
2023-02-01 21:05:44 +02:00
|
|
|
return push(move(T(value)));
|
2023-01-13 14:11:02 +02:00
|
|
|
}
|
|
|
|
|
2023-01-17 12:02:59 +02:00
|
|
|
template<typename T>
|
|
|
|
template<typename... Args>
|
2023-02-01 21:41:51 +02:00
|
|
|
ErrorOr<void> Queue<T>::emplace(Args&&... args)
|
2023-01-17 12:02:59 +02:00
|
|
|
{
|
2023-02-01 21:05:44 +02:00
|
|
|
TRY(ensure_capacity(m_size + 1));
|
2023-02-10 00:06:58 +02:00
|
|
|
new (m_data + m_size) T(forward<Args>(args)...);
|
2023-01-17 12:02:59 +02:00
|
|
|
m_size++;
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
2023-02-22 02:07:05 +02:00
|
|
|
template<typename T>
|
|
|
|
ErrorOr<void> Queue<T>::reserve(size_type size)
|
|
|
|
{
|
|
|
|
TRY(ensure_capacity(size));
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
ErrorOr<void> Queue<T>::shrink_to_fit()
|
|
|
|
{
|
|
|
|
size_type temp = m_capacity;
|
|
|
|
m_capacity = 0;
|
|
|
|
auto error_or = ensure_capacity(m_size);
|
|
|
|
if (error_or.is_error())
|
|
|
|
{
|
|
|
|
m_capacity = temp;
|
|
|
|
return error_or;
|
|
|
|
}
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
2022-12-13 10:39:57 +02:00
|
|
|
template<typename T>
|
2023-02-01 21:05:44 +02:00
|
|
|
void Queue<T>::pop()
|
2022-12-13 10:39:57 +02:00
|
|
|
{
|
2023-01-10 17:43:18 +02:00
|
|
|
ASSERT(m_size > 0);
|
2023-01-13 14:11:02 +02:00
|
|
|
for (size_type i = 0; i < m_size - 1; i++)
|
2023-02-10 00:06:58 +02:00
|
|
|
m_data[i] = move(m_data[i + 1]);
|
|
|
|
m_data[m_size - 1].~T();
|
2023-01-13 14:11:02 +02:00
|
|
|
m_size--;
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename T>
|
2023-02-01 21:05:44 +02:00
|
|
|
void Queue<T>::clear()
|
2023-01-13 14:11:02 +02:00
|
|
|
{
|
|
|
|
for (size_type i = 0; i < m_size; i++)
|
2023-02-10 00:06:58 +02:00
|
|
|
m_data[i].~T();
|
2023-01-13 14:11:02 +02:00
|
|
|
BAN::deallocator(m_data);
|
|
|
|
m_data = nullptr;
|
|
|
|
m_capacity = 0;
|
|
|
|
m_size = 0;
|
2022-12-13 10:39:57 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
template<typename T>
|
2023-02-01 21:05:44 +02:00
|
|
|
bool Queue<T>::empty() const
|
2022-12-13 10:39:57 +02:00
|
|
|
{
|
|
|
|
return m_size == 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename T>
|
2023-02-01 21:05:44 +02:00
|
|
|
typename Queue<T>::size_type Queue<T>::size() const
|
2022-12-13 10:39:57 +02:00
|
|
|
{
|
|
|
|
return m_size;
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename T>
|
2023-02-01 21:05:44 +02:00
|
|
|
const T& Queue<T>::front() const
|
2022-12-13 10:39:57 +02:00
|
|
|
{
|
2023-01-10 17:43:18 +02:00
|
|
|
ASSERT(m_size > 0);
|
2023-02-10 00:06:58 +02:00
|
|
|
return m_data[0];
|
2022-12-13 10:39:57 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
template<typename T>
|
2023-02-01 21:05:44 +02:00
|
|
|
T& Queue<T>::front()
|
2022-12-13 10:39:57 +02:00
|
|
|
{
|
2023-01-10 17:43:18 +02:00
|
|
|
ASSERT(m_size > 0);
|
2023-02-10 00:06:58 +02:00
|
|
|
return m_data[0];
|
2022-12-13 10:39:57 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
template<typename T>
|
2023-02-01 21:05:44 +02:00
|
|
|
ErrorOr<void> Queue<T>::ensure_capacity(size_type size)
|
2022-12-13 10:39:57 +02:00
|
|
|
{
|
|
|
|
if (m_capacity > size)
|
|
|
|
return {};
|
2023-02-22 02:07:05 +02:00
|
|
|
size_type new_cap = BAN::Math::max<size_type>(size, m_capacity * 2);
|
2023-02-10 00:06:58 +02:00
|
|
|
T* new_data = (T*)BAN::allocator(new_cap * sizeof(T));
|
2022-12-13 10:39:57 +02:00
|
|
|
if (new_data == nullptr)
|
2023-03-02 21:10:44 +02:00
|
|
|
return Error::from_errno(ENOMEM);
|
2023-01-13 14:11:02 +02:00
|
|
|
for (size_type i = 0; i < m_size; i++)
|
|
|
|
{
|
2023-02-10 00:06:58 +02:00
|
|
|
new (new_data + i) T(move(m_data[i]));
|
|
|
|
m_data[i].~T();
|
2023-01-13 14:11:02 +02:00
|
|
|
}
|
|
|
|
BAN::deallocator(m_data);
|
2023-01-13 01:09:40 +02:00
|
|
|
m_data = new_data;
|
2022-12-13 10:39:57 +02:00
|
|
|
m_capacity = new_cap;
|
2022-12-13 12:10:50 +02:00
|
|
|
return {};
|
2022-12-13 10:39:57 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|