use std::{ffi::CString, marker::PhantomPinned, net::Shutdown, os::fd::OwnedFd};
use compio_buf::{
IntoInner, IoBuf, IoBufMut, IoSlice, IoSliceMut, IoVectoredBuf, IoVectoredBufMut,
};
use libc::{sockaddr_storage, socklen_t};
use socket2::SockAddr;
use crate::{SharedFd, op::*};
pub struct OpenFile {
pub(crate) path: CString,
pub(crate) flags: i32,
pub(crate) mode: libc::mode_t,
}
impl OpenFile {
pub fn new(path: CString, flags: i32, mode: libc::mode_t) -> Self {
Self { path, flags, mode }
}
}
#[cfg(gnulinux)]
pub(crate) type Statx = libc::statx;
#[cfg(all(target_os = "linux", not(target_env = "gnu")))]
#[repr(C)]
pub(crate) struct StatxTimestamp {
pub tv_sec: i64,
pub tv_nsec: u32,
pub __statx_timestamp_pad1: [i32; 1],
}
#[cfg(all(target_os = "linux", not(target_env = "gnu")))]
#[repr(C)]
pub(crate) struct Statx {
pub stx_mask: u32,
pub stx_blksize: u32,
pub stx_attributes: u64,
pub stx_nlink: u32,
pub stx_uid: u32,
pub stx_gid: u32,
pub stx_mode: u16,
__statx_pad1: [u16; 1],
pub stx_ino: u64,
pub stx_size: u64,
pub stx_blocks: u64,
pub stx_attributes_mask: u64,
pub stx_atime: StatxTimestamp,
pub stx_btime: StatxTimestamp,
pub stx_ctime: StatxTimestamp,
pub stx_mtime: StatxTimestamp,
pub stx_rdev_major: u32,
pub stx_rdev_minor: u32,
pub stx_dev_major: u32,
pub stx_dev_minor: u32,
pub stx_mnt_id: u64,
pub stx_dio_mem_align: u32,
pub stx_dio_offset_align: u32,
__statx_pad3: [u64; 12],
}
#[cfg(target_os = "linux")]
pub(crate) const fn statx_to_stat(statx: Statx) -> libc::stat {
let mut stat: libc::stat = unsafe { std::mem::zeroed() };
stat.st_dev = libc::makedev(statx.stx_dev_major, statx.stx_dev_minor);
stat.st_ino = statx.stx_ino;
stat.st_nlink = statx.stx_nlink as _;
stat.st_mode = statx.stx_mode as _;
stat.st_uid = statx.stx_uid;
stat.st_gid = statx.stx_gid;
stat.st_rdev = libc::makedev(statx.stx_rdev_major, statx.stx_rdev_minor);
stat.st_size = statx.stx_size as _;
stat.st_blksize = statx.stx_blksize as _;
stat.st_blocks = statx.stx_blocks as _;
stat.st_atime = statx.stx_atime.tv_sec;
stat.st_atime_nsec = statx.stx_atime.tv_nsec as _;
stat.st_mtime = statx.stx_mtime.tv_sec;
stat.st_mtime_nsec = statx.stx_mtime.tv_nsec as _;
stat.st_ctime = statx.stx_btime.tv_sec;
stat.st_ctime_nsec = statx.stx_btime.tv_nsec as _;
stat
}
pub struct ReadVectoredAt<T: IoVectoredBufMut, S> {
pub(crate) fd: SharedFd<S>,
pub(crate) offset: u64,
pub(crate) buffer: T,
pub(crate) slices: Vec<IoSliceMut>,
#[cfg(freebsd)]
pub(crate) aiocb: libc::aiocb,
_p: PhantomPinned,
}
impl<T: IoVectoredBufMut, S> ReadVectoredAt<T, S> {
pub fn new(fd: SharedFd<S>, offset: u64, buffer: T) -> Self {
Self {
fd,
offset,
buffer,
slices: vec![],
#[cfg(freebsd)]
aiocb: unsafe { std::mem::zeroed() },
_p: PhantomPinned,
}
}
}
impl<T: IoVectoredBufMut, S> IntoInner for ReadVectoredAt<T, S> {
type Inner = T;
fn into_inner(self) -> Self::Inner {
self.buffer
}
}
pub struct WriteVectoredAt<T: IoVectoredBuf, S> {
pub(crate) fd: SharedFd<S>,
pub(crate) offset: u64,
pub(crate) buffer: T,
pub(crate) slices: Vec<IoSlice>,
#[cfg(freebsd)]
pub(crate) aiocb: libc::aiocb,
_p: PhantomPinned,
}
impl<T: IoVectoredBuf, S> WriteVectoredAt<T, S> {
pub fn new(fd: SharedFd<S>, offset: u64, buffer: T) -> Self {
Self {
fd,
offset,
buffer,
slices: vec![],
#[cfg(freebsd)]
aiocb: unsafe { std::mem::zeroed() },
_p: PhantomPinned,
}
}
}
impl<T: IoVectoredBuf, S> IntoInner for WriteVectoredAt<T, S> {
type Inner = T;
fn into_inner(self) -> Self::Inner {
self.buffer
}
}
pub struct Unlink {
pub(crate) path: CString,
pub(crate) dir: bool,
}
impl Unlink {
pub fn new(path: CString, dir: bool) -> Self {
Self { path, dir }
}
}
pub struct CreateDir {
pub(crate) path: CString,
pub(crate) mode: libc::mode_t,
}
impl CreateDir {
pub fn new(path: CString, mode: libc::mode_t) -> Self {
Self { path, mode }
}
}
pub struct Rename {
pub(crate) old_path: CString,
pub(crate) new_path: CString,
}
impl Rename {
pub fn new(old_path: CString, new_path: CString) -> Self {
Self { old_path, new_path }
}
}
pub struct Symlink {
pub(crate) source: CString,
pub(crate) target: CString,
}
impl Symlink {
pub fn new(source: CString, target: CString) -> Self {
Self { source, target }
}
}
pub struct HardLink {
pub(crate) source: CString,
pub(crate) target: CString,
}
impl HardLink {
pub fn new(source: CString, target: CString) -> Self {
Self { source, target }
}
}
pub struct CreateSocket {
pub(crate) domain: i32,
pub(crate) socket_type: i32,
pub(crate) protocol: i32,
}
impl CreateSocket {
pub fn new(domain: i32, socket_type: i32, protocol: i32) -> Self {
Self {
domain,
socket_type,
protocol,
}
}
}
impl<S> ShutdownSocket<S> {
pub(crate) fn how(&self) -> i32 {
match self.how {
Shutdown::Write => libc::SHUT_WR,
Shutdown::Read => libc::SHUT_RD,
Shutdown::Both => libc::SHUT_RDWR,
}
}
}
pub struct Accept<S> {
pub(crate) fd: SharedFd<S>,
pub(crate) buffer: sockaddr_storage,
pub(crate) addr_len: socklen_t,
pub(crate) accepted_fd: Option<OwnedFd>,
_p: PhantomPinned,
}
impl<S> Accept<S> {
pub fn new(fd: SharedFd<S>) -> Self {
Self {
fd,
buffer: unsafe { std::mem::zeroed() },
addr_len: std::mem::size_of::<sockaddr_storage>() as _,
accepted_fd: None,
_p: PhantomPinned,
}
}
pub fn into_addr(mut self) -> SockAddr {
std::mem::forget(self.accepted_fd.take());
unsafe { SockAddr::new(self.buffer, self.addr_len) }
}
}
pub struct Recv<T: IoBufMut, S> {
pub(crate) fd: SharedFd<S>,
pub(crate) buffer: T,
_p: PhantomPinned,
}
impl<T: IoBufMut, S> Recv<T, S> {
pub fn new(fd: SharedFd<S>, buffer: T) -> Self {
Self {
fd,
buffer,
_p: PhantomPinned,
}
}
}
impl<T: IoBufMut, S> IntoInner for Recv<T, S> {
type Inner = T;
fn into_inner(self) -> Self::Inner {
self.buffer
}
}
pub struct RecvVectored<T: IoVectoredBufMut, S> {
pub(crate) fd: SharedFd<S>,
pub(crate) buffer: T,
pub(crate) slices: Vec<IoSliceMut>,
_p: PhantomPinned,
}
impl<T: IoVectoredBufMut, S> RecvVectored<T, S> {
pub fn new(fd: SharedFd<S>, buffer: T) -> Self {
Self {
fd,
buffer,
slices: vec![],
_p: PhantomPinned,
}
}
}
impl<T: IoVectoredBufMut, S> IntoInner for RecvVectored<T, S> {
type Inner = T;
fn into_inner(self) -> Self::Inner {
self.buffer
}
}
pub struct Send<T: IoBuf, S> {
pub(crate) fd: SharedFd<S>,
pub(crate) buffer: T,
_p: PhantomPinned,
}
impl<T: IoBuf, S> Send<T, S> {
pub fn new(fd: SharedFd<S>, buffer: T) -> Self {
Self {
fd,
buffer,
_p: PhantomPinned,
}
}
}
impl<T: IoBuf, S> IntoInner for Send<T, S> {
type Inner = T;
fn into_inner(self) -> Self::Inner {
self.buffer
}
}
pub struct SendVectored<T: IoVectoredBuf, S> {
pub(crate) fd: SharedFd<S>,
pub(crate) buffer: T,
pub(crate) slices: Vec<IoSlice>,
_p: PhantomPinned,
}
impl<T: IoVectoredBuf, S> SendVectored<T, S> {
pub fn new(fd: SharedFd<S>, buffer: T) -> Self {
Self {
fd,
buffer,
slices: vec![],
_p: PhantomPinned,
}
}
}
impl<T: IoVectoredBuf, S> IntoInner for SendVectored<T, S> {
type Inner = T;
fn into_inner(self) -> Self::Inner {
self.buffer
}
}
pub struct RecvMsg<T: IoVectoredBufMut, C: IoBufMut, S> {
pub(crate) msg: libc::msghdr,
pub(crate) addr: sockaddr_storage,
pub(crate) fd: SharedFd<S>,
pub(crate) buffer: T,
pub(crate) control: C,
pub(crate) slices: Vec<IoSliceMut>,
_p: PhantomPinned,
}
impl<T: IoVectoredBufMut, C: IoBufMut, S> RecvMsg<T, C, S> {
pub fn new(fd: SharedFd<S>, buffer: T, control: C) -> Self {
assert!(
control.as_buf_ptr().cast::<libc::cmsghdr>().is_aligned(),
"misaligned control message buffer"
);
Self {
addr: unsafe { std::mem::zeroed() },
msg: unsafe { std::mem::zeroed() },
fd,
buffer,
control,
slices: vec![],
_p: PhantomPinned,
}
}
pub(crate) unsafe fn set_msg(&mut self) {
self.slices = self.buffer.io_slices_mut();
self.msg.msg_name = std::ptr::addr_of_mut!(self.addr) as _;
self.msg.msg_namelen = std::mem::size_of_val(&self.addr) as _;
self.msg.msg_iov = self.slices.as_mut_ptr() as _;
self.msg.msg_iovlen = self.slices.len() as _;
self.msg.msg_control = self.control.as_buf_mut_ptr() as _;
self.msg.msg_controllen = self.control.buf_capacity() as _;
}
}
impl<T: IoVectoredBufMut, C: IoBufMut, S> IntoInner for RecvMsg<T, C, S> {
type Inner = ((T, C), sockaddr_storage, socklen_t, usize);
fn into_inner(self) -> Self::Inner {
(
(self.buffer, self.control),
self.addr,
self.msg.msg_namelen,
self.msg.msg_controllen as _,
)
}
}
pub struct SendMsg<T: IoVectoredBuf, C: IoBuf, S> {
pub(crate) msg: libc::msghdr,
pub(crate) fd: SharedFd<S>,
pub(crate) buffer: T,
pub(crate) control: C,
pub(crate) addr: SockAddr,
pub(crate) slices: Vec<IoSlice>,
_p: PhantomPinned,
}
impl<T: IoVectoredBuf, C: IoBuf, S> SendMsg<T, C, S> {
pub fn new(fd: SharedFd<S>, buffer: T, control: C, addr: SockAddr) -> Self {
assert!(
control.as_buf_ptr().cast::<libc::cmsghdr>().is_aligned(),
"misaligned control message buffer"
);
Self {
msg: unsafe { std::mem::zeroed() },
fd,
buffer,
control,
addr,
slices: vec![],
_p: PhantomPinned,
}
}
pub(crate) unsafe fn set_msg(&mut self) {
self.slices = self.buffer.io_slices();
self.msg.msg_name = self.addr.as_ptr() as _;
self.msg.msg_namelen = self.addr.len();
self.msg.msg_iov = self.slices.as_ptr() as _;
self.msg.msg_iovlen = self.slices.len() as _;
self.msg.msg_control = self.control.as_buf_ptr() as _;
self.msg.msg_controllen = self.control.buf_len() as _;
}
}
impl<T: IoVectoredBuf, C: IoBuf, S> IntoInner for SendMsg<T, C, S> {
type Inner = (T, C);
fn into_inner(self) -> Self::Inner {
(self.buffer, self.control)
}
}
#[derive(Debug, Clone, Copy, PartialEq, Eq)]
pub enum Interest {
Readable,
Writable,
}
pub struct PollOnce<S> {
pub(crate) fd: SharedFd<S>,
pub(crate) interest: Interest,
}
impl<S> PollOnce<S> {
pub fn new(fd: SharedFd<S>, interest: Interest) -> Self {
Self { fd, interest }
}
}