use std::io;
use std::os::unix::net::{self, SocketAddr};
use std::sync::atomic::Ordering;
#[cfg(feature = "io_cancel")]
use crate::coroutine_impl::co_cancel_data;
use crate::coroutine_impl::{is_coroutine, CoroutineImpl, EventSource};
use crate::io::sys::{co_io_result, IoData};
use crate::io::{AsIoData, CoIo};
use crate::os::unix::net::{UnixListener, UnixStream};
use crate::yield_now::yield_with_io;
pub struct UnixListenerAccept<'a> {
io_data: &'a IoData,
socket: &'a net::UnixListener,
pub(crate) is_coroutine: bool,
}
impl<'a> UnixListenerAccept<'a> {
pub fn new(socket: &'a UnixListener) -> io::Result<Self> {
Ok(UnixListenerAccept {
io_data: socket.0.as_io_data(),
socket: socket.0.inner(),
is_coroutine: is_coroutine(),
})
}
pub fn done(&mut self) -> io::Result<(UnixStream, SocketAddr)> {
loop {
co_io_result(self.is_coroutine)?;
self.io_data.io_flag.store(false, Ordering::Relaxed);
match self.socket.accept() {
Ok((s, a)) => {
let s = UnixStream::from_coio(CoIo::new(s)?);
return Ok((s, a));
}
Err(e) => {
let raw_err = e.raw_os_error();
if raw_err == Some(libc::EAGAIN) || raw_err == Some(libc::EWOULDBLOCK) {
} else {
return Err(e);
}
}
}
if self.io_data.io_flag.load(Ordering::Relaxed) {
continue;
}
yield_with_io(self, self.is_coroutine);
}
}
}
impl<'a> EventSource for UnixListenerAccept<'a> {
fn subscribe(&mut self, co: CoroutineImpl) {
#[cfg(feature = "io_cancel")]
let cancel = co_cancel_data(&co);
let io_data = self.io_data;
io_data.co.store(co);
if io_data.io_flag.load(Ordering::Acquire) {
#[allow(clippy::needless_return)]
return io_data.fast_schedule();
}
#[cfg(feature = "io_cancel")]
{
cancel.set_io((*io_data).clone());
if cancel.is_canceled() {
unsafe { cancel.cancel() };
}
}
}
}