Skip to content

Commit

Permalink
benchmarks in the readme
Browse files Browse the repository at this point in the history
  • Loading branch information
jdonszelmann committed Jun 9, 2023
1 parent 796ac90 commit 12fed3e
Show file tree
Hide file tree
Showing 3 changed files with 48 additions and 4 deletions.
30 changes: 28 additions & 2 deletions README.md
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
# Ringbuffer

![Github Workflows](https://img.shields.io/github/actions/workflow/status/NULLx76/ringbuffer/rust.yml?style=for-the-badge)
[![Docs.rs](https://img.shields.io/badge/docs.rs-ringbuffer-66c2a5?style=for-the-badge&labelColor=555555&logoColor=white&logo=)](https://docs.rs/ringbuffer)
[![Crates.io](https://img.shields.io/crates/v/ringbuffer?logo=rust&style=for-the-badge)](https://crates.io/crates/ringbuffer)
Expand All @@ -15,13 +16,13 @@ Implementations for three kinds of ringbuffers, with a mostly similar API are pr

All of these ringbuffers also implement the RingBuffer trait for their shared API surface.


MSRV: Rust 1.59

# Usage

```rust
use ringbuffer::{AllocRingBuffer, RingBuffer, RingBufferExt, RingBufferWrite};

fn main() {
let mut buffer = AllocRingBuffer::with_capacity(2);

Expand All @@ -46,14 +47,39 @@ fn main() {
# Comparison of ringbuffer types

| type | heap allocated | growable | size must be power of 2 | `no_std` |
| ----------------------------------------------|----------------|----------|-------------------------|----------|
|-----------------------------------------------|----------------|----------|-------------------------|----------|
| `AllocRingBuffer<T, PowerOfTwo>` | yes | no | yes | no |
| `AllocRingBuffer<T, NonPowerOfTwo>` | yes | no | no | no |
| `GrowableAllocRingBuffer<T>` | yes | yes | no | no |
| `ConstGenericRingBuffer<T, const CAP: usize>` | no | no | no[^1] | yes |

[^1]: Using a size that is not a power of 2 will be ~3x slower.

## To other ringbuffers

We ran a benchmark, pushing `10 000` elements to a ringbuffer with a capacity of `1000` (where it was possible to
configure one) to compare
`ringbuffer` to a few common alternatives.
The outcomes show that using the [`ConstGenericRingBuffer`] is about 23 times faster than using an `std::channel` (
although `ringbuffer` doesn't give the thread safety a channel does).
A maybe more reasonable comparison is to an `std::VecDeque` and `heapless::Deque`,
where ringbuffer is slightly faster as well (among 100 measurements).

| implementation | time (95% confidence interval, \[lower estimate upper\]) |
|---------------------------------------|----------------------------------------------------------|
| `std::Vec` | `[13.190 ms 13.206 ms 13.223 ms]` |
| `std::LinkedList` | `[225.64 µs 228.09 µs 231.06 µs]` |
| `std::channel` | `[174.86 µs 175.41 µs 176.30 µs]` |
| `std::VecDeque` (growable ringbuffer) | `[33.894 µs 33.934 µs 33.974 µs]` |
| `AllocRingBuffer` | `[30.382 µs 30.451 µs 30.551 µs]` |
| `heapless::Deque` | `[16.260 µs 16.464 µs 16.748 µs]` |
| `ConstGenericRingBuffer` | `[13.685 µs 13.712 µs 13.743 µs]` |

Note that none of the alternatives to `RingBuffer` have the exact same behavior to `RingBuffer`. All `std` datastructures
compared here can grow unbounded (though in benchmarks they weren't filled over `10 000` elements).

`heapless::Deque` doesn't drop old items like `ringbuffer` does when the deque is full. Instead, new items aren't let in on push operations.

# Features

| name | default | description |
Expand Down
2 changes: 1 addition & 1 deletion benches/bench.rs
Original file line number Diff line number Diff line change
Expand Up @@ -93,7 +93,7 @@ macro_rules! generate_benches {

fn benchmark_non_power_of_two<const L: usize>(b: &mut Bencher) {
b.iter(|| {
let mut rb = AllocRingBuffer::with_capacity_non_power_of_two(L);
let mut rb = AllocRingBuffer::with_capacity_power_of_2(L);

for i in 0..1_000_000 {
rb.push(i);
Expand Down
20 changes: 19 additions & 1 deletion benches/comparison.rs
Original file line number Diff line number Diff line change
Expand Up @@ -15,6 +15,9 @@ fn std_chan(b: &mut Bencher) {
for i in 0..ITER {
let _ = tx.send(i);
black_box(());
}

for i in 0..ITER {
let res = rx.recv();
let _ = black_box(res);
}
Expand All @@ -28,6 +31,9 @@ fn vec(b: &mut Bencher) {
for i in 0..ITER {
let _ = vd.push(i);
black_box(());
}

for i in 0..ITER {
let res = vd.remove(0);
let _ = black_box(res);
}
Expand All @@ -41,6 +47,8 @@ fn vecdeque(b: &mut Bencher) {
for i in 0..ITER {
let _ = vd.push_back(i);
black_box(());
}
for i in 0..ITER {
let res = vd.pop_front();
let _ = black_box(res);
}
Expand All @@ -54,6 +62,9 @@ fn linked_list(b: &mut Bencher) {
for i in 0..ITER {
let _ = ll.push_back(i);
black_box(());
}

for i in 0..ITER {
let res = ll.pop_front();
let _ = black_box(res);
}
Expand All @@ -67,6 +78,8 @@ fn cg_rb(b: &mut Bencher) {
for i in 0..ITER {
let _ = rb.push(i);
black_box(());
}
for i in 0..ITER {
let res = rb.dequeue();
let _ = black_box(res);
}
Expand All @@ -80,6 +93,8 @@ fn heapless_deque(b: &mut Bencher) {
for i in 0..ITER {
let _ = rb.push_back(i);
black_box(());
}
for i in 0..ITER {
let res = rb.pop_front();
let _ = black_box(res);
}
Expand All @@ -90,9 +105,11 @@ fn al_rb(b: &mut Bencher) {
let mut rb = AllocRingBuffer::with_capacity_non_power_of_two(CAP);

b.iter(|| {
for i in 0..10_000 {
for i in 0..ITER {
let _ = rb.push(i);
black_box(());
}
for i in 0..ITER {
let res = rb.dequeue();
let _ = black_box(res);
}
Expand All @@ -102,6 +119,7 @@ fn al_rb(b: &mut Bencher) {
fn criterion_benchmark(c: &mut Criterion) {
c.bench_function("comparison std channel", std_chan);
c.bench_function("comparison std vec", vec);
c.bench_function("comparison std linked list", linked_list);
c.bench_function("comparison std vecdeque (growable ringbuffer)", vecdeque);
c.bench_function("comparison const generic ringbuffer", cg_rb);
c.bench_function("comparison alloc ringbuffer", al_rb);
Expand Down

1 comment on commit 12fed3e

@github-actions
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Please sign in to comment.