Better ringbuffer implementation.
This commit is contained in:
parent
e01d1c2905
commit
f1e9c36f00
|
@ -9,59 +9,65 @@
|
||||||
|
|
||||||
#include "xenia/base/ring_buffer.h"
|
#include "xenia/base/ring_buffer.h"
|
||||||
|
|
||||||
|
#include <algorithm>
|
||||||
#include <cstring>
|
#include <cstring>
|
||||||
|
|
||||||
namespace xe {
|
namespace xe {
|
||||||
|
|
||||||
RingBuffer::RingBuffer(uint8_t *raw_buffer, size_t size, size_t write_offset)
|
RingBuffer::RingBuffer(uint8_t* raw_buffer, size_t size, size_t read_offset, size_t write_offset)
|
||||||
: raw_buffer_(raw_buffer),
|
: raw_buffer_(raw_buffer)
|
||||||
size_(size),
|
, size_(size)
|
||||||
write_offset_(write_offset) {}
|
, read_offset_(read_offset)
|
||||||
|
, write_offset_(write_offset) {}
|
||||||
|
|
||||||
size_t RingBuffer::Write(uint8_t *buffer, size_t num_bytes) {
|
size_t RingBuffer::Skip(size_t num_bytes) {
|
||||||
size_t bytes_written = 0;
|
num_bytes = std::min(read_size(), num_bytes);
|
||||||
size_t input_offset = 0;
|
if (read_offset_ + num_bytes < size_) {
|
||||||
size_t bytes_to_write = 0;
|
read_offset_ += num_bytes;
|
||||||
|
} else {
|
||||||
// write offset -> end
|
read_offset_ = num_bytes - (size_ - read_offset_);
|
||||||
bytes_to_write =
|
|
||||||
num_bytes < size_ - write_offset_ ? num_bytes : size_ - write_offset_;
|
|
||||||
|
|
||||||
std::memcpy(raw_buffer_ + write_offset_, buffer, bytes_to_write);
|
|
||||||
input_offset = bytes_to_write;
|
|
||||||
write_offset_ += bytes_to_write;
|
|
||||||
|
|
||||||
bytes_written += bytes_to_write;
|
|
||||||
|
|
||||||
// Wraparound (begin -> num_bytes)
|
|
||||||
if (input_offset < num_bytes) {
|
|
||||||
bytes_to_write = num_bytes - input_offset;
|
|
||||||
|
|
||||||
std::memcpy(raw_buffer_, buffer + input_offset, bytes_to_write);
|
|
||||||
write_offset_ = bytes_to_write;
|
|
||||||
|
|
||||||
bytes_written += bytes_to_write;
|
|
||||||
}
|
}
|
||||||
|
return num_bytes;
|
||||||
return bytes_written;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
size_t RingBuffer::DistanceToOffset(size_t offset) {
|
size_t RingBuffer::Read(uint8_t* buffer, size_t num_bytes) {
|
||||||
// Make sure the offset is in range.
|
num_bytes = std::min(read_size(), num_bytes);
|
||||||
if (offset > size_) {
|
if (!num_bytes) {
|
||||||
offset %= size_;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (offset < size_ && offset >= write_offset_) {
|
if (read_offset_ + num_bytes < size_) {
|
||||||
// Doesn't wraparound.
|
std::memcpy(buffer, raw_buffer_ + read_offset_, num_bytes);
|
||||||
return offset - write_offset_;
|
read_offset_ += num_bytes;
|
||||||
} else {
|
} else {
|
||||||
// Wraparound.
|
size_t left_half = size_ - read_offset_;
|
||||||
size_t dist = size_ - write_offset_;
|
size_t right_half = size_ - left_half;
|
||||||
dist += offset;
|
std::memcpy(buffer, raw_buffer_ + read_offset_, left_half);
|
||||||
|
std::memcpy(buffer + left_half, raw_buffer_, right_half);
|
||||||
return dist;
|
read_offset_ = right_half;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return num_bytes;
|
||||||
|
}
|
||||||
|
|
||||||
|
size_t RingBuffer::Write(uint8_t* buffer, size_t num_bytes) {
|
||||||
|
num_bytes = std::min(num_bytes, write_size());
|
||||||
|
if (!num_bytes) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (write_offset_ + num_bytes < size_) {
|
||||||
|
std::memcpy(raw_buffer_ + write_offset_, buffer, num_bytes);
|
||||||
|
write_offset_ += num_bytes;
|
||||||
|
} else {
|
||||||
|
size_t left_half = size_ - write_offset_;
|
||||||
|
size_t right_half = size_ - left_half;
|
||||||
|
std::memcpy(raw_buffer_ + write_offset_, buffer, left_half);
|
||||||
|
std::memcpy(raw_buffer_, buffer + left_half, right_half);
|
||||||
|
write_offset_ = right_half;
|
||||||
|
}
|
||||||
|
|
||||||
|
return num_bytes;
|
||||||
}
|
}
|
||||||
|
|
||||||
} // namespace xe
|
} // namespace xe
|
||||||
|
|
|
@ -18,18 +18,39 @@ namespace xe {
|
||||||
|
|
||||||
class RingBuffer {
|
class RingBuffer {
|
||||||
public:
|
public:
|
||||||
RingBuffer(uint8_t *raw_buffer, size_t size, size_t write_offset = 0);
|
RingBuffer(uint8_t* raw_buffer, size_t size, size_t read_offset, size_t write_offset);
|
||||||
|
|
||||||
size_t Write(uint8_t *buffer, size_t num_bytes);
|
size_t Read(uint8_t* buffer, size_t num_bytes);
|
||||||
|
size_t Skip(size_t num_bytes);
|
||||||
|
size_t Write(uint8_t* buffer, size_t num_bytes);
|
||||||
|
|
||||||
size_t DistanceToOffset(size_t offset);
|
size_t read_offset() { return read_offset_; }
|
||||||
|
|
||||||
void set_write_offset(size_t write_offset) { write_offset_ = write_offset; }
|
|
||||||
size_t write_offset() { return write_offset_; }
|
size_t write_offset() { return write_offset_; }
|
||||||
|
|
||||||
|
size_t read_size() {
|
||||||
|
if (read_offset_ == write_offset_) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
if (read_offset_ < write_offset_) {
|
||||||
|
return write_offset_ - read_offset_;
|
||||||
|
}
|
||||||
|
return (size_ - read_offset_) + write_offset_;
|
||||||
|
}
|
||||||
|
|
||||||
|
size_t write_size() {
|
||||||
|
if (write_offset_ == read_offset_) {
|
||||||
|
return size_;
|
||||||
|
}
|
||||||
|
if (write_offset_ < read_offset_) {
|
||||||
|
return read_offset_ - write_offset_;
|
||||||
|
}
|
||||||
|
return (size_ - write_offset_) + read_offset_;
|
||||||
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
uint8_t *raw_buffer_;
|
uint8_t* raw_buffer_;
|
||||||
size_t size_;
|
size_t size_;
|
||||||
|
size_t read_offset_;
|
||||||
size_t write_offset_;
|
size_t write_offset_;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue