use std::io;
use std::os::unix::io::{RawFd, AsRawFd, IntoRawFd};
use nix;
use nix::fcntl::{fcntl, FcntlArg};
use libc;
pub enum Stdio {
Pipe,
Inherit,
Null,
Fd(Closing),
}
pub enum Fd {
ReadPipe,
WritePipe,
Inherit,
ReadNull,
WriteNull,
Fd(Closing),
}
pub struct Closing(RawFd);
pub fn dup_file_cloexec<F: AsRawFd>(file: &F) -> io::Result<Closing> {
match fcntl(file.as_raw_fd(), FcntlArg::F_DUPFD_CLOEXEC(3)) {
Ok(fd) => Ok(Closing::new(fd)),
Err(nix::Error::Sys(errno)) => {
return Err(io::Error::from_raw_os_error(errno as i32));
}
Err(nix::Error::InvalidPath) => unreachable!(),
Err(nix::Error::InvalidUtf8) => unreachable!(),
Err(nix::Error::UnsupportedOperation) => {
return Err(io::Error::new(io::ErrorKind::Other,
"nix error: unsupported operation"));
}
}
}
impl Stdio {
pub fn piped() -> Stdio { Stdio::Pipe }
pub fn inherit() -> Stdio { Stdio::Inherit }
pub fn null() -> Stdio { Stdio::Null }
pub fn to_fd(self, write: bool) -> Fd {
match (self, write) {
(Stdio::Fd(x), _) => Fd::Fd(x),
(Stdio::Pipe, false) => Fd::ReadPipe,
(Stdio::Pipe, true) => Fd::WritePipe,
(Stdio::Inherit, _) => Fd::Inherit,
(Stdio::Null, false) => Fd::ReadNull,
(Stdio::Null, true) => Fd::WriteNull,
}
}
pub fn dup_file<F: AsRawFd>(file: &F) -> io::Result<Stdio> {
dup_file_cloexec(file).map(|f| Stdio::Fd(f))
}
pub fn from_file<F: IntoRawFd>(file: F) -> Stdio {
Stdio::Fd(Closing(file.into_raw_fd()))
}
}
impl Fd {
pub fn piped_read() -> Fd { Fd::ReadPipe }
pub fn piped_write() -> Fd { Fd::WritePipe }
pub fn inherit() -> Fd { Fd::Inherit }
pub fn read_null() -> Fd { Fd::ReadNull }
pub fn write_null() -> Fd { Fd::WriteNull }
pub fn dup_file<F: AsRawFd>(file: &F) -> io::Result<Fd> {
dup_file_cloexec(file).map(|f| Fd::Fd(f))
}
pub fn from_file<F: IntoRawFd>(file: F) -> Fd {
Fd::Fd(Closing(file.into_raw_fd()))
}
}
impl Closing {
pub fn new(fd: RawFd) -> Closing {
Closing(fd)
}
}
impl AsRawFd for Closing {
fn as_raw_fd(&self) -> RawFd {
return self.0;
}
}
impl Drop for Closing {
fn drop(&mut self) {
unsafe {
libc::close(self.0);
}
}
}