Create a ring buffer, backed by a ring_buffer_bytes
,
where each element corresponds to a fixed-size vector of one of
R's atomic numeric types (logical, integer, double, and complex).
Arguments
- size
The maximum number of elements the buffer can hold. Each element will be multiple bytes long.
- what
Either a vector on the style of
vapply
(e.g.,integer(4)
to indicate that each element of the buffer is a 4-element integer, or thename
of a storage mode iflen
is also provided.- len
If given, then the length of the storage. If it is given, then if
length(what)
is zero, the storage mode ofwhat
is used as the type. Otherwisewhat
is interpreted as the name of the storage mode (one of "logical", "integer", "double" or "complex".- on_overflow
Behaviour on buffer overflow. The default is to overwrite the oldest elements in the buffer (
"overwrite"
). Alternative actions are"error"
which will throw an error if a function tries to add more elements than there are space for, or"grow"
which will grow the buffer to accept the new elements (this uses an approximately golden ratio approach; see details below).
Details
Note that a logical ring buffer and an integer ring buffer take the same number of bytes because a logical vector is stored as an integer (4 bytes per element) to deal with missing values; see "writing R extensions".
Note that it is not possible to store character vectors in a ring buffer of this type because each element of a character vector can be any number of bytes.
Methods
Note that this methods reference section is repeated verbatim between
the three main ring buffer classes; ring_buffer_env
("env"), ring_buffer_bytes
("bytes") and
ring_buffer_bytes_typed
("typed"). Almost all methods have
the same arguments and behaviour, but hopefully by listing everything together,
the differences between implementations will be a bit more apparent.
reset
Reset the state of the buffer. This "zeros" the head and tail pointer (and may or may not actually reset the data) so that the buffer can be used as if fresh.
Usage:
reset(clear = FALSE)
Arguments:
clear
: Logical, indicating if the memory should also be cleared. Generally this is not necessary, but with environment buffers this can let the garbage collector clean up large elements. For the bytes buffer this zeros the memory.
Return value: Nothing; called for the side effect only.
duplicate
Clone the ring buffer, creating a copy. Copies both the underlying data and the position of the head and tail.
Usage:
duplicate()
Return value: A new ring buffer object
grow
Increase the size of the buffer by
n
elements.Usage:
bytes, typed:
grow(n)
env:
grow(n, exact = FALSE)
Arguments:
n
: The number of additional elements that space should be reserved for (scalar non-negative integer).
Return value: Nothing; called for the side effect only.
size
Return the capacity (maximum size) of the ring buffer
Usage:
env:
size()
bytes, typed:
size(bytes = FALSE)
Arguments:
bytes
: (forring_buffer_bytes
only) Logical, indicating if the size should be returned in bytes (rather than logical entries, which is the default).
Return value: A scalar integer
bytes_data
Return the total size of the data storage used in this object.
Usage:
env: (not supported)
bytes, typed:
bytes_data()
Return value: A scalar integer
stride
Length of each element in the ring buffer, in bytes. Only implemented (and meaningful) for the bytes buffer; the environment buffer does not support this function as it makes no sense there.
Usage:
env: (not supported)
bytes, typed:
stride()
Return value: A scalar integer
used
Return the amount of space used in the ring buffer.
Usage:
env:
used()
bytes, typed:
used(bytes = FALSE)
Arguments:
bytes
: (forring_buffer_bytes
only) Logical, indicating if the size should be returned in bytes (rather than logical entries, which is the default).
Return value: A scalar integer
free
Return the amount of space free in the ring buffer.
Usage:
env:
free()
bytes, typed:
free(bytes = FALSE)
Arguments:
bytes
: (forring_buffer_bytes
only) Logical, indicating if the size should be returned in bytes (rather than logical entries, which is the default).
Return value: A scalar integer
is_empty
Test if the ring buffer is empty
Usage:
is_empty()
Return value: A scalar logical
is_full
Test if the ring buffer is full
Usage:
is_full()
Return value: A scalar logical
head_pos
Return the number of entries from the "start" of the ring buffer the head is. This is mostly useful for debugging.
Usage:
env:
head_pos()
bytes, typed:
head_pos(bytes = FALSE)
Arguments:
bytes
: (forring_buffer_bytes
only) Logical, indicating if the position should be returned in bytes (rather than logical entries, which is the default).
Return value: A scalar integer
tail_pos
Return the number of entries from the "start" of the ring buffer the tail is. This is mostly useful for debugging.
Usage:
env:
tail_pos()
bytes, typed:
tail_pos(bytes = FALSE)
Arguments:
bytes
: (forring_buffer_bytes
only) Logical, indicating if the position should be returned in bytes (rather than logical entries, which is the default).
Return value: A scalar integer
head
Return the contents of the head (the most recently written element in the ring buffer).
Usage:
head()
Return value: It depends a little here. For
ring_buffer_env
this is a single R object. Forring_buffer_bytes
it is a raw vector, the same length as the stride of the ring buffer. Forring_buffer_bytes_typed
, a single R object that has been translated from raw.tail
Return the contents of the tail (the least recently written element in the ring buffer).
Usage:
tail()
Return value: As for
head
set
Set a number of ring entries to the same value. The exact behaviour here varies depending on the type of ring buffer. This function may overflow the ring buffer; in this case the tail will be moved.
Usage:
set(data, n)
Arguments:
data
: The data to set each ring element to. For an environment buffer, this may be any R object. For a bytes buffer it may be either a single byte (in which case each ring element will be set to that byte, repeatedstride
times), or a raw vector of lengthstride
.
Return value: Invisibly returns the number of elements actually written (which may be less than
n
if the buffer overflows). Primarily called for its side effect.push
Push elements onto the ring buffer head. This may overflow the ring buffer, destroying the oldest elements in the buffer (and moving the position of the tail).
Usage:
env:
push(data, iterate = TRUE)
bytes, typed:
push(data)
Arguments:
data
: Data to push onto the ring buffer. Forring_buffer_bytes
, this must be a raw vector with a length that is a multiple of the buffer stride. Forring_buffer_bytes_typed
it must be a vector of the appropriate type. Forring_buffer_env
it may be an arbitrary R object (but seeiterate
.\item{\code{iterate}: For \code{ring_buffer_env} only, changes the behaviour with vectors and lists. Because each element of a \code{ring_buffer_env} can b an arbitrary R object, for a list \code{x} it is ambiguous if \code{push(x)} should push one object onto the buffer, or \code{length(x)} objects (i.e. equivalent to \code{push(x[[1]])}, \code{push(x[[2]])}, etc. The \code{iterate} argument switches between interpretations; if \code{TRUE} (the default) the push will iterate over the object using \code{for (el in x)} (with appropriate S3 dispatch). If \code{iterate = FALSE}, then the entire object is pushed at once, so always updating only by a single element. }
Return value: For
ring_buffer_bytes
, the data invisibly. Forring_buffer_bytes
andring_buffer_bytes_typed
, the position of the head pointer (relative to the beginning of the storage region).take
Destructively take elements from the ring buffer. This consumes from the tail (the least recently added elements). It is not possibly to underflow the buffer; if more elements are requested than can be supplied then an error will be thrown and the state of the buffer unmodified.
Usage:
take(n)
Arguments:
n
: The number of elements to take.
Return value: For
ring_buffer_env
alist
ofn
elements. Forring_buffer_bytes
, a raw vector ofn * stride
bytes. Forring_buffer_bytes_typed
, an vector ofn
elements of the storage mode of the ring.read
Nondestructively read elements from the ring buffer. This is identical to
take
except that the state of the buffer is not modified.Usage:
read(n)
Arguments:
n
: The number of elements to read.
Return value: For
ring_buffer_env
alist
ofn
elements. Forring_buffer_bytes
, a raw vector ofn * stride
bytes. Forring_buffer_bytes_typed
, an vector ofn
elements of the storage mode of the ring.copy
Copy from this ring buffer into a different ring buffer. This is destructive with respect to both ring buffers; the tail pointer will be moved in this ring buffer as data are taken, and if the destination ring buffer overflows, the tail pointer will be moved too.
Usage:
copy(dest, n)
Arguments:
dest
: The destination ring buffer - will be modified by this call.
mirror
Mirror the contents of this ring buffer into a different ring buffer. This differs from
copy
in that this ring buffer is unaffected and in that all of this ring buffer is copied over (including head/tail positions). This provides an alternative way of duplicating state toduplicate
if you already have an appropriately sized ring buffer handy. No allocations will be done.Usage:
mirror(dest)
Arguments:
dest
: The destination ring buffer - will be modified by this call.
Return value: Nothing; called for the side effect only.
head_offset
Nondestructively read the contents of the
head
of the buffer, offset byn
entries.Usage:
head_offset(n)
Arguments:
n
: Head offset. This moves away from the most recently added item. An offset of 0 reads the most recently added element, 1 reads the element added before that.
Return value: As for
head
tail_offset
Nondestructively read the contents of the
tail
of the buffer, offset byn
entries.Usage:
tail_offset(n)
Arguments:
n
: Tail offset. This moves away from the oldest item. An offset of 0 reads the oldest element, 1 reads the element added after that.
Return value: As for
tail
(seehead
)take_head
As for
take
, but operating on the head rather than the tail. This is destructive with respect to the head.Usage:
take_head(n)
Arguments:
n
: Number of elements to take.
Return value: As for
take
read_head
As for
read
, but operating on the head rather than the tail. This is not destructive with respect to the tail.Usage:
read_head(n)
Arguments:
n
: Number of elements to read.
Return value: As for
read
head_set
Set data to the head without advancing. This is useful in cases where the head data will be set and advanced separately (with
head_advance
). This is unlikely to be useful for all users. It is used extensively in dde (but called from C).Usage:
head_set(data)
Arguments:
data
: Data to set into the head. For the bytes buffer this must be exactlystride
bytes long, and for the environment buffer it corresponds to a single "element".
Return value: Nothing; called for the side effect only.
head_data
Retrieve the current data stored in the head but not advanced. For many cases this may be junk - if the byte buffer has looped then it will be the bytes that will be overwritten on the next write. However, when using
head_set
it will be the data that have been set into the buffer but not yet committed withhead_advance
.Usage:
head_data()
Return value: As for
head
head_advance
Shift the head around one position. This commits any data written by
head_set
.Usage:
head_advance()
Return value: Nothing; called for the side effect only.
Examples
# Create a ring buffer of 30 integers:
b <- ring_buffer_bytes_typed(30, integer(1))
# Alternatively you can create the same buffer this way:
b <- ring_buffer_bytes_typed(30, "integer", 1)
# The buffer is empty to start with
b$is_empty()
#> [1] TRUE
# Note that the buffer has a stride of 4 (see ?ring_buffer_bytes)
b$stride()
#> [1] 4
# Push some numbers into the buffer:
b$push(as.integer(1:10))
# Report the number of elements used:
b$used()
#> [1] 10
# Get the first added element:
b$tail()
#> [1] 1
# The buffer behaves basically the same way now as
# "ring_buffer_env" but will typecheck all inputs:
if (FALSE) { # \dontrun{
b$push(pi) # error because not an integer
b$push(1) # error because not an integer (you must convert to int)
} # }
# Recycling: the typed buffer operates by converting the input
# vector to a set of bytes and then pushing them onto the buffer;
# this works so long as the vector of bytes has the correct
# length.
b <- ring_buffer_bytes_typed(30, integer(3))
# These both fail because 2 and 4 do not end up as multiples of 3:
if (FALSE) { # \dontrun{
b$push(c(1L, 2L))
b$push(c(1L, 2L, 3L, 4L))
} # }
# But this is fine:
b$push(seq_len(6))
b$tail()
#> [1] 1 2 3
b$tail_offset(1)
#> [1] 4 5 6