Skip to content
Snippets Groups Projects
mod.rs 14.4 KiB
Newer Older
  • Learn to ignore specific revisions
  • Carl Lerche's avatar
    Carl Lerche committed
    pub mod append;
    pub mod block;
    pub mod byte;
    pub mod ring;
    pub mod take;
    
    use {Bytes};
    
    Carl Lerche's avatar
    Carl Lerche committed
    use byteorder::ByteOrder;
    
    use std::{cmp, fmt, io, ptr, usize};
    
    
    /// A trait for values that provide sequential read access to bytes.
    pub trait Buf {
    
        /// Returns the number of bytes that can be accessed from the Buf
        fn remaining(&self) -> usize;
    
        /// Returns a slice starting at the current Buf position and of length
        /// between 0 and `Buf::remaining()`.
    
    Carl Lerche's avatar
    Carl Lerche committed
        fn bytes(&self) -> &[u8];
    
    
        /// Advance the internal cursor of the Buf
        fn advance(&mut self, cnt: usize);
    
        /// Returns true if there are any more bytes to consume
        fn has_remaining(&self) -> bool {
            self.remaining() > 0
        }
    
    
    Carl Lerche's avatar
    Carl Lerche committed
        fn copy_to<S: Sink>(&mut self, dst: S) -> usize
    
    Carl Lerche's avatar
    Carl Lerche committed
                where Self: Sized {
    
    Carl Lerche's avatar
    Carl Lerche committed
            let rem = self.remaining();
            dst.copy_from(self);
            rem - self.remaining()
    
    Carl Lerche's avatar
    Carl Lerche committed
        }
    
    
        /// Read bytes from the `Buf` into the given slice and advance the cursor by
        /// the number of bytes read.
    
        /// Returns the number of bytes read.
    
    Carl Lerche's avatar
    Carl Lerche committed
        /// use std::io::Cursor;
        /// use bytes::Buf;
    
    Carl Lerche's avatar
    Carl Lerche committed
        /// let mut buf = Cursor::new(b"hello world");
    
        /// let mut dst = [0; 5];
        ///
        /// buf.read_slice(&mut dst);
        /// assert_eq!(b"hello", &dst);
        /// assert_eq!(6, buf.remaining());
        /// ```
    
    Carl Lerche's avatar
    Carl Lerche committed
        fn read_slice(&mut self, dst: &mut [u8]) {
    
            let mut off = 0;
    
    
    Carl Lerche's avatar
    Carl Lerche committed
            assert!(self.remaining() >= dst.len());
    
            while off < dst.len() {
    
                let cnt;
    
    
                unsafe {
                    let src = self.bytes();
    
    Carl Lerche's avatar
    Carl Lerche committed
                    cnt = cmp::min(src.len(), dst.len() - off);
    
    
                    ptr::copy_nonoverlapping(
                        src.as_ptr(), dst[off..].as_mut_ptr(), cnt);
    
                    off += src.len();
                }
    
                self.advance(cnt);
            }
        }
    
    
    Carl Lerche's avatar
    Carl Lerche committed
        /// Reads an unsigned 8 bit integer from the `Buf` without advancing the
        /// buffer cursor
        fn peek_u8(&self) -> Option<u8> {
    
    Carl Lerche's avatar
    Carl Lerche committed
            if self.has_remaining() {
    
    Carl Lerche's avatar
    Carl Lerche committed
                Some(self.bytes()[0])
    
    Carl Lerche's avatar
    Carl Lerche committed
            } else {
                None
    
    Carl Lerche's avatar
    Carl Lerche committed
        /// Reads an unsigned 8 bit integer from the `Buf`.
        fn read_u8(&mut self) -> u8 {
            let mut buf = [0; 1];
            self.read_slice(&mut buf);
            buf[0]
        }
    
        /// Reads a signed 8 bit integer from the `Buf`.
        fn read_i8(&mut self) -> i8 {
            let mut buf = [0; 1];
            self.read_slice(&mut buf);
            buf[0] as i8
        }
    
        /// Reads an unsigned 16 bit integer from the `Buf`
        fn read_u16<T: ByteOrder>(&mut self) -> u16 {
            let mut buf = [0; 2];
            self.read_slice(&mut buf);
            T::read_u16(&buf)
        }
    
        /// Reads a signed 16 bit integer from the `Buf`
        fn read_i16<T: ByteOrder>(&mut self) -> i16 {
            let mut buf = [0; 2];
            self.read_slice(&mut buf);
            T::read_i16(&buf)
        }
    
        /// Reads an unsigned 32 bit integer from the `Buf`
        fn read_u32<T: ByteOrder>(&mut self) -> u32 {
            let mut buf = [0; 4];
            self.read_slice(&mut buf);
            T::read_u32(&buf)
        }
    
        /// Reads a signed 32 bit integer from the `Buf`
        fn read_i32<T: ByteOrder>(&mut self) -> i32 {
            let mut buf = [0; 4];
            self.read_slice(&mut buf);
            T::read_i32(&buf)
        }
    
        /// Reads an unsigned 64 bit integer from the `Buf`
        fn read_u64<T: ByteOrder>(&mut self) -> u64 {
            let mut buf = [0; 8];
            self.read_slice(&mut buf);
            T::read_u64(&buf)
        }
    
        /// Reads a signed 64 bit integer from the `Buf`
        fn read_i64<T: ByteOrder>(&mut self) -> i64 {
            let mut buf = [0; 8];
            self.read_slice(&mut buf);
            T::read_i64(&buf)
        }
    
        /// Reads an unsigned n-bytes integer from the `Buf`
        fn read_uint<T: ByteOrder>(&mut self, nbytes: usize) -> u64 {
            let mut buf = [0; 8];
            self.read_slice(&mut buf[..nbytes]);
            T::read_uint(&buf[..nbytes], nbytes)
        }
    
        /// Reads a signed n-bytes integer from the `Buf`
        fn read_int<T: ByteOrder>(&mut self, nbytes: usize) -> i64 {
            let mut buf = [0; 8];
            self.read_slice(&mut buf[..nbytes]);
            T::read_int(&buf[..nbytes], nbytes)
        }
    
        /// Reads a IEEE754 single-precision (4 bytes) floating point number from
        /// the `Buf`
        fn read_f32<T: ByteOrder>(&mut self) -> f32 {
            let mut buf = [0; 4];
            self.read_slice(&mut buf);
            T::read_f32(&buf)
        }
    
        /// Reads a IEEE754 double-precision (8 bytes) floating point number from
        /// the `Buf`
        fn read_f64<T: ByteOrder>(&mut self) -> f64 {
            let mut buf = [0; 8];
            self.read_slice(&mut buf);
            T::read_f64(&buf)
    
        }
    }
    
    /// A trait for values that provide sequential write access to bytes.
    
    Carl Lerche's avatar
    Carl Lerche committed
    pub trait MutBuf {
    
        /// Returns the number of bytes that can be written to the MutBuf
    
        fn remaining(&self) -> usize;
    
    
        /// Advance the internal cursor of the MutBuf
    
        unsafe fn advance(&mut self, cnt: usize);
    
        /// Returns true iff there is any more space for bytes to be written
    
        fn has_remaining(&self) -> bool {
            self.remaining() > 0
        }
    
    
        /// Returns a mutable slice starting at the current MutBuf position and of
        /// length between 0 and `MutBuf::remaining()`.
    
        ///
        /// The returned byte slice may represent uninitialized memory.
        unsafe fn mut_bytes<'a>(&'a mut self) -> &'a mut [u8];
    
    Carl Lerche's avatar
    Carl Lerche committed
        fn copy_from<S: Source>(&mut self, src: S) -> usize
    
    Carl Lerche's avatar
    Carl Lerche committed
                where Self: Sized {
    
    Carl Lerche's avatar
    Carl Lerche committed
            let rem = self.remaining();
            src.copy_to(self);
            rem - self.remaining()
    
        /// Write bytes from the given slice into the `MutBuf` and advance the
        /// cursor by the number of bytes written.
        /// Returns the number of bytes written.
    
    Carl Lerche's avatar
    Carl Lerche committed
        /// use bytes::MutBuf;
        /// use std::io::Cursor;
    
        ///
        /// let mut dst = [0; 6];
        ///
        /// {
    
    Carl Lerche's avatar
    Carl Lerche committed
        ///     let mut buf = Cursor::new(&mut dst);
    
        ///     buf.write_slice(b"hello");
        ///
        ///     assert_eq!(1, buf.remaining());
        /// }
        ///
        /// assert_eq!(b"hello\0", &dst);
        /// ```
    
    Carl Lerche's avatar
    Carl Lerche committed
        fn write_slice(&mut self, src: &[u8]) {
    
            let mut off = 0;
    
    
    Carl Lerche's avatar
    Carl Lerche committed
            assert!(self.remaining() >= src.len(), "buffer overflow");
    
            while off < src.len() {
    
                let cnt;
    
    
                unsafe {
                    let dst = self.mut_bytes();
    
    Carl Lerche's avatar
    Carl Lerche committed
                    cnt = cmp::min(dst.len(), src.len() - off);
    
    
                    ptr::copy_nonoverlapping(
                        src[off..].as_ptr(),
                        dst.as_mut_ptr(),
                        cnt);
    
                    off += cnt;
    
                unsafe { self.advance(cnt); }
    
    Carl Lerche's avatar
    Carl Lerche committed
        fn write_str(&mut self, src: &str) {
            self.write_slice(src.as_bytes());
    
    Carl Lerche's avatar
    Carl Lerche committed
    
        /// Writes an unsigned 8 bit integer to the MutBuf.
        fn write_u8(&mut self, n: u8) {
            self.write_slice(&[n])
        }
    
        /// Writes a signed 8 bit integer to the MutBuf.
        fn write_i8(&mut self, n: i8) {
            self.write_slice(&[n as u8])
        }
    
        /// Writes an unsigned 16 bit integer to the MutBuf.
        fn write_u16<T: ByteOrder>(&mut self, n: u16) {
            let mut buf = [0; 2];
            T::write_u16(&mut buf, n);
            self.write_slice(&buf)
        }
    
        /// Writes a signed 16 bit integer to the MutBuf.
        fn write_i16<T: ByteOrder>(&mut self, n: i16) {
            let mut buf = [0; 2];
            T::write_i16(&mut buf, n);
            self.write_slice(&buf)
        }
    
        /// Writes an unsigned 32 bit integer to the MutBuf.
        fn write_u32<T: ByteOrder>(&mut self, n: u32) {
            let mut buf = [0; 4];
            T::write_u32(&mut buf, n);
            self.write_slice(&buf)
        }
    
        /// Writes a signed 32 bit integer to the MutBuf.
        fn write_i32<T: ByteOrder>(&mut self, n: i32) {
            let mut buf = [0; 4];
            T::write_i32(&mut buf, n);
            self.write_slice(&buf)
        }
    
        /// Writes an unsigned 64 bit integer to the MutBuf.
        fn write_u64<T: ByteOrder>(&mut self, n: u64) {
            let mut buf = [0; 8];
            T::write_u64(&mut buf, n);
            self.write_slice(&buf)
        }
    
        /// Writes a signed 64 bit integer to the MutBuf.
        fn write_i64<T: ByteOrder>(&mut self, n: i64) {
            let mut buf = [0; 8];
            T::write_i64(&mut buf, n);
            self.write_slice(&buf)
        }
    
        /// Writes an unsigned n-bytes integer to the MutBuf.
        ///
        /// If the given integer is not representable in the given number of bytes,
        /// this method panics. If `nbytes > 8`, this method panics.
        fn write_uint<T: ByteOrder>(&mut self, n: u64, nbytes: usize) {
            let mut buf = [0; 8];
            T::write_uint(&mut buf, n, nbytes);
            self.write_slice(&buf[0..nbytes])
        }
    
        /// Writes a signed n-bytes integer to the MutBuf.
        ///
        /// If the given integer is not representable in the given number of bytes,
        /// this method panics. If `nbytes > 8`, this method panics.
        fn write_int<T: ByteOrder>(&mut self, n: i64, nbytes: usize) {
            let mut buf = [0; 8];
            T::write_int(&mut buf, n, nbytes);
            self.write_slice(&buf[0..nbytes])
        }
    
        /// Writes a IEEE754 single-precision (4 bytes) floating point number to
        /// the MutBuf.
        fn write_f32<T: ByteOrder>(&mut self, n: f32) {
            let mut buf = [0; 4];
            T::write_f32(&mut buf, n);
            self.write_slice(&buf)
        }
    
        /// Writes a IEEE754 double-precision (8 bytes) floating point number to
        /// the MutBuf.
        fn write_f64<T: ByteOrder>(&mut self, n: f64) {
            let mut buf = [0; 8];
            T::write_f64(&mut buf, n);
            self.write_slice(&buf)
        }
    
    Carl Lerche's avatar
    Carl Lerche committed
     * ===== Sink / Source =====
    
    Carl Lerche's avatar
    Carl Lerche committed
    
    /// A value that writes bytes from itself into a `MutBuf`.
    pub trait Source {
    
    Carl Lerche's avatar
    Carl Lerche committed
        fn copy_to<B: MutBuf>(self, buf: &mut B);
    
    Carl Lerche's avatar
    Carl Lerche committed
    }
    
    impl<'a> Source for &'a [u8] {
    
    Carl Lerche's avatar
    Carl Lerche committed
        fn copy_to<B: MutBuf>(self, buf: &mut B) {
            buf.write_slice(self);
    
    Carl Lerche's avatar
    Carl Lerche committed
    impl Source for u8 {
    
    Carl Lerche's avatar
    Carl Lerche committed
        fn copy_to<B: MutBuf>(self, buf: &mut B) {
    
    Carl Lerche's avatar
    Carl Lerche committed
            let src = [self];
    
    Carl Lerche's avatar
    Carl Lerche committed
            buf.write_slice(&src);
    
    Carl Lerche's avatar
    Carl Lerche committed
    impl Source for Bytes {
        fn copy_to<B: MutBuf>(self, buf: &mut B) {
            Source::copy_to(&self, buf);
        }
    }
    
    Carl Lerche's avatar
    Carl Lerche committed
    impl<'a> Source for &'a Bytes {
        fn copy_to<B: MutBuf>(self, buf: &mut B) {
            Source::copy_to(self.buf(), buf);
        }
    }
    
    impl<T: Buf> Source for T {
        fn copy_to<B: MutBuf>(mut self, buf: &mut B) {
            while self.has_remaining() && buf.has_remaining() {
    
    Carl Lerche's avatar
    Carl Lerche committed
                let l;
    
    Carl Lerche's avatar
    Carl Lerche committed
                unsafe {
    
    Carl Lerche's avatar
    Carl Lerche committed
                    let s = self.bytes();
    
    Carl Lerche's avatar
    Carl Lerche committed
                    let d = buf.mut_bytes();
                    l = cmp::min(s.len(), d.len());
    
    Carl Lerche's avatar
    Carl Lerche committed
                    ptr::copy_nonoverlapping(
                        s.as_ptr(),
                        d.as_mut_ptr(),
                        l);
                }
    
    
    Carl Lerche's avatar
    Carl Lerche committed
                self.advance(l);
    
    Carl Lerche's avatar
    Carl Lerche committed
                unsafe { buf.advance(l); }
            }
        }
    
    Carl Lerche's avatar
    Carl Lerche committed
    pub trait Sink {
    
    Carl Lerche's avatar
    Carl Lerche committed
        fn copy_from<B: Buf>(self, buf: &mut B);
    
    Carl Lerche's avatar
    Carl Lerche committed
    }
    
    Carl Lerche's avatar
    Carl Lerche committed
    impl<'a> Sink for &'a mut [u8] {
    
    Carl Lerche's avatar
    Carl Lerche committed
        fn copy_from<B: Buf>(self, buf: &mut B) {
            buf.read_slice(self);
    
        }
    }
    
    impl<'a> Sink for &'a mut Vec<u8> {
    
    Carl Lerche's avatar
    Carl Lerche committed
        fn copy_from<B: Buf>(self, buf: &mut B) {
    
            use std::slice;
    
            self.clear();
    
            let rem = buf.remaining();
            let cap = self.capacity();
    
            // Ensure that the vec is big enough
            if rem > self.capacity() {
                self.reserve(rem - cap);
            }
    
            unsafe {
                {
                    let dst = &mut self[..];
    
    Carl Lerche's avatar
    Carl Lerche committed
                    buf.read_slice(slice::from_raw_parts_mut(dst.as_mut_ptr(), rem));
    
    Carl Lerche's avatar
    Carl Lerche committed
    /*
     *
     * ===== Read / Write =====
     *
     */
    
    Carl Lerche's avatar
    Carl Lerche committed
    pub trait ReadExt {
        fn read_buf<B: MutBuf>(&mut self, buf: &mut B) -> io::Result<usize>;
    
    Carl Lerche's avatar
    Carl Lerche committed
    impl<T: io::Read> ReadExt for T {
        fn read_buf<B: MutBuf>(&mut self, buf: &mut B) -> io::Result<usize> {
            if !buf.has_remaining() {
                return Ok(0);
            }
    
            unsafe {
                let i = try!(self.read(buf.mut_bytes()));
    
    Carl Lerche's avatar
    Carl Lerche committed
                buf.advance(i);
                Ok(i)
            }
    
    Carl Lerche's avatar
    Carl Lerche committed
    pub trait WriteExt {
        fn write_buf<B: Buf>(&mut self, buf: &mut B) -> io::Result<usize>;
    }
    
    Carl Lerche's avatar
    Carl Lerche committed
    impl<T: io::Write> WriteExt for T {
        fn write_buf<B: Buf>(&mut self, buf: &mut B) -> io::Result<usize> {
            if !buf.has_remaining() {
                return Ok(0);
    
    Carl Lerche's avatar
    Carl Lerche committed
            let i = try!(self.write(buf.bytes()));
            buf.advance(i);
            Ok(i)
    
    impl<T: AsRef<[u8]>> Buf for io::Cursor<T> {
    
        fn remaining(&self) -> usize {
    
            self.get_ref().as_ref().len() - self.position() as usize
    
        }
    
        fn bytes(&self) -> &[u8] {
            let pos = self.position() as usize;
    
            &(self.get_ref().as_ref())[pos..]
    
        }
    
        fn advance(&mut self, cnt: usize) {
            let pos = self.position() as usize;
    
            let pos = cmp::min(self.get_ref().as_ref().len(), pos + cnt);
    
            self.set_position(pos as u64);
        }
    }
    
    
    impl<T: AsMut<[u8]> + AsRef<[u8]>> MutBuf for io::Cursor<T> {
    
        fn remaining(&self) -> usize {
            self.get_ref().as_ref().len() - self.position() as usize
        }
    
        /// Advance the internal cursor of the MutBuf
        unsafe fn advance(&mut self, cnt: usize) {
            let pos = self.position() as usize;
            let pos = cmp::min(self.get_mut().as_mut().len(), pos + cnt);
            self.set_position(pos as u64);
        }
    
        /// Returns a mutable slice starting at the current MutBuf position and of
        /// length between 0 and `MutBuf::remaining()`.
        ///
        /// The returned byte slice may represent uninitialized memory.
        unsafe fn mut_bytes<'a>(&'a mut self) -> &'a mut [u8] {
            let pos = self.position() as usize;
            &mut (self.get_mut().as_mut())[pos..]
        }
    }
    
    
    impl MutBuf for Vec<u8> {
        fn remaining(&self) -> usize {
            usize::MAX - self.len()
        }
    
    
        unsafe fn advance(&mut self, cnt: usize) {
    
            let len = self.len() + cnt;
    
            if len > self.capacity() {
                // Reserve additional
                // TODO: Should this case panic?
                let cap = self.capacity();
                self.reserve(cap - len);
            }
    
    
            self.set_len(len);
    
        unsafe fn mut_bytes(&mut self) -> &mut [u8] {
    
            use std::slice;
    
            if self.capacity() == self.len() {
                self.reserve(64); // Grow the vec
            }
    
            let cap = self.capacity();
            let len = self.len();
    
    
            let ptr = self.as_mut_ptr();
            &mut slice::from_raw_parts_mut(ptr, cap)[len..]
    
    Carl Lerche's avatar
    Carl Lerche committed
     * ===== fmt impls =====
    
    Carl Lerche's avatar
    Carl Lerche committed
    pub struct Fmt<'a, B: 'a>(pub &'a mut B);
    
    Carl Lerche's avatar
    Carl Lerche committed
    impl<'a, B: MutBuf> fmt::Write for Fmt<'a, B> {
        fn write_str(&mut self, s: &str) -> fmt::Result {
            self.0.write_str(s);
            Ok(())
    
    Carl Lerche's avatar
    Carl Lerche committed
        fn write_fmt(&mut self, args: fmt::Arguments) -> fmt::Result {
            fmt::write(self, args)