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.

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.