use std::{cmp, mem, ptr};
macro_rules! buf_get_impl {
($this:ident, $typ:tt::$conv:tt) => {{
const SIZE: usize = mem::size_of::<$typ>();
#[allow(clippy::ptr_as_ptr)]
let ret = $this.chunk().get(..SIZE).map(|src| unsafe {
$typ::$conv(*(std::ptr::from_ref(src) as *const [_; SIZE]))
});
if let Some(ret) = ret {
$this.advance(SIZE);
return ret;
}
let mut buf = [0; SIZE];
$this.copy_to_slice(&mut buf); return $typ::$conv(buf);
}};
(le => $this:ident, $typ:tt, $len_to_read:expr) => {{
debug_assert!(mem::size_of::<$typ>() >= $len_to_read);
let mut buf = [0; (mem::size_of::<$typ>())];
$this.copy_to_slice(&mut buf[..($len_to_read)]);
return $typ::from_le_bytes(buf);
}};
(be => $this:ident, $typ:tt, $len_to_read:expr) => {{
debug_assert!(mem::size_of::<$typ>() >= $len_to_read);
let mut buf = [0; (mem::size_of::<$typ>())];
$this.copy_to_slice(&mut buf[mem::size_of::<$typ>() - ($len_to_read)..]);
return $typ::from_be_bytes(buf);
}};
}
pub trait Buf {
fn remaining(&self) -> usize;
fn chunk(&self) -> &[u8];
fn advance(&mut self, cnt: usize);
fn has_remaining(&self) -> bool {
self.remaining() > 0
}
fn copy_to_slice(&mut self, dst: &mut [u8]) {
let mut off = 0;
assert!(self.remaining() >= dst.len());
while off < dst.len() {
let cnt;
unsafe {
let src = self.chunk();
cnt = cmp::min(src.len(), dst.len() - off);
ptr::copy_nonoverlapping(src.as_ptr(), dst[off..].as_mut_ptr(), cnt);
off += cnt;
}
self.advance(cnt);
}
}
#[inline]
fn get_u8(&mut self) -> u8 {
assert!(self.remaining() >= 1);
let ret = self.chunk()[0];
self.advance(1);
ret
}
#[inline]
fn get_i8(&mut self) -> i8 {
self.get_u8() as i8
}
#[inline]
fn get_u16(&mut self) -> u16 {
buf_get_impl!(self, u16::from_be_bytes);
}
#[inline]
fn get_u16_le(&mut self) -> u16 {
buf_get_impl!(self, u16::from_le_bytes);
}
#[inline]
fn get_i16(&mut self) -> i16 {
buf_get_impl!(self, i16::from_be_bytes);
}
#[inline]
fn get_i16_le(&mut self) -> i16 {
buf_get_impl!(self, i16::from_le_bytes);
}
#[inline]
fn get_u32(&mut self) -> u32 {
buf_get_impl!(self, u32::from_be_bytes);
}
#[inline]
fn get_u32_le(&mut self) -> u32 {
buf_get_impl!(self, u32::from_le_bytes);
}
#[inline]
fn get_i32(&mut self) -> i32 {
buf_get_impl!(self, i32::from_be_bytes);
}
#[inline]
fn get_i32_le(&mut self) -> i32 {
buf_get_impl!(self, i32::from_le_bytes);
}
#[inline]
fn get_u64(&mut self) -> u64 {
buf_get_impl!(self, u64::from_be_bytes);
}
#[inline]
fn get_u64_le(&mut self) -> u64 {
buf_get_impl!(self, u64::from_le_bytes);
}
#[inline]
fn get_i64(&mut self) -> i64 {
buf_get_impl!(self, i64::from_be_bytes);
}
#[inline]
fn get_i64_le(&mut self) -> i64 {
buf_get_impl!(self, i64::from_le_bytes);
}
#[inline]
fn get_u128(&mut self) -> u128 {
buf_get_impl!(self, u128::from_be_bytes);
}
#[inline]
fn get_u128_le(&mut self) -> u128 {
buf_get_impl!(self, u128::from_le_bytes);
}
#[inline]
fn get_i128(&mut self) -> i128 {
buf_get_impl!(self, i128::from_be_bytes);
}
#[inline]
fn get_i128_le(&mut self) -> i128 {
buf_get_impl!(self, i128::from_le_bytes);
}
#[inline]
fn get_uint(&mut self, nbytes: usize) -> u64 {
buf_get_impl!(be => self, u64, nbytes);
}
#[inline]
fn get_uint_le(&mut self, nbytes: usize) -> u64 {
buf_get_impl!(le => self, u64, nbytes);
}
#[inline]
fn get_int(&mut self, nbytes: usize) -> i64 {
buf_get_impl!(be => self, i64, nbytes);
}
#[inline]
fn get_int_le(&mut self, nbytes: usize) -> i64 {
buf_get_impl!(le => self, i64, nbytes);
}
#[inline]
fn get_f32(&mut self) -> f32 {
f32::from_bits(Self::get_u32(self))
}
#[inline]
fn get_f32_le(&mut self) -> f32 {
f32::from_bits(Self::get_u32_le(self))
}
#[inline]
fn get_f64(&mut self) -> f64 {
f64::from_bits(Self::get_u64(self))
}
#[inline]
fn get_f64_le(&mut self) -> f64 {
f64::from_bits(Self::get_u64_le(self))
}
#[inline]
#[allow(clippy::wrong_self_convention)]
fn to_bytes(&mut self) -> crate::Bytes {
use super::BufMut;
let mut ret = crate::BytesMut::with_capacity(self.remaining());
ret.put(self);
ret.freeze()
}
}
impl<T: Buf + ?Sized> Buf for &mut T {
#[inline]
fn remaining(&self) -> usize {
(**self).remaining()
}
#[inline]
fn chunk(&self) -> &[u8] {
(**self).chunk()
}
#[inline]
fn advance(&mut self, cnt: usize) {
(**self).advance(cnt);
}
}
impl<T: Buf + ?Sized> Buf for Box<T> {
#[inline]
fn remaining(&self) -> usize {
(**self).remaining()
}
#[inline]
fn chunk(&self) -> &[u8] {
(**self).chunk()
}
fn advance(&mut self, cnt: usize) {
(**self).advance(cnt);
}
}
impl Buf for &[u8] {
#[inline]
fn remaining(&self) -> usize {
self.len()
}
#[inline]
fn chunk(&self) -> &[u8] {
self
}
#[inline]
fn advance(&mut self, cnt: usize) {
*self = &self[cnt..];
}
#[inline]
fn get_u8(&mut self) -> u8 {
let ret = self[0];
self.advance(1);
ret
}
}
impl Buf for &str {
#[inline]
fn remaining(&self) -> usize {
self.len()
}
#[inline]
fn chunk(&self) -> &[u8] {
self.as_bytes()
}
#[inline]
fn advance(&mut self, cnt: usize) {
*self = &self[cnt..];
}
#[inline]
fn get_u8(&mut self) -> u8 {
let ret = self.as_bytes()[0];
self.advance(1);
ret
}
}
impl<T: AsRef<[u8]>> Buf for std::io::Cursor<T> {
fn remaining(&self) -> usize {
let len = self.get_ref().as_ref().len();
let pos = self.position();
if pos >= len as u64 {
return 0;
}
len - pos as usize
}
fn chunk(&self) -> &[u8] {
let len = self.get_ref().as_ref().len();
let pos = self.position();
if pos >= len as u64 {
return &[];
}
&self.get_ref().as_ref()[pos as usize..]
}
fn advance(&mut self, cnt: usize) {
let pos = (self.position() as usize)
.checked_add(cnt)
.expect("overflow");
assert!(pos <= self.get_ref().as_ref().len());
self.set_position(pos as u64);
}
}
fn _assert_trait_object(_b: &dyn Buf) {}
#[cfg(test)]
#[allow(clippy::float_cmp)]
mod tests {
use super::*;
#[test]
fn buf_tests() {
let mut buf = &b"a"[..];
assert!(buf.has_remaining());
buf.get_u8();
assert!(!buf.has_remaining());
let mut buf = &b"hello world"[..];
let mut dst = [0; 5];
buf.copy_to_slice(&mut dst);
assert_eq!(&b"hello"[..], &dst);
assert_eq!(6, buf.remaining());
buf.advance(1);
assert_eq!(Buf::chunk(&buf), &b"world"[..]);
let mut buf = &b"hello world"[..];
buf.advance(5);
let mut buf = Box::new(buf);
assert_eq!(buf.remaining(), 6);
assert_eq!(buf.chunk(), b" world");
buf.advance(1);
assert_eq!(buf.chunk(), b"world");
let mut buf = std::io::Cursor::new(b" world");
assert_eq!(buf.remaining(), 6);
assert_eq!(buf.chunk(), b" world");
buf.advance(1);
assert_eq!(buf.chunk(), b"world");
buf.advance(5);
assert_eq!(buf.remaining(), 0);
assert_eq!(buf.chunk(), b"");
let mut buf = &b"\x08 hello"[..];
assert_eq!(8, buf.get_u8());
let mut buf = &b"\x08 hello"[..];
assert_eq!(8, buf.get_i8());
let mut buf = &b"\x08\x09 hello"[..];
assert_eq!(0x0809, buf.get_u16());
let mut buf = &b"\x09\x08 hello"[..];
assert_eq!(0x0809, buf.get_u16_le());
let mut buf = &b"\x08\x09 hello"[..];
assert_eq!(0x0809, buf.get_i16());
let mut buf = &b"\x09\x08 hello"[..];
assert_eq!(0x0809, buf.get_i16_le());
let mut buf = &b"\x08\x09\xA0\xA1 hello"[..];
assert_eq!(0x0809_A0A1, buf.get_u32());
let mut buf = &b"\xA1\xA0\x09\x08 hello"[..];
assert_eq!(0x0809_A0A1, buf.get_u32_le());
let mut buf = &b"\x08\x09\xA0\xA1 hello"[..];
assert_eq!(0x0809_A0A1, buf.get_i32());
let mut buf = &b"\xA1\xA0\x09\x08 hello"[..];
assert_eq!(0x0809_A0A1, buf.get_i32_le());
let mut buf = &b"\x01\x02\x03\x04\x05\x06\x07\x08 hello"[..];
assert_eq!(0x0102_0304_0506_0708, buf.get_u64());
let mut buf = &b"\x08\x07\x06\x05\x04\x03\x02\x01 hello"[..];
assert_eq!(0x0102_0304_0506_0708, buf.get_u64_le());
let mut buf = &b"\x01\x02\x03\x04\x05\x06\x07\x08 hello"[..];
assert_eq!(0x0102_0304_0506_0708, buf.get_i64());
let mut buf = &b"\x08\x07\x06\x05\x04\x03\x02\x01 hello"[..];
assert_eq!(0x0102_0304_0506_0708, buf.get_i64_le());
let mut buf =
&b"\x01\x02\x03\x04\x05\x06\x07\x08\x09\x10\x11\x12\x13\x14\x15\x16 hello"[..];
assert_eq!(0x0102_0304_0506_0708_0910_1112_1314_1516, buf.get_u128());
let mut buf =
&b"\x16\x15\x14\x13\x12\x11\x10\x09\x08\x07\x06\x05\x04\x03\x02\x01 hello"[..];
assert_eq!(0x0102_0304_0506_0708_0910_1112_1314_1516, buf.get_u128_le());
let mut buf =
&b"\x01\x02\x03\x04\x05\x06\x07\x08\x09\x10\x11\x12\x13\x14\x15\x16 hello"[..];
assert_eq!(0x0102_0304_0506_0708_0910_1112_1314_1516, buf.get_i128());
let mut buf =
&b"\x16\x15\x14\x13\x12\x11\x10\x09\x08\x07\x06\x05\x04\x03\x02\x01 hello"[..];
assert_eq!(0x0102_0304_0506_0708_0910_1112_1314_1516, buf.get_i128_le());
let mut buf = &b"\x01\x02\x03 hello"[..];
assert_eq!(0x01_0203, buf.get_uint(3));
let mut buf = &b"\x03\x02\x01 hello"[..];
assert_eq!(0x01_0203, buf.get_uint_le(3));
let mut buf = &b"\x01\x02\x03 hello"[..];
assert_eq!(0x01_0203, buf.get_int(3));
let mut buf = &b"\x03\x02\x01 hello"[..];
assert_eq!(0x01_0203, buf.get_int_le(3));
let mut buf = &b"\x3F\x99\x99\x9A hello"[..];
assert_eq!(1.2f32, buf.get_f32());
let mut buf = &b"\x9A\x99\x99\x3F hello"[..];
assert_eq!(1.2f32, buf.get_f32_le());
let mut buf = &b"\x3F\xF3\x33\x33\x33\x33\x33\x33 hello"[..];
assert_eq!(1.2f64, buf.get_f64());
let mut buf = &b"\x33\x33\x33\x33\x33\x33\xF3\x3F hello"[..];
assert_eq!(1.2f64, buf.get_f64_le());
let bytes = "hello world".to_bytes();
assert_eq!(&bytes[..], &b"hello world"[..]);
}
}