10#include <condition_variable>
15#pragma GCC diagnostic push
16#pragma GCC diagnostic ignored "-Wunknown-pragmas"
27 template <
class T,
size_t N_cap>
31#pragma warning(suppress:26495)
43 std::tuple<T*, size_t>
back()
45 std::unique_lock<std::mutex> lk(m_mutex);
47 m_head_moved.wait(lk, [&]{
return m_quit || space();});
48 if (m_quit) _Unlikely_
49 return {
nullptr, 0 };
51 size_t tail = wrap(m_head + m_size);
52 return { &m_data[tail], m_head <= tail ? N_cap - tail : m_head - tail };
60 void push(_In_
size_t size)
63 const std::lock_guard<std::mutex> lg(m_mutex);
65 size_t tail = wrap(m_head + m_size);
66 stdex_assert(size <= (m_head <= tail ? N_cap - tail : m_head - tail));
70 m_tail_moved.notify_one();
80 std::unique_lock<std::mutex> lk(m_mutex);
82 m_tail_moved.wait(lk, [&]{
return m_quit || !empty();});
83 if (m_quit && empty()) _Unlikely_
84 return {
nullptr, 0 };
86 size_t tail = wrap(m_head + m_size);
87 return { &m_data[m_head], m_head < tail ? m_size : N_cap - m_head };
95 void pop(_In_
size_t size)
98 const std::lock_guard<std::mutex> lg(m_mutex);
100 size_t tail = wrap(m_head + m_size);
101 stdex_assert(size <= (m_head < tail ? m_size : N_cap - m_head));
103 m_head = wrap(m_head + size);
106 m_head_moved.notify_one();
115 const std::lock_guard<std::mutex> lg(m_mutex);
118 m_head_moved.notify_one();
119 m_tail_moved.notify_one();
127 std::unique_lock<std::mutex> lk(m_mutex);
128 m_head_moved.wait(lk, [&]{
return m_quit || empty();});
132 size_t wrap(_In_
size_t idx)
const
140 return N_cap - m_size;
150 std::condition_variable m_head_moved, m_tail_moved;
151 size_t m_head, m_size;
158#pragma GCC diagnostic pop
Ring buffer.
Definition ring.hpp:29
void pop(size_t size)
Notifies the sender the data was consumed.
Definition ring.hpp:95
std::tuple< T *, size_t > back()
Allocates the data after the ring tail. Use push() after the allocated data is populated.
Definition ring.hpp:43
void quit()
Cancells waiting sender and receiver.
Definition ring.hpp:112
std::tuple< T *, size_t > front()
Peeks the data at the ring head. Use pop() after the data was consumed.
Definition ring.hpp:78
void push(size_t size)
Notifies the receiver the data was populated.
Definition ring.hpp:60
void sync()
Waits until the ring is flush.
Definition ring.hpp:125