BiP Buffer Made Easy
After reading through the article by Simon Cooke on BiP buffer I thought the idea expressed therein was a great one. However after thinking about it for a while, I realized that similar functionality could be achieved with considerably less state information as used in the article. This article deals with a modification of the idea of BiP buffer that is, in my opinion, way easier to understand and also to implement.
The main idea behind this modification is the realization that the BiP buffer is essentially just a regular ring buffer, whose capacity can be temporarily shortened as needed. This implementation requires, besides the buffer itself, only three integers, two of which are part of the regular ring buffer - head and tail pointers. The one additional state information is the currently active capacity of the buffer, as shown in the class definition below.
The details of the implementation will be discussed below the class definition.
class BipBuffer
{
public:
static constexpr std::size_t DEFAULT_CAPACITY = 4096;
BipBuffer(std::size_t capacity = DEFAULT_CAPACITY):
active_capacity_(capacity)
{
buffer_.resize(capacity);
}
bool isEmpty() const
{
return head_ == tail_;
}
void changeChapacity(std::size_t size)
{
buffer_.resize(size);
clear();
}
void clear()
{
head_ = 0; tail_ = 0;
active_capacity_ = buffer_.size();
}
std::pair<char *, std::size_t> reserve(std::size_t requested_size);
void commit(std::size_t commit_size);
std::pair<const char *, std::size_t> getContiguousBlock();
void decommit(std::size_t size_to_decommit);
private:
std::vector<char> buffer_;
std::size_t head_ = 0;
std::size_t tail_ = 0;
std::size_t active_capacity_ = 0;
bool isContiguous() const
{
return head_ >= tail_;
}
std::size_t spaceAhead() const
{
if (isContiguous())
return buffer_.size() - head_ - (0 == tail_ ? 1 : 0);
else
return (tail_ - head_ - 1);
}
std::size_t spaceBehind() const
{
if (isContiguous())
return 0 == tail_ ? 0 : (tail_ - 1);
else
return 0;
}
};
Before jumping in to the details of the individual methods, it is necessary to clear out several concepts used in this implementation.
- Contiguous state - state of the ring buffer, where all data in the buffer are stored in one contiguous region of memory, i.e. the ring buffer is not wrapped
- Space ahead - free space ahead of the contiguous data in the buffer
- Space behind - free space behind the contiguous data in the buffer
Reserve
First the code will check whether there is enough free space in front of the ring buffer's head to satisfy the requirements. If that is not the case, it will further try to check if more space can be found behind the committed data (in case the data is in contiguous state). If that is the case the remaining free space ahead of the data in the buffer is marked as temporarily deactivated and the head of the ring buffer is wrapped to the beginning of the buffer.
std::pair<char *, std::size_t> BipBuffer::reserve(std::size_t requested_size)
{
const std::size_t space_ahead = spaceAhead();
if (space_ahead < requested_size)
{
const std::size_t space_behind = spaceBehind();
if (space_behind > space_ahead)
{
active_capacity_ = head_;
head_ = 0;
return {buffer_.data(), min(space_behind, requested_size)};
}
}
return {buffer_.data() + head_, min(space_ahead, requested_size)};
}
Commit
Commit operation simply advances the head of the ring buffer. There is no need to treat the situation where the buffer's capacity is temporarily shortened, as that was done during reservation of the buffer space.
void BipBuffer::commit(std::size_t commit_size)
{
head_ += min(commit_size, spaceAhead());
}
Peek
To peek at the contiguous data block stored in the buffer, the code will simply look at the tail of the ring buffer, however the length of the area needs to be determined based on the state of the ring buffer.
std::pair<const char *, std::size_t> BipBuffer::getContiguousBlock()
{
if (isContiguous())
return {buffer_.data() + tail_, head_ - tail_};
else
return {buffer_.data() + tail_, active_capacity_ - tail_};
}
De-commit
Generally the decommission of data from the buffer is done by advancing the tail of the ring buffer (as in regular ring buffer), however the new tail of the buffer may bump into two barriers: either the end of the active area of the buffer or the head of the buffer. The former case extends the capacity of the underlying buffer back to its full size, undoing any shortening that might have been done. The latter case happens when the buffer is fully emptied. In this case, both head and tail are reset to the beginning of the buffer, maximizing the amount of space that can be reserved.
void BipBuffer::decommit(std::size_t size_to_decommit)
{
const std::size_t new_tail = tail_ + size_to_decommit;
if (isContiguous())
{
if (new_tail < head_)
tail_ = new_tail;
else
{
head_ = 0;
tail_ = 0;
}
}
else
{
if (new_tail < active_capacity_)
tail_ = new_tail;
else
{
tail_ = 0;
active_capacity_ = buffer_.size();
}
}
}
Notes
This implementation made several relaxations to the design in the original
article. Most notably the reserve()
and getContiguousBlock()
methods do not
return nullptr
in case the operations fail. This is done, because this is not
really necessary, as it would be redundant and it would add additional branches
to this implementation. Secondly this implementation does not store the
information on the amount of reserved space.