2014-01-25 20:37:51 +13:00
|
|
|
// Copyright 2013-2014 The Rust Project Developers. See the COPYRIGHT
|
2013-04-17 17:55:21 -07:00
|
|
|
// file at the top-level directory of this distribution and at
|
|
|
|
// http://rust-lang.org/COPYRIGHT.
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
|
|
|
|
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
|
|
|
|
// option. This file may not be copied, modified, or distributed
|
|
|
|
// except according to those terms.
|
|
|
|
|
|
|
|
//! Utility mixins that apply to all Readers and Writers
|
|
|
|
|
2014-10-27 15:37:07 -07:00
|
|
|
#![allow(missing_docs)]
|
2014-02-24 00:31:08 -08:00
|
|
|
|
2014-01-26 03:43:42 -05:00
|
|
|
// FIXME: Not sure how this should be structured
|
|
|
|
// FIXME: Iteration should probably be considered separately
|
2013-04-17 17:55:21 -07:00
|
|
|
|
2014-09-17 10:47:05 -07:00
|
|
|
use io::{IoError, IoResult, Reader};
|
|
|
|
use io;
|
2013-09-08 11:01:16 -04:00
|
|
|
use iter::Iterator;
|
2014-09-17 10:47:05 -07:00
|
|
|
use num::Int;
|
2014-12-07 14:15:25 -05:00
|
|
|
use ops::FnOnce;
|
2014-11-28 11:57:41 -05:00
|
|
|
use option::Option;
|
|
|
|
use option::Option::{Some, None};
|
2014-09-17 10:47:05 -07:00
|
|
|
use ptr::RawPtr;
|
2014-11-28 11:57:41 -05:00
|
|
|
use result::Result::{Ok, Err};
|
2014-12-11 09:44:17 -08:00
|
|
|
use slice::{SliceExt, AsSlice};
|
2013-09-02 01:30:21 +02:00
|
|
|
|
|
|
|
/// An iterator that reads a single byte on each iteration,
|
2014-02-19 21:53:46 -05:00
|
|
|
/// until `.read_byte()` returns `EndOfFile`.
|
2013-09-03 20:00:32 +02:00
|
|
|
///
|
|
|
|
/// # Notes about the Iteration Protocol
|
|
|
|
///
|
2014-01-14 22:32:24 -05:00
|
|
|
/// The `Bytes` may yield `None` and thus terminate
|
2013-09-03 20:00:32 +02:00
|
|
|
/// an iteration, but continue to yield elements if iteration
|
|
|
|
/// is attempted again.
|
2013-09-02 01:30:21 +02:00
|
|
|
///
|
2014-02-19 21:53:46 -05:00
|
|
|
/// # Error
|
2013-09-02 01:30:21 +02:00
|
|
|
///
|
2014-02-19 21:53:46 -05:00
|
|
|
/// Any error other than `EndOfFile` that is produced by the underlying Reader
|
|
|
|
/// is returned by the iterator and should be handled by the caller.
|
2014-08-27 21:46:52 -04:00
|
|
|
pub struct Bytes<'r, T:'r> {
|
|
|
|
reader: &'r mut T,
|
|
|
|
}
|
|
|
|
|
2014-01-14 22:32:24 -05:00
|
|
|
impl<'r, R: Reader> Bytes<'r, R> {
|
2014-03-16 15:59:04 -07:00
|
|
|
/// Constructs a new byte iterator from the given Reader instance.
|
2014-01-14 22:32:24 -05:00
|
|
|
pub fn new(r: &'r mut R) -> Bytes<'r, R> {
|
2014-03-16 15:59:04 -07:00
|
|
|
Bytes {
|
|
|
|
reader: r,
|
|
|
|
}
|
2013-10-25 18:08:45 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-02-19 21:53:46 -05:00
|
|
|
impl<'r, R: Reader> Iterator<IoResult<u8>> for Bytes<'r, R> {
|
2013-09-02 01:30:21 +02:00
|
|
|
#[inline]
|
2014-02-19 21:53:46 -05:00
|
|
|
fn next(&mut self) -> Option<IoResult<u8>> {
|
|
|
|
match self.reader.read_byte() {
|
|
|
|
Ok(x) => Some(Ok(x)),
|
|
|
|
Err(IoError { kind: io::EndOfFile, .. }) => None,
|
|
|
|
Err(e) => Some(Err(e))
|
|
|
|
}
|
2013-09-02 01:30:21 +02:00
|
|
|
}
|
2013-05-13 16:56:16 -07:00
|
|
|
}
|
|
|
|
|
2014-03-16 15:59:04 -07:00
|
|
|
/// Converts an 8-bit to 64-bit unsigned value to a little-endian byte
|
|
|
|
/// representation of the given size. If the size is not big enough to
|
|
|
|
/// represent the value, then the high-order bytes are truncated.
|
|
|
|
///
|
|
|
|
/// Arguments:
|
|
|
|
///
|
|
|
|
/// * `n`: The value to convert.
|
|
|
|
/// * `size`: The size of the value, in bytes. This must be 8 or less, or task
|
2014-10-09 15:17:22 -04:00
|
|
|
/// panic occurs. If this is less than 8, then a value of that
|
2014-03-16 15:59:04 -07:00
|
|
|
/// many bytes is produced. For example, if `size` is 4, then a
|
|
|
|
/// 32-bit byte representation is produced.
|
|
|
|
/// * `f`: A callback that receives the value.
|
|
|
|
///
|
|
|
|
/// This function returns the value returned by the callback, for convenience.
|
2014-12-07 14:15:25 -05:00
|
|
|
pub fn u64_to_le_bytes<T, F>(n: u64, size: uint, f: F) -> T where
|
|
|
|
F: FnOnce(&[u8]) -> T,
|
|
|
|
{
|
core: Remove the cast module
This commit revisits the `cast` module in libcore and libstd, and scrutinizes
all functions inside of it. The result was to remove the `cast` module entirely,
folding all functionality into the `mem` module. Specifically, this is the fate
of each function in the `cast` module.
* transmute - This function was moved to `mem`, but it is now marked as
#[unstable]. This is due to planned changes to the `transmute`
function and how it can be invoked (see the #[unstable] comment).
For more information, see RFC 5 and #12898
* transmute_copy - This function was moved to `mem`, with clarification that is
is not an error to invoke it with T/U that are different
sizes, but rather that it is strongly discouraged. This
function is now #[stable]
* forget - This function was moved to `mem` and marked #[stable]
* bump_box_refcount - This function was removed due to the deprecation of
managed boxes as well as its questionable utility.
* transmute_mut - This function was previously deprecated, and removed as part
of this commit.
* transmute_mut_unsafe - This function doesn't serve much of a purpose when it
can be achieved with an `as` in safe code, so it was
removed.
* transmute_lifetime - This function was removed because it is likely a strong
indication that code is incorrect in the first place.
* transmute_mut_lifetime - This function was removed for the same reasons as
`transmute_lifetime`
* copy_lifetime - This function was moved to `mem`, but it is marked
`#[unstable]` now due to the likelihood of being removed in
the future if it is found to not be very useful.
* copy_mut_lifetime - This function was also moved to `mem`, but had the same
treatment as `copy_lifetime`.
* copy_lifetime_vec - This function was removed because it is not used today,
and its existence is not necessary with DST
(copy_lifetime will suffice).
In summary, the cast module was stripped down to these functions, and then the
functions were moved to the `mem` module.
transmute - #[unstable]
transmute_copy - #[stable]
forget - #[stable]
copy_lifetime - #[unstable]
copy_mut_lifetime - #[unstable]
[breaking-change]
2014-05-09 10:34:51 -07:00
|
|
|
use mem::transmute;
|
2014-02-01 14:56:51 +01:00
|
|
|
|
|
|
|
// LLVM fails to properly optimize this when using shifts instead of the to_le* intrinsics
|
2013-10-21 23:06:12 -07:00
|
|
|
assert!(size <= 8u);
|
|
|
|
match size {
|
|
|
|
1u => f(&[n as u8]),
|
2014-11-17 21:39:01 +13:00
|
|
|
2u => f(unsafe { & transmute::<_, [u8, ..2]>((n as u16).to_le()) }),
|
|
|
|
4u => f(unsafe { & transmute::<_, [u8, ..4]>((n as u32).to_le()) }),
|
|
|
|
8u => f(unsafe { & transmute::<_, [u8, ..8]>(n.to_le()) }),
|
2013-10-21 23:06:12 -07:00
|
|
|
_ => {
|
|
|
|
|
2014-04-09 11:43:33 +10:00
|
|
|
let mut bytes = vec!();
|
2013-10-21 23:06:12 -07:00
|
|
|
let mut i = size;
|
|
|
|
let mut n = n;
|
|
|
|
while i > 0u {
|
|
|
|
bytes.push((n & 255_u64) as u8);
|
2014-04-21 17:58:52 -04:00
|
|
|
n >>= 8;
|
2013-10-21 23:06:12 -07:00
|
|
|
i -= 1u;
|
|
|
|
}
|
2014-04-09 11:43:33 +10:00
|
|
|
f(bytes.as_slice())
|
2013-10-21 23:06:12 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-03-16 15:59:04 -07:00
|
|
|
/// Converts an 8-bit to 64-bit unsigned value to a big-endian byte
|
|
|
|
/// representation of the given size. If the size is not big enough to
|
|
|
|
/// represent the value, then the high-order bytes are truncated.
|
|
|
|
///
|
|
|
|
/// Arguments:
|
|
|
|
///
|
|
|
|
/// * `n`: The value to convert.
|
|
|
|
/// * `size`: The size of the value, in bytes. This must be 8 or less, or task
|
2014-10-09 15:17:22 -04:00
|
|
|
/// panic occurs. If this is less than 8, then a value of that
|
2014-03-16 15:59:04 -07:00
|
|
|
/// many bytes is produced. For example, if `size` is 4, then a
|
|
|
|
/// 32-bit byte representation is produced.
|
|
|
|
/// * `f`: A callback that receives the value.
|
|
|
|
///
|
|
|
|
/// This function returns the value returned by the callback, for convenience.
|
2014-12-07 14:15:25 -05:00
|
|
|
pub fn u64_to_be_bytes<T, F>(n: u64, size: uint, f: F) -> T where
|
|
|
|
F: FnOnce(&[u8]) -> T,
|
|
|
|
{
|
core: Remove the cast module
This commit revisits the `cast` module in libcore and libstd, and scrutinizes
all functions inside of it. The result was to remove the `cast` module entirely,
folding all functionality into the `mem` module. Specifically, this is the fate
of each function in the `cast` module.
* transmute - This function was moved to `mem`, but it is now marked as
#[unstable]. This is due to planned changes to the `transmute`
function and how it can be invoked (see the #[unstable] comment).
For more information, see RFC 5 and #12898
* transmute_copy - This function was moved to `mem`, with clarification that is
is not an error to invoke it with T/U that are different
sizes, but rather that it is strongly discouraged. This
function is now #[stable]
* forget - This function was moved to `mem` and marked #[stable]
* bump_box_refcount - This function was removed due to the deprecation of
managed boxes as well as its questionable utility.
* transmute_mut - This function was previously deprecated, and removed as part
of this commit.
* transmute_mut_unsafe - This function doesn't serve much of a purpose when it
can be achieved with an `as` in safe code, so it was
removed.
* transmute_lifetime - This function was removed because it is likely a strong
indication that code is incorrect in the first place.
* transmute_mut_lifetime - This function was removed for the same reasons as
`transmute_lifetime`
* copy_lifetime - This function was moved to `mem`, but it is marked
`#[unstable]` now due to the likelihood of being removed in
the future if it is found to not be very useful.
* copy_mut_lifetime - This function was also moved to `mem`, but had the same
treatment as `copy_lifetime`.
* copy_lifetime_vec - This function was removed because it is not used today,
and its existence is not necessary with DST
(copy_lifetime will suffice).
In summary, the cast module was stripped down to these functions, and then the
functions were moved to the `mem` module.
transmute - #[unstable]
transmute_copy - #[stable]
forget - #[stable]
copy_lifetime - #[unstable]
copy_mut_lifetime - #[unstable]
[breaking-change]
2014-05-09 10:34:51 -07:00
|
|
|
use mem::transmute;
|
2014-02-01 14:56:51 +01:00
|
|
|
|
|
|
|
// LLVM fails to properly optimize this when using shifts instead of the to_be* intrinsics
|
2013-10-21 23:06:12 -07:00
|
|
|
assert!(size <= 8u);
|
|
|
|
match size {
|
|
|
|
1u => f(&[n as u8]),
|
2014-11-17 21:39:01 +13:00
|
|
|
2u => f(unsafe { & transmute::<_, [u8, ..2]>((n as u16).to_be()) }),
|
|
|
|
4u => f(unsafe { & transmute::<_, [u8, ..4]>((n as u32).to_be()) }),
|
|
|
|
8u => f(unsafe { & transmute::<_, [u8, ..8]>(n.to_be()) }),
|
2013-10-21 23:06:12 -07:00
|
|
|
_ => {
|
2014-04-09 11:43:33 +10:00
|
|
|
let mut bytes = vec!();
|
2013-10-21 23:06:12 -07:00
|
|
|
let mut i = size;
|
|
|
|
while i > 0u {
|
2014-04-21 17:58:52 -04:00
|
|
|
let shift = (i - 1u) * 8u;
|
2013-10-21 23:06:12 -07:00
|
|
|
bytes.push((n >> shift) as u8);
|
|
|
|
i -= 1u;
|
|
|
|
}
|
2014-04-09 11:43:33 +10:00
|
|
|
f(bytes.as_slice())
|
2013-10-21 23:06:12 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-03-16 15:59:04 -07:00
|
|
|
/// Extracts an 8-bit to 64-bit unsigned big-endian value from the given byte
|
|
|
|
/// buffer and returns it as a 64-bit value.
|
|
|
|
///
|
|
|
|
/// Arguments:
|
|
|
|
///
|
|
|
|
/// * `data`: The buffer in which to extract the value.
|
|
|
|
/// * `start`: The offset at which to extract the value.
|
|
|
|
/// * `size`: The size of the value in bytes to extract. This must be 8 or
|
2014-10-09 15:17:22 -04:00
|
|
|
/// less, or task panic occurs. If this is less than 8, then only
|
2014-03-16 15:59:04 -07:00
|
|
|
/// that many bytes are parsed. For example, if `size` is 4, then a
|
|
|
|
/// 32-bit value is parsed.
|
|
|
|
pub fn u64_from_be_bytes(data: &[u8], start: uint, size: uint) -> u64 {
|
2014-02-10 16:50:42 -05:00
|
|
|
use ptr::{copy_nonoverlapping_memory};
|
2014-12-11 09:44:17 -08:00
|
|
|
use slice::SliceExt;
|
2014-01-09 10:35:36 +01:00
|
|
|
|
|
|
|
assert!(size <= 8u);
|
|
|
|
|
|
|
|
if data.len() - start < size {
|
2014-10-09 15:17:22 -04:00
|
|
|
panic!("index out of bounds");
|
2014-01-09 10:35:36 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
let mut buf = [0u8, ..8];
|
|
|
|
unsafe {
|
2014-02-10 16:50:42 -05:00
|
|
|
let ptr = data.as_ptr().offset(start as int);
|
2014-01-09 10:35:36 +01:00
|
|
|
let out = buf.as_mut_ptr();
|
2014-02-10 16:50:42 -05:00
|
|
|
copy_nonoverlapping_memory(out.offset((8 - size) as int), ptr, size);
|
2014-09-17 10:47:05 -07:00
|
|
|
(*(out as *const u64)).to_be()
|
2013-10-21 23:06:12 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-05-09 17:37:31 -07:00
|
|
|
#[cfg(test)]
|
|
|
|
mod test {
|
2014-01-06 22:33:37 -08:00
|
|
|
use prelude::*;
|
2014-01-30 14:10:53 -08:00
|
|
|
use io;
|
2014-11-11 16:01:29 -05:00
|
|
|
use io::{MemReader, BytesReader};
|
2013-10-30 18:28:37 -07:00
|
|
|
|
|
|
|
struct InitialZeroByteReader {
|
|
|
|
count: int,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Reader for InitialZeroByteReader {
|
2014-01-30 14:10:53 -08:00
|
|
|
fn read(&mut self, buf: &mut [u8]) -> io::IoResult<uint> {
|
2013-10-30 18:28:37 -07:00
|
|
|
if self.count == 0 {
|
|
|
|
self.count = 1;
|
2014-01-30 14:10:53 -08:00
|
|
|
Ok(0)
|
2013-10-30 18:28:37 -07:00
|
|
|
} else {
|
|
|
|
buf[0] = 10;
|
2014-01-30 14:10:53 -08:00
|
|
|
Ok(1)
|
2013-10-30 18:28:37 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
struct EofReader;
|
|
|
|
|
|
|
|
impl Reader for EofReader {
|
2014-01-30 14:10:53 -08:00
|
|
|
fn read(&mut self, _: &mut [u8]) -> io::IoResult<uint> {
|
|
|
|
Err(io::standard_error(io::EndOfFile))
|
2013-10-30 18:28:37 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
struct ErroringReader;
|
|
|
|
|
|
|
|
impl Reader for ErroringReader {
|
2014-01-30 14:10:53 -08:00
|
|
|
fn read(&mut self, _: &mut [u8]) -> io::IoResult<uint> {
|
|
|
|
Err(io::standard_error(io::InvalidInput))
|
2013-10-30 18:28:37 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
struct PartialReader {
|
|
|
|
count: int,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Reader for PartialReader {
|
2014-01-30 14:10:53 -08:00
|
|
|
fn read(&mut self, buf: &mut [u8]) -> io::IoResult<uint> {
|
2013-10-30 18:28:37 -07:00
|
|
|
if self.count == 0 {
|
|
|
|
self.count = 1;
|
|
|
|
buf[0] = 10;
|
|
|
|
buf[1] = 11;
|
2014-01-30 14:10:53 -08:00
|
|
|
Ok(2)
|
2013-10-30 18:28:37 -07:00
|
|
|
} else {
|
|
|
|
buf[0] = 12;
|
|
|
|
buf[1] = 13;
|
2014-01-30 14:10:53 -08:00
|
|
|
Ok(2)
|
2013-10-30 18:28:37 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
struct ErroringLaterReader {
|
|
|
|
count: int,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Reader for ErroringLaterReader {
|
2014-01-30 14:10:53 -08:00
|
|
|
fn read(&mut self, buf: &mut [u8]) -> io::IoResult<uint> {
|
2013-10-30 18:28:37 -07:00
|
|
|
if self.count == 0 {
|
|
|
|
self.count = 1;
|
|
|
|
buf[0] = 10;
|
2014-01-30 14:10:53 -08:00
|
|
|
Ok(1)
|
2013-10-30 18:28:37 -07:00
|
|
|
} else {
|
2014-01-30 14:10:53 -08:00
|
|
|
Err(io::standard_error(io::InvalidInput))
|
2013-10-30 18:28:37 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
struct ThreeChunkReader {
|
|
|
|
count: int,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Reader for ThreeChunkReader {
|
2014-01-30 14:10:53 -08:00
|
|
|
fn read(&mut self, buf: &mut [u8]) -> io::IoResult<uint> {
|
2013-10-30 18:28:37 -07:00
|
|
|
if self.count == 0 {
|
|
|
|
self.count = 1;
|
|
|
|
buf[0] = 10;
|
|
|
|
buf[1] = 11;
|
2014-01-30 14:10:53 -08:00
|
|
|
Ok(2)
|
2013-10-30 18:28:37 -07:00
|
|
|
} else if self.count == 1 {
|
|
|
|
self.count = 2;
|
|
|
|
buf[0] = 12;
|
|
|
|
buf[1] = 13;
|
2014-01-30 14:10:53 -08:00
|
|
|
Ok(2)
|
2013-10-30 18:28:37 -07:00
|
|
|
} else {
|
2014-01-30 14:10:53 -08:00
|
|
|
Err(io::standard_error(io::EndOfFile))
|
2013-10-30 18:28:37 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2013-05-09 17:37:31 -07:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn read_byte() {
|
2014-03-26 22:46:25 -07:00
|
|
|
let mut reader = MemReader::new(vec!(10));
|
2013-05-09 17:37:31 -07:00
|
|
|
let byte = reader.read_byte();
|
2014-01-30 14:10:53 -08:00
|
|
|
assert!(byte == Ok(10));
|
2013-05-09 17:37:31 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn read_byte_0_bytes() {
|
2013-10-30 18:28:37 -07:00
|
|
|
let mut reader = InitialZeroByteReader {
|
|
|
|
count: 0,
|
2013-05-09 17:37:31 -07:00
|
|
|
};
|
|
|
|
let byte = reader.read_byte();
|
2014-01-30 14:10:53 -08:00
|
|
|
assert!(byte == Ok(10));
|
2013-05-09 17:37:31 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn read_byte_eof() {
|
2013-10-30 18:28:37 -07:00
|
|
|
let mut reader = EofReader;
|
2013-05-09 17:37:31 -07:00
|
|
|
let byte = reader.read_byte();
|
2014-01-30 14:10:53 -08:00
|
|
|
assert!(byte.is_err());
|
2013-05-09 17:37:31 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn read_byte_error() {
|
2013-10-30 18:28:37 -07:00
|
|
|
let mut reader = ErroringReader;
|
2014-01-30 14:10:53 -08:00
|
|
|
let byte = reader.read_byte();
|
|
|
|
assert!(byte.is_err());
|
2013-05-09 17:37:31 -07:00
|
|
|
}
|
|
|
|
|
2013-09-02 01:30:21 +02:00
|
|
|
#[test]
|
|
|
|
fn bytes_0_bytes() {
|
2013-12-08 16:12:07 +09:00
|
|
|
let mut reader = InitialZeroByteReader {
|
2013-10-30 18:28:37 -07:00
|
|
|
count: 0,
|
2013-09-02 01:30:21 +02:00
|
|
|
};
|
|
|
|
let byte = reader.bytes().next();
|
2014-02-19 21:53:46 -05:00
|
|
|
assert!(byte == Some(Ok(10)));
|
2013-09-02 01:30:21 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn bytes_eof() {
|
2013-12-08 16:12:07 +09:00
|
|
|
let mut reader = EofReader;
|
2013-09-02 01:30:21 +02:00
|
|
|
let byte = reader.bytes().next();
|
2014-01-30 14:10:53 -08:00
|
|
|
assert!(byte.is_none());
|
2013-09-02 01:30:21 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn bytes_error() {
|
2013-12-08 16:12:07 +09:00
|
|
|
let mut reader = ErroringReader;
|
2013-09-02 01:30:21 +02:00
|
|
|
let mut it = reader.bytes();
|
2014-01-30 14:10:53 -08:00
|
|
|
let byte = it.next();
|
2014-02-19 21:53:46 -05:00
|
|
|
assert!(byte.unwrap().is_err());
|
2013-09-02 01:30:21 +02:00
|
|
|
}
|
|
|
|
|
2013-05-09 17:37:31 -07:00
|
|
|
#[test]
|
|
|
|
fn read_bytes() {
|
2014-03-26 22:46:25 -07:00
|
|
|
let mut reader = MemReader::new(vec!(10, 11, 12, 13));
|
2014-03-14 22:42:24 -07:00
|
|
|
let bytes = reader.read_exact(4).unwrap();
|
2014-03-26 09:24:16 -07:00
|
|
|
assert!(bytes == vec!(10, 11, 12, 13));
|
2013-05-09 17:37:31 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn read_bytes_partial() {
|
2013-10-30 18:28:37 -07:00
|
|
|
let mut reader = PartialReader {
|
|
|
|
count: 0,
|
2013-05-09 17:37:31 -07:00
|
|
|
};
|
2014-03-14 22:42:24 -07:00
|
|
|
let bytes = reader.read_exact(4).unwrap();
|
2014-03-26 09:24:16 -07:00
|
|
|
assert!(bytes == vec!(10, 11, 12, 13));
|
2013-05-09 17:37:31 -07:00
|
|
|
}
|
|
|
|
|
2013-05-13 15:23:52 -07:00
|
|
|
#[test]
|
|
|
|
fn read_bytes_eof() {
|
2014-03-26 22:46:25 -07:00
|
|
|
let mut reader = MemReader::new(vec!(10, 11));
|
2014-03-14 22:42:24 -07:00
|
|
|
assert!(reader.read_exact(4).is_err());
|
2013-05-13 15:23:52 -07:00
|
|
|
}
|
|
|
|
|
2013-05-09 17:37:31 -07:00
|
|
|
#[test]
|
2014-03-24 23:22:23 -07:00
|
|
|
fn push_at_least() {
|
|
|
|
let mut reader = MemReader::new(vec![10, 11, 12, 13]);
|
|
|
|
let mut buf = vec![8, 9];
|
|
|
|
assert!(reader.push_at_least(4, 4, &mut buf).is_ok());
|
|
|
|
assert!(buf == vec![8, 9, 10, 11, 12, 13]);
|
2013-05-09 17:37:31 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2014-03-24 23:22:23 -07:00
|
|
|
fn push_at_least_partial() {
|
2013-10-30 18:28:37 -07:00
|
|
|
let mut reader = PartialReader {
|
|
|
|
count: 0,
|
2013-05-09 17:37:31 -07:00
|
|
|
};
|
2014-03-24 23:22:23 -07:00
|
|
|
let mut buf = vec![8, 9];
|
|
|
|
assert!(reader.push_at_least(4, 4, &mut buf).is_ok());
|
|
|
|
assert!(buf == vec![8, 9, 10, 11, 12, 13]);
|
2013-05-09 17:37:31 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2014-03-24 23:22:23 -07:00
|
|
|
fn push_at_least_eof() {
|
|
|
|
let mut reader = MemReader::new(vec![10, 11]);
|
|
|
|
let mut buf = vec![8, 9];
|
|
|
|
assert!(reader.push_at_least(4, 4, &mut buf).is_err());
|
|
|
|
assert!(buf == vec![8, 9, 10, 11]);
|
2013-05-09 17:37:31 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2014-03-24 23:22:23 -07:00
|
|
|
fn push_at_least_error() {
|
2013-10-30 18:28:37 -07:00
|
|
|
let mut reader = ErroringLaterReader {
|
|
|
|
count: 0,
|
2013-05-09 17:37:31 -07:00
|
|
|
};
|
2014-03-24 23:22:23 -07:00
|
|
|
let mut buf = vec![8, 9];
|
|
|
|
assert!(reader.push_at_least(4, 4, &mut buf).is_err());
|
|
|
|
assert!(buf == vec![8, 9, 10]);
|
2013-05-09 17:37:31 -07:00
|
|
|
}
|
|
|
|
|
2013-05-13 16:56:16 -07:00
|
|
|
#[test]
|
|
|
|
fn read_to_end() {
|
2013-10-30 18:28:37 -07:00
|
|
|
let mut reader = ThreeChunkReader {
|
|
|
|
count: 0,
|
2013-05-13 16:56:16 -07:00
|
|
|
};
|
2014-01-30 14:10:53 -08:00
|
|
|
let buf = reader.read_to_end().unwrap();
|
2014-03-26 09:24:16 -07:00
|
|
|
assert!(buf == vec!(10, 11, 12, 13));
|
2013-05-13 16:56:16 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
#[should_fail]
|
|
|
|
fn read_to_end_error() {
|
2013-10-30 18:28:37 -07:00
|
|
|
let mut reader = ThreeChunkReader {
|
|
|
|
count: 0,
|
2013-05-13 16:56:16 -07:00
|
|
|
};
|
2014-01-30 14:10:53 -08:00
|
|
|
let buf = reader.read_to_end().unwrap();
|
2014-03-26 09:24:16 -07:00
|
|
|
assert!(buf == vec!(10, 11));
|
2013-05-13 16:56:16 -07:00
|
|
|
}
|
|
|
|
|
2013-07-12 14:43:57 -07:00
|
|
|
#[test]
|
|
|
|
fn test_read_write_le_mem() {
|
2014-01-25 20:37:51 +13:00
|
|
|
let uints = [0, 1, 2, 42, 10_123, 100_123_456, ::u64::MAX];
|
2013-05-13 19:14:14 -07:00
|
|
|
|
2014-11-11 16:01:29 -05:00
|
|
|
let mut writer = Vec::new();
|
2013-07-12 14:43:57 -07:00
|
|
|
for i in uints.iter() {
|
2014-01-30 14:10:53 -08:00
|
|
|
writer.write_le_u64(*i).unwrap();
|
2013-05-13 19:14:14 -07:00
|
|
|
}
|
|
|
|
|
2014-11-11 16:01:29 -05:00
|
|
|
let mut reader = MemReader::new(writer);
|
2013-07-12 14:43:57 -07:00
|
|
|
for i in uints.iter() {
|
2014-01-30 14:10:53 -08:00
|
|
|
assert!(reader.read_le_u64().unwrap() == *i);
|
2013-05-13 19:14:14 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-07-12 14:43:57 -07:00
|
|
|
|
2013-05-13 19:14:14 -07:00
|
|
|
#[test]
|
|
|
|
fn test_read_write_be() {
|
2014-01-25 20:37:51 +13:00
|
|
|
let uints = [0, 1, 2, 42, 10_123, 100_123_456, ::u64::MAX];
|
2013-05-13 19:14:14 -07:00
|
|
|
|
2014-11-11 16:01:29 -05:00
|
|
|
let mut writer = Vec::new();
|
2013-07-12 14:43:57 -07:00
|
|
|
for i in uints.iter() {
|
2014-01-30 14:10:53 -08:00
|
|
|
writer.write_be_u64(*i).unwrap();
|
2013-05-13 19:14:14 -07:00
|
|
|
}
|
|
|
|
|
2014-11-11 16:01:29 -05:00
|
|
|
let mut reader = MemReader::new(writer);
|
2013-07-12 14:43:57 -07:00
|
|
|
for i in uints.iter() {
|
2014-01-30 14:10:53 -08:00
|
|
|
assert!(reader.read_be_u64().unwrap() == *i);
|
2013-05-13 19:14:14 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_read_be_int_n() {
|
2014-01-25 20:37:51 +13:00
|
|
|
let ints = [::i32::MIN, -123456, -42, -5, 0, 1, ::i32::MAX];
|
2013-05-13 19:14:14 -07:00
|
|
|
|
2014-11-11 16:01:29 -05:00
|
|
|
let mut writer = Vec::new();
|
2013-07-12 14:43:57 -07:00
|
|
|
for i in ints.iter() {
|
2014-01-30 14:10:53 -08:00
|
|
|
writer.write_be_i32(*i).unwrap();
|
2013-05-13 19:14:14 -07:00
|
|
|
}
|
|
|
|
|
2014-11-11 16:01:29 -05:00
|
|
|
let mut reader = MemReader::new(writer);
|
2013-07-12 14:43:57 -07:00
|
|
|
for i in ints.iter() {
|
2013-05-13 19:14:14 -07:00
|
|
|
// this tests that the sign extension is working
|
|
|
|
// (comparing the values as i32 would not test this)
|
2014-01-30 14:10:53 -08:00
|
|
|
assert!(reader.read_be_int_n(4).unwrap() == *i as i64);
|
2013-05-13 19:14:14 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_read_f32() {
|
|
|
|
//big-endian floating-point 8.1250
|
2014-06-06 10:27:49 -07:00
|
|
|
let buf = vec![0x41, 0x02, 0x00, 0x00];
|
2013-05-13 19:14:14 -07:00
|
|
|
|
2014-11-11 16:01:29 -05:00
|
|
|
let mut writer = Vec::new();
|
2014-06-06 10:27:49 -07:00
|
|
|
writer.write(buf.as_slice()).unwrap();
|
2013-05-13 19:14:14 -07:00
|
|
|
|
2014-11-11 16:01:29 -05:00
|
|
|
let mut reader = MemReader::new(writer);
|
2014-01-30 14:10:53 -08:00
|
|
|
let f = reader.read_be_f32().unwrap();
|
2013-05-13 19:14:14 -07:00
|
|
|
assert!(f == 8.1250);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_read_write_f32() {
|
|
|
|
let f:f32 = 8.1250;
|
|
|
|
|
2014-11-11 16:01:29 -05:00
|
|
|
let mut writer = Vec::new();
|
2014-01-30 14:10:53 -08:00
|
|
|
writer.write_be_f32(f).unwrap();
|
|
|
|
writer.write_le_f32(f).unwrap();
|
2013-05-13 19:14:14 -07:00
|
|
|
|
2014-11-11 16:01:29 -05:00
|
|
|
let mut reader = MemReader::new(writer);
|
2014-01-30 14:10:53 -08:00
|
|
|
assert!(reader.read_be_f32().unwrap() == 8.1250);
|
|
|
|
assert!(reader.read_le_f32().unwrap() == 8.1250);
|
2013-07-12 14:43:57 -07:00
|
|
|
}
|
2013-05-13 19:14:14 -07:00
|
|
|
|
2014-01-09 00:20:14 +01:00
|
|
|
#[test]
|
|
|
|
fn test_u64_from_be_bytes() {
|
|
|
|
use super::u64_from_be_bytes;
|
|
|
|
|
|
|
|
let buf = [0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0x09];
|
|
|
|
|
|
|
|
// Aligned access
|
2014-11-17 21:39:01 +13:00
|
|
|
assert_eq!(u64_from_be_bytes(&buf, 0, 0), 0);
|
|
|
|
assert_eq!(u64_from_be_bytes(&buf, 0, 1), 0x01);
|
|
|
|
assert_eq!(u64_from_be_bytes(&buf, 0, 2), 0x0102);
|
|
|
|
assert_eq!(u64_from_be_bytes(&buf, 0, 3), 0x010203);
|
|
|
|
assert_eq!(u64_from_be_bytes(&buf, 0, 4), 0x01020304);
|
|
|
|
assert_eq!(u64_from_be_bytes(&buf, 0, 5), 0x0102030405);
|
|
|
|
assert_eq!(u64_from_be_bytes(&buf, 0, 6), 0x010203040506);
|
|
|
|
assert_eq!(u64_from_be_bytes(&buf, 0, 7), 0x01020304050607);
|
|
|
|
assert_eq!(u64_from_be_bytes(&buf, 0, 8), 0x0102030405060708);
|
2014-01-09 00:20:14 +01:00
|
|
|
|
|
|
|
// Unaligned access
|
2014-11-17 21:39:01 +13:00
|
|
|
assert_eq!(u64_from_be_bytes(&buf, 1, 0), 0);
|
|
|
|
assert_eq!(u64_from_be_bytes(&buf, 1, 1), 0x02);
|
|
|
|
assert_eq!(u64_from_be_bytes(&buf, 1, 2), 0x0203);
|
|
|
|
assert_eq!(u64_from_be_bytes(&buf, 1, 3), 0x020304);
|
|
|
|
assert_eq!(u64_from_be_bytes(&buf, 1, 4), 0x02030405);
|
|
|
|
assert_eq!(u64_from_be_bytes(&buf, 1, 5), 0x0203040506);
|
|
|
|
assert_eq!(u64_from_be_bytes(&buf, 1, 6), 0x020304050607);
|
|
|
|
assert_eq!(u64_from_be_bytes(&buf, 1, 7), 0x02030405060708);
|
|
|
|
assert_eq!(u64_from_be_bytes(&buf, 1, 8), 0x0203040506070809);
|
2014-01-09 00:20:14 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod bench {
|
2014-02-14 09:49:11 +08:00
|
|
|
extern crate test;
|
2014-04-17 15:28:14 -07:00
|
|
|
|
|
|
|
use prelude::*;
|
|
|
|
use self::test::Bencher;
|
2014-01-09 00:20:14 +01:00
|
|
|
|
2014-07-03 13:10:54 -07:00
|
|
|
// why is this a macro? wouldn't an inlined function work just as well?
|
2014-11-14 09:18:10 -08:00
|
|
|
macro_rules! u64_from_be_bytes_bench_impl {
|
2014-07-03 13:10:54 -07:00
|
|
|
($b:expr, $size:expr, $stride:expr, $start_index:expr) =>
|
2014-01-09 00:20:14 +01:00
|
|
|
({
|
|
|
|
use super::u64_from_be_bytes;
|
|
|
|
|
2014-04-17 15:28:14 -07:00
|
|
|
let data = Vec::from_fn($stride*100+$start_index, |i| i as u8);
|
2014-01-09 00:20:14 +01:00
|
|
|
let mut sum = 0u64;
|
2014-07-03 13:10:54 -07:00
|
|
|
$b.iter(|| {
|
2014-01-09 00:20:14 +01:00
|
|
|
let mut i = $start_index;
|
2014-01-19 19:21:14 +11:00
|
|
|
while i < data.len() {
|
2014-04-17 15:28:14 -07:00
|
|
|
sum += u64_from_be_bytes(data.as_slice(), i, $size);
|
2014-01-09 00:20:14 +01:00
|
|
|
i += $stride;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
})
|
2014-11-14 09:18:10 -08:00
|
|
|
}
|
2014-01-09 00:20:14 +01:00
|
|
|
|
|
|
|
#[bench]
|
2014-04-01 09:16:35 +08:00
|
|
|
fn u64_from_be_bytes_4_aligned(b: &mut Bencher) {
|
2014-07-03 13:10:54 -07:00
|
|
|
u64_from_be_bytes_bench_impl!(b, 4, 4, 0);
|
2014-01-09 00:20:14 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
#[bench]
|
2014-04-01 09:16:35 +08:00
|
|
|
fn u64_from_be_bytes_4_unaligned(b: &mut Bencher) {
|
2014-07-03 13:10:54 -07:00
|
|
|
u64_from_be_bytes_bench_impl!(b, 4, 4, 1);
|
2014-01-09 00:20:14 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
#[bench]
|
2014-04-01 09:16:35 +08:00
|
|
|
fn u64_from_be_bytes_7_aligned(b: &mut Bencher) {
|
2014-07-03 13:10:54 -07:00
|
|
|
u64_from_be_bytes_bench_impl!(b, 7, 8, 0);
|
2014-01-09 00:20:14 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
#[bench]
|
2014-04-01 09:16:35 +08:00
|
|
|
fn u64_from_be_bytes_7_unaligned(b: &mut Bencher) {
|
2014-07-03 13:10:54 -07:00
|
|
|
u64_from_be_bytes_bench_impl!(b, 7, 8, 1);
|
2014-01-09 00:20:14 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
#[bench]
|
2014-04-01 09:16:35 +08:00
|
|
|
fn u64_from_be_bytes_8_aligned(b: &mut Bencher) {
|
2014-07-03 13:10:54 -07:00
|
|
|
u64_from_be_bytes_bench_impl!(b, 8, 8, 0);
|
2014-01-09 00:20:14 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
#[bench]
|
2014-04-01 09:16:35 +08:00
|
|
|
fn u64_from_be_bytes_8_unaligned(b: &mut Bencher) {
|
2014-07-03 13:10:54 -07:00
|
|
|
u64_from_be_bytes_bench_impl!(b, 8, 8, 1);
|
2014-01-09 00:20:14 +01:00
|
|
|
}
|
2013-05-09 17:37:31 -07:00
|
|
|
}
|