2018-12-03 01:51:44 +01:00
|
|
|
#pragma once
|
|
|
|
|
|
|
|
#include <AK/Types.h>
|
2019-10-18 15:58:06 +02:00
|
|
|
#include <AK/Vector.h>
|
2019-03-16 13:18:22 +01:00
|
|
|
#include <Kernel/Lock.h>
|
2018-12-03 01:51:44 +01:00
|
|
|
|
|
|
|
class DoubleBuffer {
|
|
|
|
public:
|
2019-10-18 15:58:06 +02:00
|
|
|
DoubleBuffer()
|
|
|
|
: m_write_buffer(&m_buffer1)
|
|
|
|
, m_read_buffer(&m_buffer2)
|
|
|
|
{
|
|
|
|
}
|
2018-12-03 01:51:44 +01:00
|
|
|
|
2019-07-03 21:17:35 +02:00
|
|
|
ssize_t write(const u8*, ssize_t);
|
|
|
|
ssize_t read(u8*, ssize_t);
|
2018-12-03 01:51:44 +01:00
|
|
|
|
2019-01-15 21:43:38 +01:00
|
|
|
bool is_empty() const { return m_empty; }
|
2018-12-03 01:51:44 +01:00
|
|
|
|
2019-10-18 15:58:06 +02:00
|
|
|
// FIXME: Isn't this racy? What if we get interrupted between getting the buffer pointer and dereferencing it?
|
|
|
|
ssize_t bytes_in_write_buffer() const { return (ssize_t)m_write_buffer->size(); }
|
2019-01-15 09:17:22 +01:00
|
|
|
|
2018-12-03 01:51:44 +01:00
|
|
|
private:
|
|
|
|
void flip();
|
2019-10-18 15:58:06 +02:00
|
|
|
void compute_emptiness();
|
2018-12-03 01:51:44 +01:00
|
|
|
|
2019-10-18 15:58:06 +02:00
|
|
|
Vector<u8>* m_write_buffer { nullptr };
|
|
|
|
Vector<u8>* m_read_buffer { nullptr };
|
|
|
|
Vector<u8> m_buffer1;
|
|
|
|
Vector<u8> m_buffer2;
|
|
|
|
ssize_t m_read_buffer_index { 0 };
|
2019-01-15 21:43:38 +01:00
|
|
|
bool m_empty { true };
|
2019-10-18 15:58:06 +02:00
|
|
|
Lock m_lock { "DoubleBuffer" };
|
2018-12-03 01:51:44 +01:00
|
|
|
};
|