#![doc = include_str!("../README.md")]
#![no_std]
use core::ops::{Deref, DerefMut};
use core::borrow::{Borrow, BorrowMut};
use core::fmt::{Debug, Formatter, Pointer};
pub struct DoubleBuffer<T> {
swapped: bool,
buffers: [T; 2],
}
impl<T> DoubleBuffer<T> {
#[inline]
pub const fn new(current: T, next: T) -> Self {
Self { swapped: false, buffers: [current, next] }
}
#[inline]
pub fn swap(&mut self) {
self.swapped = !self.swapped;
}
#[inline]
const fn current_offset(&self) -> usize {
if self.swapped {
return 1;
}
return 0;
}
#[inline]
const fn next_offset(&self) -> usize {
if self.swapped {
return 0;
}
return 1;
}
#[inline]
const fn current(&self) -> &T {
&self.buffers[self.current_offset()]
}
#[inline]
const fn next(&self) -> &T {
&self.buffers[self.next_offset()]
}
#[inline]
fn current_mut(&mut self) -> &mut T {
&mut self.buffers[self.current_offset()]
}
#[inline]
fn next_mut(&mut self) -> &mut T {
&mut self.buffers[self.next_offset()]
}
}
impl<T: Clone> DoubleBuffer<T> {
#[inline]
pub fn swap_with_clone(&mut self) {
let next = self.next().clone();
let current = self.current_mut();
*current = next;
}
}
impl<T: Default> DoubleBuffer<T> {
#[inline]
pub fn swap_with_default(&mut self) {
self.swap();
let next = self.next_mut();
*next = T::default();
}
}
impl<T: Debug> Debug for DoubleBuffer<T> {
#[inline]
fn fmt(&self, f: &mut Formatter<'_>) -> core::fmt::Result {
f.debug_struct("DoubleBuffer")
.field("current", self.current())
.field("next", self.next())
.finish()
}
}
impl<T> Pointer for DoubleBuffer<T> {
#[inline]
fn fmt(&self, f: &mut Formatter<'_>) -> core::fmt::Result {
write!(f, "{:p}", self.current())
}
}
impl<T: Default> Default for DoubleBuffer<T> {
#[inline]
fn default() -> Self {
Self::new(T::default(), T::default())
}
}
impl<T> Deref for DoubleBuffer<T> {
type Target = T;
#[inline]
fn deref(&self) -> &Self::Target {
self.current()
}
}
impl<T> DerefMut for DoubleBuffer<T> {
#[inline]
fn deref_mut(&mut self) -> &mut Self::Target {
self.next_mut()
}
}
impl<T> Borrow<T> for DoubleBuffer<T> {
#[inline]
fn borrow(&self) -> &T {
self.current()
}
}
impl<T> BorrowMut<T> for DoubleBuffer<T> {
#[inline]
fn borrow_mut(&mut self) -> &mut T {
self.next_mut()
}
}
impl<T> AsRef<T> for DoubleBuffer<T> {
#[inline]
fn as_ref(&self) -> &T {
self.current()
}
}
impl<T> AsMut<T> for DoubleBuffer<T> {
#[inline]
fn as_mut(&mut self) -> &mut T {
self.next_mut()
}
}
impl<T: PartialEq> PartialEq<T> for DoubleBuffer<T> {
#[inline]
fn eq(&self, other: &T) -> bool {
self.current().eq(other)
}
}
impl<T: PartialEq> PartialEq for DoubleBuffer<T> {
#[inline]
fn eq(&self, other: &Self) -> bool {
self.current().eq(other.current())
}
}
impl<T: Eq> Eq for DoubleBuffer<T> {}
impl<T: PartialOrd> PartialOrd<T> for DoubleBuffer<T> {
#[inline]
fn partial_cmp(&self, other: &T) -> Option<core::cmp::Ordering> {
self.current().partial_cmp(other)
}
}
impl<T: PartialOrd> PartialOrd for DoubleBuffer<T> {
#[inline]
fn partial_cmp(&self, other: &Self) -> Option<core::cmp::Ordering> {
self.current().partial_cmp(other.current())
}
}
impl<T: Ord> Ord for DoubleBuffer<T> {
#[inline]
fn cmp(&self, other: &Self) -> core::cmp::Ordering {
self.current().cmp(other.current())
}
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn test_access_and_modify_with_swap() {
let mut buffer: DoubleBuffer<u32> = DoubleBuffer::new(1, 2);
assert_eq!(buffer, 1);
*buffer = 3;
assert_eq!(buffer, 1);
buffer.swap();
assert_eq!(buffer, 3);
}
#[test]
fn test_access_and_modify_with_swap_with_clone() {
let mut buffer: DoubleBuffer<u32> = DoubleBuffer::new(1, 2);
assert_eq!(buffer, 1);
*buffer = 3;
assert_eq!(buffer, 1);
buffer.swap_with_clone();
assert_eq!(buffer, 3);
}
#[test]
fn test_access_and_modify_with_swap_with_default() {
let mut buffer: DoubleBuffer<u32> = DoubleBuffer::new(1, 2);
assert_eq!(buffer, 1);
*buffer = 3;
assert_eq!(buffer, 1);
buffer.swap_with_default();
assert_eq!(buffer, 3);
}
#[test]
fn test_swap() {
let mut buffer: DoubleBuffer<u32> = DoubleBuffer::new(1, 2);
assert_eq!(*buffer.current(), 1);
assert_eq!(*buffer.next(), 2);
buffer.swap();
assert_eq!(*buffer.current(), 2);
assert_eq!(*buffer.next(), 1);
}
#[test]
fn test_swap_with_clone() {
let mut buffer: DoubleBuffer<u32> = DoubleBuffer::new(1, 2);
assert_eq!(*buffer.current(), 1);
assert_eq!(*buffer.next(), 2);
buffer.swap_with_clone();
assert_eq!(*buffer.current(), 2);
assert_eq!(*buffer.next(), 2);
}
#[test]
fn test_swap_with_default() {
let mut buffer: DoubleBuffer<u32> = DoubleBuffer::new(1, 2);
assert_eq!(*buffer.current(), 1);
assert_eq!(*buffer.next(), 2);
buffer.swap_with_default();
assert_eq!(*buffer.current(), 2);
assert_eq!(*buffer.next(), 0);
}
#[test]
fn test_greater_and_less_than() {
let mut buffer: DoubleBuffer<i32> = DoubleBuffer::default();
*buffer = 1;
assert!(buffer > -1);
assert!(buffer < 1);
buffer.swap();
assert!(buffer > 0);
assert!(buffer < 2);
}
#[test]
fn test_modify_bytes_array() {
let mut buffer: DoubleBuffer<[u8; 3]> = DoubleBuffer::default();
buffer[1] = 2;
assert_eq!(buffer[1], 0);
assert_eq!(buffer, [0, 0, 0]);
assert_eq!(*buffer.current(), [0, 0, 0]);
assert_eq!(*buffer.next(), [0, 2, 0]);
buffer.swap();
assert_eq!(buffer[1], 2);
assert_eq!(buffer, [0, 2, 0]);
assert_eq!(*buffer.current(), [0, 2, 0]);
assert_eq!(*buffer.next(), [0, 0, 0]);
}
#[test]
fn test_for_iter_mut_bytes_array() {
let mut buffer: DoubleBuffer<[u8; 3]> = DoubleBuffer::default();
buffer[1] = 2;
assert_eq!(*buffer.current(), [0, 0, 0]);
assert_eq!(*buffer.next(), [0, 2, 0]);
for byte in buffer.iter_mut() {
*byte += 1;
}
assert_eq!(*buffer.current(), [0, 0, 0]);
assert_eq!(*buffer.next(), [1, 3, 1]);
}
}