use crate::ring::PicoRing;
use std::io::{self, Read, Write};
pub struct PicoByteStream<const N: usize = 0> {
ring: PicoRing<u8, N>,
}
impl<const N: usize> PicoByteStream<N> {
pub fn new_static() -> Result<Self, String> {
Ok(Self {
ring: PicoRing::new()?,
})
}
}
impl PicoByteStream<0> {
pub fn new(capacity: usize) -> Result<Self, String> {
Ok(Self {
ring: PicoRing::with_capacity(capacity)?,
})
}
}
impl<const N: usize> PicoByteStream<N> {
pub fn available_to_read(&self) -> usize {
self.ring.len()
}
pub fn available_to_write(&self) -> usize {
self.ring.available_space()
}
}
impl<const N: usize> Read for PicoByteStream<N> {
#[inline]
fn read(&mut self, buf: &mut [u8]) -> io::Result<usize> {
let n = self.available_to_read().min(buf.len());
if n == 0 {
return Ok(0);
}
unsafe {
let src = self.ring.as_mut_ptr().add(self.ring.tail());
core::ptr::copy_nonoverlapping(src, buf.as_mut_ptr(), n);
}
self.ring.advance_tail(n);
Ok(n)
}
}
impl<const N: usize> Write for PicoByteStream<N> {
#[inline]
fn write(&mut self, buf: &[u8]) -> io::Result<usize> {
let n = self.available_to_write().min(buf.len());
if n == 0 {
return Ok(0);
}
unsafe {
let dest = self.ring.as_mut_ptr().add(self.ring.head());
core::ptr::copy_nonoverlapping(buf.as_ptr(), dest, n);
}
self.ring.advance_head(n);
Ok(n)
}
#[inline]
fn flush(&mut self) -> io::Result<()> {
Ok(())
}
}
impl<const N: usize> PicoByteStream<N> {
pub fn as_read_slice(&self) -> &[u8] {
self.ring.readable_slice()
}
pub fn as_write_slice(&mut self) -> &mut [u8] {
self.ring.writable_slice()
}
pub fn consume(&mut self, n: usize) {
self.ring.advance_tail(n);
}
pub fn produce(&mut self, n: usize) {
self.ring.advance_head(n);
}
#[inline]
pub fn iter(&self) -> core::slice::Iter<'_, u8> {
self.as_read_slice().iter()
}
#[inline]
pub fn iter_mut(&mut self) -> core::slice::IterMut<'_, u8> {
unsafe {
let ptr = self.ring.as_mut_ptr().add(self.ring.tail());
core::slice::from_raw_parts_mut(ptr, self.ring.len()).iter_mut()
}
}
}
impl<const N: usize> core::ops::Index<usize> for PicoByteStream<N> {
type Output = u8;
#[inline]
fn index(&self, index: usize) -> &Self::Output {
&self.as_read_slice()[index]
}
}
impl<const N: usize> core::ops::IndexMut<usize> for PicoByteStream<N> {
#[inline]
fn index_mut(&mut self, index: usize) -> &mut Self::Output {
unsafe {
let ptr = self.ring.as_mut_ptr().add(self.ring.tail());
let slice = core::slice::from_raw_parts_mut(ptr, self.ring.len());
&mut slice[index]
}
}
}
impl<'a, const N: usize> IntoIterator for &'a PicoByteStream<N> {
type Item = &'a u8;
type IntoIter = core::slice::Iter<'a, u8>;
#[inline]
fn into_iter(self) -> Self::IntoIter {
self.iter()
}
}
impl<'a, const N: usize> IntoIterator for &'a mut PicoByteStream<N> {
type Item = &'a mut u8;
type IntoIter = core::slice::IterMut<'a, u8>;
#[inline]
fn into_iter(self) -> Self::IntoIter {
self.iter_mut()
}
}