2015-05-24 04:55:12 +00:00
|
|
|
// Copyright 2008 Dolphin Emulator Project
|
2021-07-05 01:22:19 +00:00
|
|
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
2008-12-08 04:46:09 +00:00
|
|
|
|
2014-02-10 18:54:46 +00:00
|
|
|
#pragma once
|
2008-12-08 04:46:09 +00:00
|
|
|
|
2016-06-25 15:39:54 +00:00
|
|
|
#include <array>
|
2014-02-20 03:11:52 +00:00
|
|
|
#include <cstddef>
|
2019-08-31 18:16:16 +00:00
|
|
|
#include <type_traits>
|
2016-06-25 15:39:54 +00:00
|
|
|
#include <utility>
|
2014-02-20 03:11:52 +00:00
|
|
|
|
2008-09-28 17:34:40 +00:00
|
|
|
// STL-look-a-like interface, but name is mixed case to distinguish it clearly from the
|
|
|
|
// real STL classes.
|
2016-06-25 15:39:54 +00:00
|
|
|
//
|
2008-09-28 17:34:40 +00:00
|
|
|
// Not fully featured, no safety checking yet. Add features as needed.
|
2008-12-08 04:46:09 +00:00
|
|
|
|
2008-09-28 17:34:40 +00:00
|
|
|
template <class T, int N>
|
|
|
|
class FixedSizeQueue
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
void clear()
|
2016-06-24 08:43:46 +00:00
|
|
|
{
|
2019-08-31 18:16:16 +00:00
|
|
|
if constexpr (!std::is_trivial_v<T>)
|
2019-10-06 06:23:45 +00:00
|
|
|
{
|
|
|
|
// The clear of non-trivial objects previously used "storage = {}". However, this causes GCC
|
|
|
|
// to take a very long time to compile the file/function, as well as generating huge amounts
|
|
|
|
// of debug information (~2GB object file, ~600MB of debug info).
|
|
|
|
while (count > 0)
|
|
|
|
pop();
|
|
|
|
}
|
2019-08-31 18:16:16 +00:00
|
|
|
|
2008-09-28 17:34:40 +00:00
|
|
|
head = 0;
|
|
|
|
tail = 0;
|
|
|
|
count = 0;
|
|
|
|
}
|
2008-12-08 04:46:09 +00:00
|
|
|
|
2014-08-30 20:14:56 +00:00
|
|
|
void push(T t)
|
|
|
|
{
|
2019-08-31 18:16:16 +00:00
|
|
|
if (count == N)
|
|
|
|
head = (head + 1) % N;
|
|
|
|
else
|
|
|
|
count++;
|
|
|
|
|
2016-06-25 15:39:54 +00:00
|
|
|
storage[tail] = std::move(t);
|
2019-08-31 18:16:16 +00:00
|
|
|
tail = (tail + 1) % N;
|
|
|
|
}
|
|
|
|
|
|
|
|
template <class... Args>
|
|
|
|
void emplace(Args&&... args)
|
|
|
|
{
|
|
|
|
if (count == N)
|
|
|
|
head = (head + 1) % N;
|
|
|
|
else
|
|
|
|
count++;
|
|
|
|
|
|
|
|
storage[tail] = T(std::forward<Args>(args)...);
|
|
|
|
tail = (tail + 1) % N;
|
2008-09-28 17:34:40 +00:00
|
|
|
}
|
2008-12-08 04:46:09 +00:00
|
|
|
|
2014-08-30 20:14:56 +00:00
|
|
|
void pop()
|
|
|
|
{
|
2019-08-31 18:16:16 +00:00
|
|
|
if constexpr (!std::is_trivial_v<T>)
|
|
|
|
storage[head] = {};
|
|
|
|
|
|
|
|
head = (head + 1) % N;
|
2016-01-21 20:27:56 +00:00
|
|
|
count--;
|
2008-09-28 17:34:40 +00:00
|
|
|
}
|
2008-12-08 04:46:09 +00:00
|
|
|
|
2016-01-21 20:27:56 +00:00
|
|
|
T pop_front()
|
|
|
|
{
|
2019-08-31 18:16:16 +00:00
|
|
|
T temp = std::move(front());
|
2016-01-21 20:27:56 +00:00
|
|
|
pop();
|
2019-08-31 18:16:16 +00:00
|
|
|
return temp;
|
2016-01-21 20:27:56 +00:00
|
|
|
}
|
2008-12-08 04:46:09 +00:00
|
|
|
|
2019-08-31 18:16:16 +00:00
|
|
|
T& front() noexcept { return storage[head]; }
|
|
|
|
const T& front() const noexcept { return storage[head]; }
|
|
|
|
size_t size() const noexcept { return count; }
|
|
|
|
bool empty() const noexcept { return size() == 0; }
|
2018-04-12 12:18:04 +00:00
|
|
|
|
2016-06-25 15:39:54 +00:00
|
|
|
private:
|
2021-09-04 04:43:19 +00:00
|
|
|
std::array<T, N> storage{};
|
2016-06-25 15:39:54 +00:00
|
|
|
int head = 0;
|
|
|
|
int tail = 0;
|
|
|
|
// Sacrifice 4 bytes for a simpler implementation. may optimize away in the future.
|
|
|
|
int count = 0;
|
2008-09-28 17:34:40 +00:00
|
|
|
};
|