forked from Bananymous/banan-os
				
			BAN: Rewrite Queue with move semantics
This commit is contained in:
		
							parent
							
								
									0f4e95acc5
								
							
						
					
					
						commit
						57cbd728db
					
				| 
						 | 
					@ -2,10 +2,7 @@
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#include <BAN/Errors.h>
 | 
					#include <BAN/Errors.h>
 | 
				
			||||||
#include <BAN/Math.h>
 | 
					#include <BAN/Math.h>
 | 
				
			||||||
#include <BAN/Memory.h>
 | 
					#include <BAN/Move.h>
 | 
				
			||||||
 | 
					 | 
				
			||||||
#include <stdint.h>
 | 
					 | 
				
			||||||
#include <string.h>
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
namespace BAN
 | 
					namespace BAN
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
| 
						 | 
					@ -14,15 +11,23 @@ namespace BAN
 | 
				
			||||||
	class Queue
 | 
						class Queue
 | 
				
			||||||
	{
 | 
						{
 | 
				
			||||||
	public:
 | 
						public:
 | 
				
			||||||
		using size_type = uint32_t;
 | 
							using size_type = size_t;
 | 
				
			||||||
		using value_type = T;
 | 
							using value_type = T;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	public:
 | 
						public:
 | 
				
			||||||
		Queue() = default;
 | 
							Queue() = default;
 | 
				
			||||||
 | 
							Queue(Queue<T>&&);
 | 
				
			||||||
 | 
							Queue(const Queue<T>&);
 | 
				
			||||||
		~Queue();
 | 
							~Queue();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		[[nodiscard]] ErrorOr<void> Push(const T& value);
 | 
							Queue<T>& operator=(Queue<T>&&);
 | 
				
			||||||
 | 
							Queue<T>& operator=(const Queue<T>&);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							[[nodiscard]] ErrorOr<void> Push(T&&);
 | 
				
			||||||
 | 
							[[nodiscard]] ErrorOr<void> Push(const T&);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		void Pop();
 | 
							void Pop();
 | 
				
			||||||
 | 
							void Clear();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		bool Empty() const;
 | 
							bool Empty() const;
 | 
				
			||||||
		size_type Size() const;
 | 
							size_type Size() const;
 | 
				
			||||||
| 
						 | 
					@ -32,35 +37,102 @@ namespace BAN
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	private:
 | 
						private:
 | 
				
			||||||
		[[nodiscard]] ErrorOr<void> EnsureCapacity(size_type size);
 | 
							[[nodiscard]] ErrorOr<void> EnsureCapacity(size_type size);
 | 
				
			||||||
 | 
							T* Address(size_type, uint8_t* = nullptr) const;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	private:
 | 
						private:
 | 
				
			||||||
		T*			m_data		= nullptr;
 | 
							uint8_t*	m_data		= nullptr;
 | 
				
			||||||
		size_type	m_capacity	= 0;
 | 
							size_type	m_capacity	= 0;
 | 
				
			||||||
		size_type	m_size		= 0;
 | 
							size_type	m_size		= 0;
 | 
				
			||||||
	};
 | 
						};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						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)
 | 
				
			||||||
 | 
						{
 | 
				
			||||||
 | 
							MUST(EnsureCapacity(other.Size()));
 | 
				
			||||||
 | 
							for (size_type i = 0; i < other.Size(); i++)
 | 
				
			||||||
 | 
								new (Address(i)) T(*Address(i, other.m_data));
 | 
				
			||||||
 | 
							m_size = other.m_size;
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	template<typename T>
 | 
						template<typename T>
 | 
				
			||||||
	Queue<T>::~Queue()
 | 
						Queue<T>::~Queue()
 | 
				
			||||||
	{
 | 
						{
 | 
				
			||||||
		for (size_type i = 0; i < m_size; i++)
 | 
							Clear();
 | 
				
			||||||
			m_data[i].~T();
 | 
						}
 | 
				
			||||||
		delete[] m_data;
 | 
					
 | 
				
			||||||
 | 
						template<typename T>
 | 
				
			||||||
 | 
						Queue<T>& Queue<T>::operator=(Queue<T>&& other)
 | 
				
			||||||
 | 
						{
 | 
				
			||||||
 | 
							Clear();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							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)
 | 
				
			||||||
 | 
						{
 | 
				
			||||||
 | 
							Clear();
 | 
				
			||||||
 | 
							MUST(EnsureCapacity(other.Size()));
 | 
				
			||||||
 | 
							for (size_type i = 0; i < other.Size(); i++)
 | 
				
			||||||
 | 
								new (Address(i)) T(*Address(i, other.m_data));
 | 
				
			||||||
 | 
							m_size = other.m_size;
 | 
				
			||||||
 | 
							return *this;
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						template<typename T>
 | 
				
			||||||
 | 
						ErrorOr<void> Queue<T>::Push(T&& value)
 | 
				
			||||||
 | 
						{
 | 
				
			||||||
 | 
							TRY(EnsureCapacity(m_size + 1));
 | 
				
			||||||
 | 
							new (Address(m_size)) T(Move(value));
 | 
				
			||||||
 | 
							m_size++;
 | 
				
			||||||
 | 
							return {};
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	template<typename T>
 | 
						template<typename T>
 | 
				
			||||||
	ErrorOr<void> Queue<T>::Push(const T& value)
 | 
						ErrorOr<void> Queue<T>::Push(const T& value)
 | 
				
			||||||
	{
 | 
						{
 | 
				
			||||||
		TRY(EnsureCapacity(m_size + 1));
 | 
							return Push(Move(T(value)));
 | 
				
			||||||
		m_data[m_size++] = value;
 | 
					 | 
				
			||||||
		return {};
 | 
					 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	template<typename T>
 | 
						template<typename T>
 | 
				
			||||||
	void Queue<T>::Pop()
 | 
						void Queue<T>::Pop()
 | 
				
			||||||
	{
 | 
						{
 | 
				
			||||||
		ASSERT(m_size > 0);
 | 
							ASSERT(m_size > 0);
 | 
				
			||||||
		m_data->~T();
 | 
							for (size_type i = 0; i < m_size - 1; i++)
 | 
				
			||||||
		memmove(m_data, m_data + 1, sizeof(T) * (--m_size));
 | 
								*Address(i) = Move(*Address(i + 1));
 | 
				
			||||||
 | 
							Address(m_size - 1)->~T();
 | 
				
			||||||
 | 
							m_size--;
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						template<typename T>
 | 
				
			||||||
 | 
						void Queue<T>::Clear()
 | 
				
			||||||
 | 
						{
 | 
				
			||||||
 | 
							for (size_type i = 0; i < m_size; i++)
 | 
				
			||||||
 | 
								Address(i)->~T();
 | 
				
			||||||
 | 
							BAN::deallocator(m_data);
 | 
				
			||||||
 | 
							m_data = nullptr;
 | 
				
			||||||
 | 
							m_capacity = 0;
 | 
				
			||||||
 | 
							m_size = 0;
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	template<typename T>
 | 
						template<typename T>
 | 
				
			||||||
| 
						 | 
					@ -79,14 +151,14 @@ namespace BAN
 | 
				
			||||||
	const T& Queue<T>::Front() const
 | 
						const T& Queue<T>::Front() const
 | 
				
			||||||
	{
 | 
						{
 | 
				
			||||||
		ASSERT(m_size > 0);
 | 
							ASSERT(m_size > 0);
 | 
				
			||||||
		return *m_data;
 | 
							return *Address(0);
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	template<typename T>
 | 
						template<typename T>
 | 
				
			||||||
	T& Queue<T>::Front()
 | 
						T& Queue<T>::Front()
 | 
				
			||||||
	{
 | 
						{
 | 
				
			||||||
		ASSERT(m_size > 0);
 | 
							ASSERT(m_size > 0);
 | 
				
			||||||
		return *m_data;
 | 
							return *Address(0);
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	template<typename T>
 | 
						template<typename T>
 | 
				
			||||||
| 
						 | 
					@ -95,15 +167,26 @@ namespace BAN
 | 
				
			||||||
		if (m_capacity > size)
 | 
							if (m_capacity > size)
 | 
				
			||||||
			return {};
 | 
								return {};
 | 
				
			||||||
		size_type new_cap = BAN::Math::max<size_type>(size, m_capacity * 3 / 2);
 | 
							size_type new_cap = BAN::Math::max<size_type>(size, m_capacity * 3 / 2);
 | 
				
			||||||
		T* new_data = new T[new_cap];
 | 
							uint8_t* new_data = (uint8_t*)BAN::allocator(new_cap * sizeof(T));
 | 
				
			||||||
		if (new_data == nullptr)
 | 
							if (new_data == nullptr)
 | 
				
			||||||
			return Error::FromString("Queue: Could not allocate memory");
 | 
								return Error::FromString("Queue: Could not allocate memory");
 | 
				
			||||||
		if (m_data)
 | 
							for (size_type i = 0; i < m_size; i++)
 | 
				
			||||||
			memcpy(new_data, m_data, m_size * sizeof(T));
 | 
							{
 | 
				
			||||||
		delete[] m_data;
 | 
								new (Address(i, new_data)) T(Move(*Address(i)));
 | 
				
			||||||
 | 
								Address(i)->~T();
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							BAN::deallocator(m_data);
 | 
				
			||||||
		m_data = new_data;
 | 
							m_data = new_data;
 | 
				
			||||||
		m_capacity = new_cap;
 | 
							m_capacity = new_cap;
 | 
				
			||||||
		return {};
 | 
							return {};
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						template<typename T>
 | 
				
			||||||
 | 
						T* Queue<T>::Address(size_type index, uint8_t* base) const
 | 
				
			||||||
 | 
						{
 | 
				
			||||||
 | 
							if (base == nullptr)
 | 
				
			||||||
 | 
								base = m_data;
 | 
				
			||||||
 | 
							return (T*)(base + index * sizeof(T));
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
		Loading…
	
		Reference in New Issue