#[derive(Debug, PartialEq, Eq, PartialOrd, Ord, Hash)]
#[repr(transparent)]
pub struct StreamingSocketFileDescriptor<SD: SocketData>(SocketFileDescriptor<SD>);
impl<SD: SocketData> AsRawFd for StreamingSocketFileDescriptor<SD>
{
#[inline(always)]
fn as_raw_fd(&self) -> RawFd
{
self.0.as_raw_fd()
}
}
impl<SD: SocketData> FromRawFd for StreamingSocketFileDescriptor<SD>
{
#[inline(always)]
unsafe fn from_raw_fd(fd: RawFd) -> Self
{
Self(SocketFileDescriptor::from_raw_fd(fd))
}
}
impl<SD: SocketData> IntoRawFd for StreamingSocketFileDescriptor<SD>
{
#[inline(always)]
fn into_raw_fd(self) -> RawFd
{
self.0.into_raw_fd()
}
}
impl<SD: SocketData> FileDescriptor for StreamingSocketFileDescriptor<SD>
{
}
impl<SD: SocketData> Deref for StreamingSocketFileDescriptor<SD>
{
type Target = SocketFileDescriptor<SD>;
#[inline(always)]
fn deref(&self) -> &Self::Target
{
&self.0
}
}
impl<SD: SocketData> VectoredRead for StreamingSocketFileDescriptor<SD>
{
vectored_read!();
}
impl<SD: SocketData> VectoredWrite for StreamingSocketFileDescriptor<SD>
{
vectored_write!();
}
impl<SD: SocketData> PipeLikeFileDescriptor for StreamingSocketFileDescriptor<SD>
{
}
impl<SD: SocketData> SpliceRecipient for StreamingSocketFileDescriptor<SD>
{
}
impl<SD: SocketData> SpliceSender for StreamingSocketFileDescriptor<SD>
{
}
impl<SD: SocketData> SendFile for StreamingSocketFileDescriptor<SD>
{
#[inline(always)]
fn write_output_from_file<F: AsRef<File>>(&self, from_file: &F, maximum_number_of_bytes_to_transfer: usize) -> Result<usize, StructWriteError>
{
if unlikely!(maximum_number_of_bytes_to_transfer == 0)
{
return Ok(0)
}
let result = unsafe { sendfile(self.as_raw_fd(), from_file.as_ref().as_raw_fd(), null_mut(), maximum_number_of_bytes_to_transfer) };
if likely!(result >= 0)
{
Ok(result as usize)
}
else if likely!(result == -1)
{
use self::StructWriteError::*;
Err
(
match errno().0
{
EAGAIN | ENOMEM => WouldBlock,
EINTR => Interrupted,
EIO => Cancelled,
EBADF => panic!("The input file was not opened for reading or the output file was not opened for writing"),
EFAULT => panic!("Bad address"),
EINVAL => panic!("Descriptor is not valid or locked, or an mmap(2)-like operation is not available for in_fd"),
_ => unreachable_code(format_args!("")),
}
)
}
else
{
unreachable_code(format_args!(""))
}
}
#[inline(always)]
fn write_output_from_file_with_offset<F: AsRef<File>>(&self, from_file: &F, mut offset: i64, maximum_number_of_bytes_to_transfer: usize) -> Result<(usize, i64), StructWriteError>
{
if unlikely!(maximum_number_of_bytes_to_transfer == 0)
{
return Ok((0, offset))
}
let result = unsafe { sendfile(self.as_raw_fd(), from_file.as_ref().as_raw_fd(), &mut offset, maximum_number_of_bytes_to_transfer) };
if likely!(result >= 0)
{
Ok((result as usize, offset))
}
else if likely!(result == -1)
{
use self::StructWriteError::*;
Err
(
match errno().0
{
EAGAIN | ENOMEM => WouldBlock,
EINTR => Interrupted,
EIO => Cancelled,
EBADF => panic!("The input file was not opened for reading or the output file was not opened for writing"),
EFAULT => panic!("Bad address"),
EINVAL => panic!("Descriptor is not valid or locked, or an mmap(2)-like operation is not available for in_fd"),
_ => unreachable_code(format_args!("")),
}
)
}
else
{
unreachable_code(format_args!(""))
}
}
}
impl<SD: SocketData> Read for StreamingSocketFileDescriptor<SD>
{
#[inline(always)]
fn read(&mut self, buf: &mut [u8]) -> io::Result<usize>
{
self.receive_from(buf)
}
#[inline(always)]
fn read_vectored(&mut self, bufs: &mut [IoSliceMut<'_>]) -> io::Result<usize>
{
VectoredRead::read_vectored(self, unsafe { transmute(bufs) })
}
#[inline(always)]
unsafe fn initializer(&self) -> Initializer
{
Initializer::nop()
}
}
impl<SD: SocketData> Write for StreamingSocketFileDescriptor<SD>
{
#[inline(always)]
fn write(&mut self, buf: &[u8]) -> io::Result<usize>
{
self.send_to(buf)
}
#[inline(always)]
fn write_vectored(&mut self, bufs: &[IoSlice<'_>]) -> io::Result<usize>
{
VectoredWrite::write_vectored(self, unsafe { transmute(bufs) })
}
#[inline(always)]
fn flush(&mut self) -> io::Result<()>
{
Ok(())
}
}
impl<SD: SocketData> StreamingSocketFileDescriptor<SD>
{
#[inline(always)]
pub fn shutdown(&self)
{
let result = unsafe { shutdown(self.as_raw_fd(), SHUT_RDWR) };
if likely!(result == 0)
{
return
}
else if likely!(result != -1)
{
match errno().0
{
EBADF => panic!("The argument `sockfd` is an invalid descriptor"),
EINVAL => panic!("An invalid value was specified in `how`"),
ENOTCONN => panic!("The socket is associated with a connection-oriented protocol and has not been connected"),
ENOTSOCK => panic!("The argument `sockfd` does not refer to a socket"),
_ => unreachable_code(format_args!("")),
}
}
else
{
unreachable_code(format_args!(""))
}
}
#[inline(always)]
pub fn receive_from(&self, buf: &mut [u8]) -> io::Result<usize>
{
let length = buf.len();
if unlikely!(length == 0)
{
return Ok(0)
}
let result = unsafe { recvfrom(self.as_raw_fd(), buf.as_mut_ptr() as *mut c_void, length, ReceiveFlags::empty().bits, null(), null_mut()) };
if likely!(result > 0)
{
Ok(result as usize)
}
else
{
use crate::ErrorKind::*;
Err
(
io::Error::from
(
if likely!(result == 0)
{
UnexpectedEof
}
else if likely!(result == -1)
{
match errno().0
{
EAGAIN => WouldBlock,
EINTR => Interrupted,
ENOMEM => Other,
ECONNRESET => ConnectionReset,
ECONNREFUSED => ConnectionRefused,
EBADF => panic!("The argument `sockfd` is an invalid descriptor"),
EFAULT => panic!("The receive buffer pointer(s) point outside the process's address space"),
EINVAL => panic!("Invalid argument passed"),
ENOTCONN => panic!("The socket is associated with a connection-oriented protocol and has not been connected"),
ENOTSOCK => panic!("The argument `sockfd` does not refer to a socket"),
EOPNOTSUPP => panic!("Some flags in the `flags` argument are inappropriate for the socket type"),
_ => unreachable_code(format_args!("")),
}
}
else
{
unreachable_code(format_args!(""))
}
)
)
}
}
#[inline(always)]
pub fn send_to(&self, buf: &[u8]) -> io::Result<usize>
{
let length = buf.len();
if unlikely!(length == 0)
{
return Ok(0)
}
let result = unsafe { send(self.as_raw_fd(), buf.as_ptr() as *const c_void, buf.len(), SendFlags::NoSigPipeSignal.bits) };
if likely!(result > 0)
{
Ok(result as usize)
}
else
{
use crate::ErrorKind::*;
Err
(
io::Error::from
(
if likely!(result == 0)
{
WriteZero
}
else if likely!(result == -1)
{
match errno().0
{
EAGAIN => WouldBlock,
EINTR => Interrupted,
ENOMEM | ENOBUFS => Other,
EPIPE => BrokenPipe,
EACCES => PermissionDenied,
ECONNRESET => ConnectionReset,
EBADF => panic!("The argument `sockfd` is an invalid descriptor"),
EFAULT => panic!("The receive buffer pointer(s) point outside the process's address space"),
EINVAL => panic!("Invalid argument passed"),
ENOTCONN => panic!("The socket is associated with a connection-oriented protocol and has not been connected"),
ENOTSOCK => panic!("The argument `sockfd` does not refer to a socket"),
EOPNOTSUPP => panic!("Some flags in the `flags` argument are inappropriate for the socket type"),
EMSGSIZE => panic!("The socket type requires that message be sent atomically, and the size of the message to be sent made this impossible"),
EISCONN => panic!("The connection-mode socket was connected already but a recipient was specified"),
EDESTADDRREQ => panic!("The socket is not connection-mode, and no peer address is set"),
_ => unreachable_code(format_args!("")),
}
}
else
{
unreachable_code(format_args!(""))
}
)
)
}
}
}
impl<SD: SocketData> SocketConnect for StreamingSocketFileDescriptor<SD>
{
}
impl<SD: SocketData> NonServerSocket for StreamingSocketFileDescriptor<SD>
{
}
impl StreamingSocketFileDescriptor<sockaddr_un>
{
#[inline(always)]
pub fn remote_peer_credentials(&self) -> Credentials
{
self.0.remote_peer_credentials()
}
pub fn receive_file_descriptors(&self, maximum_file_descriptors_to_receive: usize) -> Result<Vec<RawFd>, ReceiveFileDescriptorsError>
{
self.0.receive_file_descriptors(maximum_file_descriptors_to_receive)
}
#[inline(always)]
pub fn send_file_descriptors(&self, file_descriptors: &[RawFd]) -> io::Result<()>
{
self.0.send_file_descriptors(file_descriptors)
}
#[inline(always)]
pub fn send_credentials(&self, credentials: Credentials) -> io::Result<()>
{
self.0.send_credentials(credentials)
}
}