blob: a4e3490dfd8b6d31025257fcf26732e4a154fd51 [file] [log] [blame]
// Copyright 2020 The Pigweed Authors
//
// Licensed under the Apache License, Version 2.0 (the "License"); you may not
// use this file except in compliance with the License. You may obtain a copy of
// the License at
//
// https://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
// WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
// License for the specific language governing permissions and limitations under
// the License.
#pragma once
#include <cstddef>
#include <span>
#include "pw_containers/intrusive_list.h"
#include "pw_result/result.h"
#include "pw_status/status.h"
namespace pw {
namespace ring_buffer {
// A circular ring buffer for arbitrary length data entries. Each PushBack()
// produces a buffer entry. Each entry consists of a preamble followed by an
// arbitrary length data chunk. The preamble is comprised of an optional user
// preamble byte and an always present varint. The varint encodes the number of
// bytes in the data chunk. This is a FIFO queue, with the oldest entries at
// the 'front' (to be processed by readers) and the newest entries at the 'back'
// (where the writer pushes to).
//
// The ring buffer supports multiple readers, which can be attached/detached
// from the buffer. Each reader has its own read pointer and can peek and pop
// the entry at the head. Entries are not bumped out from the buffer until all
// readers have moved past that entry, or if the buffer is at capacity and space
// is needed to push a new entry. When making space, the buffer will push slow
// readers forward to the new oldest entry. Entries are internally wrapped
// around as needed.
class PrefixedEntryRingBufferMulti {
public:
typedef Status (*ReadOutput)(std::span<const std::byte>);
// A reader that provides a single-reader interface into the multi-reader ring
// buffer it has been attached to via AttachReader(). Readers maintain their
// read position in the ring buffer as well as the remaining count of entries
// from that position.
//
// If no readers are currently attached, the reader starts at the current
// write head. If readers are currently attached, the reader is set to the
// location and entry count of the slowest reader in the set.
//
// Readers can peek and pop entries similar to the single-reader interface.
// When popping entries, although the reader moves forward and drops the
// entry, the entry is not removed from the ring buffer until all other
// attached readers have moved past that entry.
//
// When the attached ring buffer needs to make space, it may push the reader
// index forward. Users of this class should consider the possibility of data
// loss if they read slower than the writer.
class Reader : public IntrusiveList<Reader>::Item {
public:
constexpr Reader() : buffer_(nullptr), read_idx_(0), entry_count_(0) {}
// TODO(pwbug/344): Add locking to the internal functions. Who owns the
// lock? This class? Does this class need a lock if it's not a multi-reader?
// (One doesn't exist today but presumably nothing prevents push + pop
// operations from happening on two different threads).
// Read the oldest stored data chunk of data from the ring buffer to
// the provided destination std::span. The number of bytes read is written
// to bytes_read
//
// Precondition: the buffer data must not be corrupt, otherwise there will
// be a crash.
//
// Return values:
// OK - Data successfully read from the ring buffer.
// FAILED_PRECONDITION - Buffer not initialized.
// OUT_OF_RANGE - No entries in ring buffer to read.
// RESOURCE_EXHAUSTED - Destination data std::span was smaller number of
// bytes than the data size of the data chunk being read. Available
// destination bytes were filled, remaining bytes of the data chunk were
// ignored.
Status PeekFront(std::span<std::byte> data, size_t* bytes_read_out) const {
return buffer_->InternalPeekFront(*this, data, bytes_read_out);
}
Status PeekFront(ReadOutput output) const {
return buffer_->InternalPeekFront(*this, output);
}
// Peek the front entry's preamble only to avoid copying data unnecessarily.
//
// Precondition: the buffer data must not be corrupt, otherwise there will
// be a crash.
Status PeekFrontPreamble(uint32_t& user_preamble_out) const {
return buffer_->InternalPeekFrontPreamble(*this, user_preamble_out);
}
// Same as PeekFront but includes the entry's preamble of optional user
// value and the varint of the data size.
// TODO(pwbug/341): Move all other APIs to passing bytes_read by reference,
// as it is required to determine the length populated in the span.
Status PeekFrontWithPreamble(std::span<std::byte> data,
uint32_t& user_preamble_out,
size_t& entry_bytes_read_out) const;
Status PeekFrontWithPreamble(std::span<std::byte> data,
size_t* bytes_read_out) const {
return buffer_->InternalPeekFrontWithPreamble(
*this, data, bytes_read_out);
}
Status PeekFrontWithPreamble(ReadOutput output) const {
return buffer_->InternalPeekFrontWithPreamble(*this, output);
}
// Pop and discard the oldest stored data chunk of data from the ring
// buffer.
//
// Precondition: the buffer data must not be corrupt, otherwise there will
// be a crash.
//
// Return values:
// OK - Data successfully read from the ring buffer.
// FAILED_PRECONDITION - Buffer not initialized.
// OUT_OF_RANGE - No entries in ring buffer to pop.
Status PopFront() { return buffer_->InternalPopFront(*this); }
// Get the size in bytes of the next chunk, not including preamble, to be
// read.
//
// Precondition: the buffer data must not be corrupt, otherwise there will
// be a crash.
size_t FrontEntryDataSizeBytes() const {
return buffer_->InternalFrontEntryDataSizeBytes(*this);
}
// Get the size in bytes of the next chunk, including preamble and data
// chunk, to be read.
//
// Precondition: the buffer data must not be corrupt, otherwise there will
// be a crash.
size_t FrontEntryTotalSizeBytes() const {
return buffer_->InternalFrontEntryTotalSizeBytes(*this);
}
// Get the number of variable-length entries currently in the ring buffer.
//
// Return value:
// Entry count.
size_t EntryCount() const { return entry_count_; }
private:
friend PrefixedEntryRingBufferMulti;
// Internal constructors for the iterator class to create Reader instances
// at specific positions. Readers constructed through this interface cannot
// be attached/detached from the multisink.
constexpr Reader(Reader& reader)
: Reader(reader.buffer_, reader.read_idx_, reader.entry_count_) {}
constexpr Reader(PrefixedEntryRingBufferMulti* buffer,
size_t read_idx,
size_t entry_count)
: buffer_(buffer), read_idx_(read_idx), entry_count_(entry_count) {}
PrefixedEntryRingBufferMulti* buffer_;
size_t read_idx_;
size_t entry_count_;
};
// An entry returned by the iterator containing the byte span of the entry
// and preamble data (if the ring buffer was configured with a preamble).
struct Entry {
std::span<const std::byte> buffer;
uint32_t preamble;
};
// An iterator that can be used to walk through all entries from a given
// Reader position, without mutating the underlying buffer. This is useful in
// crash contexts where all available entries in the buffer must be acquired,
// even those that have already been consumed by all attached readers.
class iterator {
public:
iterator() : ring_buffer_(nullptr), read_idx_(0), entry_count_(0) {}
iterator(Reader& reader)
: ring_buffer_(reader.buffer_),
read_idx_(0),
entry_count_(reader.entry_count_) {
Status dering_result = ring_buffer_->InternalDering(reader);
PW_DASSERT(dering_result.ok());
}
iterator& operator++();
iterator operator++(int) {
iterator original = *this;
++*this;
return original;
}
// Returns entry at current position.
const Entry& operator*() const;
const Entry* operator->() const { return &operator*(); }
constexpr bool operator==(const iterator& rhs) const {
return entry_count_ == rhs.entry_count_;
}
constexpr bool operator!=(const iterator& rhs) const {
return entry_count_ != rhs.entry_count_;
}
// Returns the status of the last iteration operation. If the iterator
// fails to read an entry, it will move to iterator::end() and indicate
// the failure reason here.
Status status() const { return iteration_status_; }
private:
static constexpr Entry kEndEntry = {
.buffer = std::span<const std::byte>(),
.preamble = 0,
};
void SkipToEnd(Status status) {
iteration_status_ = status;
entry_ = kEndEntry;
entry_count_ = 0;
}
PrefixedEntryRingBufferMulti* ring_buffer_;
size_t read_idx_;
size_t entry_count_;
mutable Entry entry_;
Status iteration_status_;
};
using element_type = const Entry;
using value_type = std::remove_cv_t<const Entry>;
using pointer = const Entry;
using reference = const Entry&;
using const_iterator = iterator; // Standard alias for iterable types.
iterator begin() { return iterator(GetSlowestReaderWritable()); }
iterator end() { return iterator(); }
const_iterator cbegin() { return begin(); }
const_iterator cend() { return end(); }
// TODO(pwbug/340): Consider changing bool to an enum, to explicitly enumerate
// what this variable means in clients.
PrefixedEntryRingBufferMulti(bool user_preamble = false)
: buffer_(nullptr),
buffer_bytes_(0),
write_idx_(0),
user_preamble_(user_preamble) {}
// Set the raw buffer to be used by the ring buffer.
//
// Return values:
// OK - successfully set the raw buffer.
// INVALID_ARGUMENT - Argument was nullptr, size zero, or too large.
Status SetBuffer(std::span<std::byte> buffer);
// Determines if the ring buffer has corrupted entries.
//
// Precondition: At least one reader must be attached to the ring buffer.
// Return values:
// OK - No corruption was detected.
// DATA_LOSS - Corruption was detected.
Status CheckForCorruption() {
iterator it = begin();
for (; it != end(); ++it) {
}
return it.status();
}
// Attach reader to the ring buffer. Readers can only be attached to one
// ring buffer at a time.
//
// Return values:
// OK - Successfully configured reader for ring buffer.
// INVALID_ARGUMENT - Argument was already attached to another ring buffer.
Status AttachReader(Reader& reader);
// Detach reader from the ring buffer. Readers can only be detached if they
// were previously attached.
//
// Return values:
// OK - Successfully removed reader for ring buffer.
// INVALID_ARGUMENT - Argument was not previously attached to this ring
// buffer.
Status DetachReader(Reader& reader);
// Removes all data from the ring buffer.
void Clear();
// Write a chunk of data to the ring buffer. If available space is less than
// size of data chunk to be written then silently pop and discard oldest
// stored data chunks until space is available.
//
// Preamble argument is a caller-provided value prepended to the front of the
// entry. It is only used if user_preamble was set at class construction
// time. It is varint-encoded before insertion into the buffer.
//
// Return values:
// OK - Data successfully written to the ring buffer.
// FAILED_PRECONDITION - Buffer not initialized.
// OUT_OF_RANGE - Size of data is greater than buffer size.
Status PushBack(std::span<const std::byte> data,
uint32_t user_preamble_data = 0) {
return InternalPushBack(data, user_preamble_data, true);
}
// [Deprecated] An implementation of PushBack that accepts a single-byte as
// preamble data. Clients should migrate to passing uint32_t preamble data.
Status PushBack(std::span<const std::byte> data,
std::byte user_preamble_data) {
return PushBack(data, static_cast<uint32_t>(user_preamble_data));
}
// Write a chunk of data to the ring buffer if there is space available.
//
// Preamble argument is a caller-provided value prepended to the front of the
// entry. It is only used if user_preamble was set at class construction
// time. It is varint-encoded before insertion into the buffer.
//
// Precondition: the buffer data must not be corrupt, otherwise there will
// be a crash.
//
// Return values:
// OK - Data successfully written to the ring buffer.
// INVALID_ARGUMENT - Size of data to write is zero bytes
// FAILED_PRECONDITION - Buffer not initialized.
// OUT_OF_RANGE - Size of data is greater than buffer size.
// RESOURCE_EXHAUSTED - The ring buffer doesn't have space for the data
// without popping off existing elements.
Status TryPushBack(std::span<const std::byte> data,
uint32_t user_preamble_data = 0) {
return InternalPushBack(data, user_preamble_data, false);
}
// [Deprecated] An implementation of TryPushBack that accepts a single-byte as
// preamble data. Clients should migrate to passing uint32_t preamble data.
Status TryPushBack(std::span<const std::byte> data,
std::byte user_preamble_data) {
return TryPushBack(data, static_cast<uint32_t>(user_preamble_data));
}
// Get the size in bytes of all the current entries in the ring buffer,
// including preamble and data chunk.
size_t TotalUsedBytes() const { return buffer_bytes_ - RawAvailableBytes(); }
// Dering the buffer by reordering entries internally in the buffer by
// rotating to have the oldest entry is at the lowest address/index with
// newest entry at the highest address. If no readers are attached, the buffer
// is deringed at the current write index.
//
// Return values:
// OK - Buffer data successfully deringed.
// FAILED_PRECONDITION - Buffer not initialized.
Status Dering();
private:
// Read the oldest stored data chunk of data from the ring buffer to
// the provided destination std::span. The number of bytes read is written to
// `bytes_read_out`.
//
// Precondition: the buffer data must not be corrupt, otherwise there will
// be a crash.
//
// Return values:
// OK - Data successfully read from the ring buffer.
// FAILED_PRECONDITION - Buffer not initialized.
// OUT_OF_RANGE - No entries in ring buffer to read.
// RESOURCE_EXHAUSTED - Destination data std::span was smaller number of bytes
// than the data size of the data chunk being read. Available destination
// bytes were filled, remaining bytes of the data chunk were ignored.
Status InternalPeekFront(const Reader& reader,
std::span<std::byte> data,
size_t* bytes_read_out) const;
Status InternalPeekFront(const Reader& reader, ReadOutput output) const;
Status InternalPeekFrontPreamble(const Reader& reader,
uint32_t& user_preamble_out) const;
// Same as Read but includes the entry's preamble of optional user value and
// the varint of the data size
Status InternalPeekFrontWithPreamble(const Reader& reader,
std::span<std::byte> data,
size_t* bytes_read_out) const;
Status InternalPeekFrontWithPreamble(const Reader& reader,
ReadOutput output) const;
// Pop and discard the oldest stored data chunk of data from the ring buffer.
//
// Precondition: the buffer data must not be corrupt, otherwise there will
// be a crash.
//
// Return values:
// OK - Data successfully read from the ring buffer.
// FAILED_PRECONDITION - Buffer not initialized.
// OUT_OF_RANGE - No entries in ring buffer to pop.
Status InternalPopFront(Reader& reader);
// Get the size in bytes of the next chunk, not including preamble, to be
// read.
size_t InternalFrontEntryDataSizeBytes(const Reader& reader) const;
// Get the size in bytes of the next chunk, including preamble and data
// chunk, to be read.
size_t InternalFrontEntryTotalSizeBytes(const Reader& reader) const;
// Internal version of Read used by all the public interface versions. T
// should be of type ReadOutput.
template <typename T>
Status InternalRead(const Reader& reader,
T read_output,
bool include_preamble_in_output,
uint32_t* user_preamble_out = nullptr) const;
// Dering the buffer by reordering entries internally in the buffer by
// rotating to have the oldest entry is at the lowest address/index with
// newest entry at the highest address. If no readers are attached, the buffer
// is deringed at the current write index.
//
// Return values:
// OK - Buffer data successfully deringed.
// FAILED_PRECONDITION - Buffer not initialized.
Status InternalDering(Reader& reader);
struct EntryInfo {
size_t preamble_bytes;
uint32_t user_preamble;
size_t data_bytes;
};
// Push back implementation, which optionally discards front elements to fit
// the incoming element.
Status InternalPushBack(std::span<const std::byte> data,
uint32_t user_preamble_data,
bool pop_front_if_needed);
// Internal function to pop all of the slowest readers. This function may pop
// multiple readers if multiple are slow.
//
// Precondition: This function requires that at least one reader is attached
// and has at least one entry to pop. There will be a crash if data is
// corrupted.
void InternalPopFrontAll();
// Returns a the slowest reader in the list.
//
// Precondition: This function requires that at least one reader is attached.
const Reader& GetSlowestReader() const;
Reader& GetSlowestReaderWritable() {
return const_cast<Reader&>(GetSlowestReader());
}
// Get info struct with the size of the preamble and data chunk for the next
// entry to be read. Calls RawFrontEntryInfo and asserts on failure.
//
// Precondition: the buffer data must not be corrupt, otherwise there will
// be a crash.
EntryInfo FrontEntryInfo(const Reader& reader) const;
// Get info struct with the size of the preamble and data chunk for the next
// entry to be read.
//
// Returns:
// OK - EntryInfo containing the next entry metadata.
// DATA_LOSS - Failed to read the metadata at this location.
Result<EntryInfo> RawFrontEntryInfo(size_t source_idx) const;
// Get the raw number of available bytes free in the ring buffer. This is
// not available bytes for data, since there is a variable size preamble for
// each entry.
size_t RawAvailableBytes() const;
// Do the basic write of the specified number of bytes starting at the last
// write index of the ring buffer to the destination, handing any wrap-around
// of the ring buffer. This is basic, raw operation with no safety checks.
void RawWrite(std::span<const std::byte> source);
// Do the basic read of the specified number of bytes starting at the given
// index of the ring buffer to the destination, handing any wrap-around of
// the ring buffer. This is basic, raw operation with no safety checks.
void RawRead(std::byte* destination,
size_t source_idx,
size_t length_bytes) const;
size_t IncrementIndex(size_t index, size_t count) const;
std::byte* buffer_;
size_t buffer_bytes_;
size_t write_idx_;
const bool user_preamble_;
// List of attached readers.
IntrusiveList<Reader> readers_;
// Maximum bufer size allowed. Restricted to this to allow index aliasing to
// not overflow.
static constexpr size_t kMaxBufferBytes =
std::numeric_limits<size_t>::max() / 2;
};
class PrefixedEntryRingBuffer : public PrefixedEntryRingBufferMulti,
public PrefixedEntryRingBufferMulti::Reader {
public:
PrefixedEntryRingBuffer(bool user_preamble = false)
: PrefixedEntryRingBufferMulti(user_preamble) {
AttachReader(*this)
.IgnoreError(); // TODO(pwbug/387): Handle Status properly
}
};
} // namespace ring_buffer
} // namespace pw