pub use self::error::{Error, ErrorKind, RawOsError, Result};
#[macro_use]
mod error;
mod impls;
#[cfg(feature = "stdio")]
pub use self::stdio::{IsTerminal, Stderr, Stdin, Stdout, stderr, stdin, stdout};
#[cfg(feature = "stdio")]
mod stdio;
use core::fmt;
const _: fn() = || {
fn assert_dyn_compatibility<T: ?Sized>() {}
assert_dyn_compatibility::<dyn Read>();
assert_dyn_compatibility::<dyn Write>();
assert_dyn_compatibility::<dyn Seek>();
};
pub(crate) fn default_read_exact<R: ?Sized + Read>(this: &mut R, mut buf: &mut [u8]) -> Result<()> {
while !buf.is_empty() {
match this.read(buf) {
Ok(0) => break,
Ok(n) => {
buf = &mut buf[n..];
}
Err(ref e) if e.is_interrupted() => {}
Err(e) => return Err(e),
}
}
if buf.is_empty() { Ok(()) } else { Err(Error::READ_EXACT_EOF) }
}
pub trait Read {
fn read(&mut self, buf: &mut [u8]) -> Result<usize>;
fn read_exact(&mut self, buf: &mut [u8]) -> Result<()> {
default_read_exact(self, buf)
}
}
pub trait Write {
fn write(&mut self, buf: &[u8]) -> Result<usize>;
fn flush(&mut self) -> Result<()>;
fn write_all(&mut self, mut buf: &[u8]) -> Result<()> {
while !buf.is_empty() {
match self.write(buf) {
Ok(0) => return Err(Error::WRITE_ALL_EOF),
Ok(n) => buf = &buf[n..],
Err(ref e) if e.is_interrupted() => {}
Err(e) => return Err(e),
}
}
Ok(())
}
fn write_fmt(&mut self, f: fmt::Arguments<'_>) -> Result<()> {
struct Adapter<'a, T: ?Sized> {
inner: &'a mut T,
error: Result<()>,
}
impl<T: ?Sized + Write> fmt::Write for Adapter<'_, T> {
fn write_str(&mut self, s: &str) -> fmt::Result {
match self.inner.write_all(s.as_bytes()) {
Ok(()) => Ok(()),
Err(e) => {
self.error = Err(e);
Err(fmt::Error)
}
}
}
}
let mut output = Adapter { inner: self, error: Ok(()) };
match fmt::write(&mut output, f) {
Ok(()) => Ok(()),
Err(..) => {
if output.error.is_err() {
output.error
} else {
panic!(
"a formatting trait implementation returned an error when the underlying stream did not"
);
}
}
}
}
}
pub trait Seek {
fn seek(&mut self, pos: SeekFrom) -> Result<u64>;
fn rewind(&mut self) -> Result<()> {
self.seek(SeekFrom::Start(0))?;
Ok(())
}
}
#[derive(Debug, Clone, Copy, PartialEq, Eq)]
#[non_exhaustive]
pub enum SeekFrom {
Start(u64),
End(i64),
}