use {io, Evented, EventSet, Io, PollOpt, Selector, Token, TryAccept};
use io::MapNonBlock;
use sys::unix::{net, nix, Socket};
use std::io::{Read, Write};
use std::path::Path;
use std::os::unix::io::{RawFd, AsRawFd, FromRawFd};
use nix::sys::socket::MsgFlags;
#[derive(Debug)]
pub struct UnixSocket {
io: Io,
}
impl UnixSocket {
pub fn stream() -> io::Result<UnixSocket> {
UnixSocket::new(nix::SockType::Stream)
}
fn new(ty: nix::SockType) -> io::Result<UnixSocket> {
let fd = try!(net::socket(nix::AddressFamily::Unix, ty, true));
Ok(From::from(Io::from_raw_fd(fd)))
}
pub fn connect<P: AsRef<Path> + ?Sized>(&self, addr: &P) -> io::Result<bool> {
net::connect(&self.io, &try!(to_nix_addr(addr)))
}
pub fn listen(&self, backlog: usize) -> io::Result<()> {
net::listen(&self.io, backlog)
}
pub fn accept(&self) -> io::Result<Option<UnixSocket>> {
net::accept(&self.io, true)
.map(|fd| From::from(Io::from_raw_fd(fd)))
.map_non_block()
}
pub fn bind<P: AsRef<Path> + ?Sized>(&self, addr: &P) -> io::Result<()> {
net::bind(&self.io, &try!(to_nix_addr(addr)))
}
pub fn try_clone(&self) -> io::Result<UnixSocket> {
net::dup(&self.io)
.map(From::from)
}
pub fn read_recv_fd(&mut self, buf: &mut [u8]) -> io::Result<(usize, Option<RawFd>)> {
let iov = [nix::IoVec::from_mut_slice(buf)];
let mut cmsgspace: nix::CmsgSpace<[RawFd; 1]> = nix::CmsgSpace::new();
let msg = try!(nix::recvmsg(self.io.as_raw_fd(), &iov, Some(&mut cmsgspace), MsgFlags::empty())
.map_err(super::from_nix_error));
let mut fd = None;
for cmsg in msg.cmsgs() {
if let nix::ControlMessage::ScmRights(fds) = cmsg {
if fds.len() == 1 {
fd = Some(fds[0]);
break;
}
}
}
Ok((msg.bytes, fd))
}
pub fn write_send_fd(&mut self, buf: &[u8], fd: RawFd) -> io::Result<usize> {
let iov = [nix::IoVec::from_slice(buf)];
let fds = [fd];
let cmsg = nix::ControlMessage::ScmRights(&fds);
nix::sendmsg(self.io.as_raw_fd(),&iov, &[cmsg], MsgFlags::empty(), None)
.map_err(super::from_nix_error)
}
}
impl Read for UnixSocket {
fn read(&mut self, buf: &mut [u8]) -> io::Result<usize> {
self.io.read(buf)
}
}
impl Write for UnixSocket {
fn write(&mut self, buf: &[u8]) -> io::Result<usize> {
self.io.write(buf)
}
fn flush(&mut self) -> io::Result<()> {
self.io.flush()
}
}
impl Evented for UnixSocket {
fn register(&self, selector: &mut Selector, token: Token, interest: EventSet, opts: PollOpt) -> io::Result<()> {
self.io.register(selector, token, interest, opts)
}
fn reregister(&self, selector: &mut Selector, token: Token, interest: EventSet, opts: PollOpt) -> io::Result<()> {
self.io.reregister(selector, token, interest, opts)
}
fn deregister(&self, selector: &mut Selector) -> io::Result<()> {
self.io.deregister(selector)
}
}
impl TryAccept for UnixSocket {
type Output = UnixSocket;
fn accept(&self) -> io::Result<Option<UnixSocket>> {
UnixSocket::accept(self)
}
}
impl Socket for UnixSocket {
}
impl From<Io> for UnixSocket {
fn from(io: Io) -> UnixSocket {
UnixSocket { io: io }
}
}
impl FromRawFd for UnixSocket {
unsafe fn from_raw_fd(fd: RawFd) -> UnixSocket {
UnixSocket { io: Io::from_raw_fd(fd) }
}
}
impl AsRawFd for UnixSocket {
fn as_raw_fd(&self) -> RawFd {
self.io.as_raw_fd()
}
}
fn to_nix_addr<P: AsRef<Path> + ?Sized>(path: &P) -> io::Result<nix::SockAddr> {
nix::SockAddr::new_unix(path.as_ref())
.map_err(super::from_nix_error)
}