use std::{
io,
os::fd::{AsRawFd, FromRawFd, OwnedFd, RawFd},
};
use compio_buf::{arrayvec::ArrayVec, BufResult};
use compio_driver::{op::Recv, syscall};
use crate::{attacher::Attacher, submit};
#[derive(Debug)]
pub struct Event {
sender: OwnedFd,
receiver: OwnedFd,
attacher: Attacher,
}
impl Event {
pub fn new() -> io::Result<Self> {
let mut fds = [-1, -1];
syscall!(pipe(fds.as_mut_ptr()))?;
let receiver = unsafe { OwnedFd::from_raw_fd(fds[0]) };
let sender = unsafe { OwnedFd::from_raw_fd(fds[1]) };
syscall!(fcntl(receiver.as_raw_fd(), libc::F_SETFD, libc::FD_CLOEXEC))?;
syscall!(fcntl(receiver.as_raw_fd(), libc::F_SETFL, libc::O_NONBLOCK))?;
syscall!(fcntl(sender.as_raw_fd(), libc::F_SETFD, libc::FD_CLOEXEC))?;
Ok(Self {
sender,
receiver,
attacher: Attacher::new(),
})
}
pub fn handle(&self) -> io::Result<EventHandle> {
Ok(EventHandle::new(self.sender.try_clone()?))
}
pub async fn wait(&self) -> io::Result<()> {
self.attacher.attach(&self.receiver)?;
let buffer = ArrayVec::<u8, 1>::new();
let op = Recv::new(self.receiver.as_raw_fd(), buffer);
let BufResult(res, _) = submit(op).await;
res?;
Ok(())
}
}
impl AsRawFd for Event {
fn as_raw_fd(&self) -> RawFd {
self.receiver.as_raw_fd()
}
}
pub struct EventHandle {
fd: OwnedFd,
}
impl EventHandle {
pub(crate) fn new(fd: OwnedFd) -> Self {
Self { fd }
}
pub fn notify(&self) -> io::Result<()> {
let data = &[1];
syscall!(write(self.fd.as_raw_fd(), data.as_ptr() as _, 1))?;
Ok(())
}
}