Files
server/src/core/eventloop.rs

104 lines
2.8 KiB
Rust

use super::client::Client;
use anyhow::Result;
use libstardustxr::server;
use mio::net::UnixListener;
use mio::unix::pipe;
use mio::{Events, Interest, Poll, Token};
use slab::Slab;
use std::io::Write;
use std::sync::atomic::AtomicU64;
use std::sync::Arc;
use std::thread::{self, JoinHandle};
pub static FRAME: AtomicU64 = AtomicU64::new(0);
pub struct EventLoop {
pub socket_path: String,
join_handle: Option<JoinHandle<Result<()>>>,
stop_write: pipe::Sender,
}
impl EventLoop {
pub fn new(timeout: Option<core::time::Duration>) -> Result<Self> {
let socket_path = server::get_free_socket_path()
.ok_or_else(|| std::io::Error::from(std::io::ErrorKind::Other))?;
let mut socket = UnixListener::bind(socket_path.clone())?;
let (sender, mut receiver) = pipe::new()?;
let join_handle = thread::Builder::new()
.name("event_loop".to_owned())
.spawn(move || -> Result<()> {
let mut clients: Slab<Option<Arc<Client>>> = Slab::new();
let mut poll = Poll::new()?;
let mut events = Events::with_capacity(1024);
const LISTENER: Token = Token(usize::MAX - 1);
poll.registry()
.register(&mut socket, LISTENER, Interest::READABLE)?;
const STOP: Token = Token(usize::MAX);
poll.registry()
.register(&mut receiver, STOP, Interest::READABLE)?;
loop {
poll.poll(&mut events, timeout)?;
for event in &events {
match event.token() {
LISTENER => loop {
match socket.accept() {
Ok((mut socket, _)) => {
let client_number = clients.insert(None);
poll.registry().register(
&mut socket,
Token(client_number),
Interest::READABLE,
)?;
let client = Client::from_connection(socket);
*clients.get_mut(client_number).unwrap() = Some(client);
}
Err(e) => {
if e.kind() == std::io::ErrorKind::WouldBlock {
break;
}
return Err(e.into());
}
}
},
STOP => return Ok(()),
token => {
let client =
clients.get(token.0).and_then(|client| client.as_ref());
if let Some(client) = client {
loop {
match client.dispatch() {
Ok(_) => continue,
Err(e) => match e.kind() {
std::io::ErrorKind::WouldBlock => break,
_ => {
clients.remove(token.0);
break;
}
},
}
}
}
}
}
}
}
})
.ok();
Ok(EventLoop {
socket_path,
join_handle,
stop_write: sender,
})
}
}
impl Drop for EventLoop {
fn drop(&mut self) {
let buf: [u8; 1] = [1; 1];
let _ = self.stop_write.write(buf.as_slice());
if let Some(handle) = self.join_handle.take() {
handle.join().unwrap().unwrap();
}
}
}