use core::ptr;
use core::str;
use core::fmt;
use core::mem::{self, MaybeUninit};
use core::borrow::Borrow;
#[derive(Clone, Copy)]
pub struct StrBuffer<const N: usize> {
data: [MaybeUninit<u8>; N],
len: u8,
}
impl<'s, const N: usize> TryFrom<&'s String> for StrBuffer<N> {
type Error = &'s String;
#[inline]
fn try_from(value: &'s String) -> Result<Self, Self::Error> {
Self::try_from(value)
}
}
impl<'s, const N: usize> TryFrom<&'s str> for StrBuffer<N> {
type Error = &'s str;
#[inline]
fn try_from(value: &'s str) -> Result<Self, Self::Error> {
Self::try_from(value)
}
}
impl<const N: usize> StrBuffer<N> {
#[inline]
pub fn new() -> Self {
Self {
len: 0,
data: [MaybeUninit::<u8>::uninit(); N]
}
}
#[inline]
fn try_from<S: AsRef<str>>(s: S) -> Result<Self, S> {
let s_ref = s.as_ref();
if s_ref.len() <= Self::capacity() {
unsafe { Ok(Self::new_unchecked(s_ref)) }
} else {
Err(s)
}
}
#[inline]
unsafe fn new_unchecked(s: &str) -> Self {
let mut data = [MaybeUninit::<u8>::uninit(); N];
ptr::copy_nonoverlapping(s.as_ptr(), data.as_mut_ptr().cast::<u8>(), s.len());
Self {
len: s.len() as u8,
data,
}
}
#[inline]
fn from_array(data: [MaybeUninit<u8>; N], len: u8) -> Self {
Self { data, len }
}
#[inline]
pub fn capacity() -> usize { N }
#[inline]
pub fn len(&self) -> usize { self.len as usize }
#[inline]
pub fn is_empty(&self) -> bool { self.len == 0 }
pub fn try_push(&mut self, ch: char) -> fmt::Result {
let mut buf = [0u8; 4];
self.try_push_str(ch.encode_utf8(&mut buf))
}
#[inline]
pub fn try_push_str<S: AsRef<str>>(&mut self, s: S) -> fmt::Result {
let s_ref = s.as_ref();
if self.len() + s_ref.len() <= Self::capacity() {
let data = self.data[self.len as usize..].as_mut_ptr().cast::<u8>();
unsafe {
ptr::copy_nonoverlapping(s_ref.as_ptr(), data, s_ref.len());
}
self.len += s_ref.len() as u8;
Ok(())
} else {
Err(fmt::Error::default())
}
}
}
impl<const N: usize> core::ops::Deref for StrBuffer<N> {
type Target = str;
#[inline]
fn deref(&self) -> &Self::Target {
let data = &self.data[..self.len()];
unsafe {
let data = &*(data as *const [mem::MaybeUninit<u8>] as *const [u8]);
str::from_utf8_unchecked(data)
}
}
}
impl<const N: usize> AsRef<str> for StrBuffer<N> {
fn as_ref(&self) -> &str { &*self }
}
impl<const N: usize> Borrow<str> for StrBuffer<N> {
fn borrow(&self) -> &str { &*self }
}
impl<const N: usize> fmt::Write for StrBuffer<N> {
fn write_str(&mut self, s: &str) -> fmt::Result {
self.try_push_str(s)
}
fn write_char(&mut self, c: char) -> fmt::Result {
self.try_push(c)
}
}
impl<const N: usize> fmt::Debug for StrBuffer<N> {
#[inline]
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
<str as fmt::Debug>::fmt(self, f)
}
}
impl<const N: usize> fmt::Display for StrBuffer<N> {
#[inline]
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
<str as fmt::Display>::fmt(self, f)
}
}
#[cfg(test)]
mod tests {
use crate::runtime::strings::buffer::StrBuffer;
#[test]
fn empty() {
let lit = "";
let s: StrBuffer<22> = lit.try_into().expect("bad inline str");
assert_eq!(&*s, lit);
assert_eq!(s.len(), lit.len())
}
#[test]
fn good_init() {
let lit = "inline";
let s: StrBuffer<22> = lit.try_into().expect("bad inline str");
assert_eq!(&*s, lit);
assert_eq!(s.len(), lit.len())
}
#[test]
fn bad_init() {
let lit = "This is way too long to be an inline string!!!";
let s = <StrBuffer<22>>::try_from(lit).unwrap_err();
assert_eq!(s, lit);
assert_eq!(s.len(), lit.len())
}
#[test]
fn good_concat() {
let lit = "Inline";
let lit2 = " me";
let mut s = <StrBuffer<22>>::try_from(lit).expect("bad inline str");
assert!(s.try_push_str(lit2).is_ok());
assert_eq!(&*s, lit.to_string() + lit2);
}
#[test]
fn bad_concat() {
let lit = "This is";
let lit2 = " way too long to be an inline string!!!";
let mut s = <StrBuffer<22>>::try_from(lit).expect("bad inline str");
assert!(s.try_push_str(lit2).is_err());
assert_eq!(&*s, lit);
}
}