use nix::fcntl::Fd;
use nix::sys::epoll::*;
use nix::unistd::close;
use io;
use event::{IoEvent, Interest, PollOpt};
pub struct Selector {
epfd: Fd
}
impl Selector {
pub fn new() -> io::Result<Selector> {
let epfd = try!(epoll_create().map_err(io::from_nix_error));
Ok(Selector { epfd: epfd })
}
pub fn select(&mut self, evts: &mut Events, timeout_ms: usize) -> io::Result<()> {
use std::slice;
let dst = unsafe {
slice::from_raw_parts_mut(
evts.events.as_mut_ptr(),
evts.events.capacity())
};
let cnt = try!(epoll_wait(self.epfd, dst, timeout_ms)
.map_err(io::from_nix_error));
unsafe { evts.events.set_len(cnt); }
Ok(())
}
pub fn register(&mut self, fd: Fd, token: usize, interests: Interest, opts: PollOpt) -> io::Result<()> {
let info = EpollEvent {
events: ioevent_to_epoll(interests, opts),
data: token as u64
};
epoll_ctl(self.epfd, EpollOp::EpollCtlAdd, fd, &info)
.map_err(io::from_nix_error)
}
pub fn reregister(&mut self, fd: Fd, token: usize, interests: Interest, opts: PollOpt) -> io::Result<()> {
let info = EpollEvent {
events: ioevent_to_epoll(interests, opts),
data: token as u64
};
epoll_ctl(self.epfd, EpollOp::EpollCtlMod, fd, &info)
.map_err(io::from_nix_error)
}
pub fn deregister(&mut self, fd: Fd) -> io::Result<()> {
let info = EpollEvent {
events: EpollEventKind::empty(),
data: 0
};
epoll_ctl(self.epfd, EpollOp::EpollCtlDel, fd, &info)
.map_err(io::from_nix_error)
}
}
fn ioevent_to_epoll(interest: Interest, opts: PollOpt) -> EpollEventKind {
let mut kind = EpollEventKind::empty();
if interest.is_readable() {
kind.insert(EPOLLIN);
}
if interest.is_writable() {
kind.insert(EPOLLOUT);
}
if interest.is_hup() {
kind.insert(EPOLLRDHUP);
}
if opts.is_edge() {
kind.insert(EPOLLET);
}
if opts.is_oneshot() {
kind.insert(EPOLLONESHOT);
}
if opts.is_level() {
kind.remove(EPOLLET);
}
kind
}
impl Drop for Selector {
fn drop(&mut self) {
let _ = close(self.epfd);
}
}
pub struct Events {
events: Vec<EpollEvent>,
}
impl Events {
pub fn new() -> Events {
Events { events: Vec::with_capacity(1024) }
}
#[inline]
pub fn len(&self) -> usize {
self.events.len()
}
#[inline]
pub fn get(&self, idx: usize) -> IoEvent {
if idx >= self.len() {
panic!("invalid index");
}
let epoll = self.events[idx].events;
let mut kind = Interest::hinted();
if epoll.contains(EPOLLIN) {
kind = kind | Interest::readable();
}
if epoll.contains(EPOLLOUT) {
kind = kind | Interest::writable();
}
if epoll.contains(EPOLLERR) {
kind = kind | Interest::error();
}
if epoll.contains(EPOLLRDHUP) | epoll.contains(EPOLLHUP) {
kind = kind | Interest::hup();
}
let token = self.events[idx].data;
IoEvent::new(kind, token as usize)
}
}