extern crate syscall; use std::collections::BTreeMap; use std::fs::File; use std::io::{Read, Error, Result}; use std::os::unix::io::RawFd; pub struct EventQueue { /// The file to read events from file: File, /// A map of registered file descriptors to their handler callbacks callbacks: BTreeMap Result>>> } impl EventQueue { /// Create a new event queue pub fn new() -> Result> { Ok(EventQueue { file: File::open("event:")?, callbacks: BTreeMap::new() }) } /// Add a file to the event queue, calling a callback when an event occurs /// /// The callback is given a mutable reference to the file and the event data /// (typically the length of data available for read) /// /// The callback returns Ok(None) if it wishes to continue the event loop, /// or Ok(Some(R)) to break the event loop and return the value. /// Err can be used to allow the callback to return an I/O error, and break the /// event loop pub fn add Result> + 'static>(&mut self, fd: RawFd, callback: F) -> Result<()> { syscall::fevent(fd, syscall::EVENT_READ).map_err(|x| Error::from_raw_os_error(x.errno))?; self.callbacks.insert(fd, Box::new(callback)); Ok(()) } /// Remove a file from the event queue, returning its callback if found pub fn remove(&mut self, fd: RawFd) -> Result Result>>>> { if let Some(callback) = self.callbacks.remove(&fd) { syscall::fevent(fd, 0).map_err(|x| Error::from_raw_os_error(x.errno))?; Ok(Some(callback)) } else { Ok(None) } } /// Send an event to a descriptor callback pub fn trigger(&mut self, fd: RawFd, count: usize) -> Result> { if let Some(callback) = self.callbacks.get_mut(&fd) { callback(count) } else { Ok(None) } } /// Send an event to all descriptor callbacks, useful for cleaning out buffers after init pub fn trigger_all(&mut self, count: usize) -> Result> { let mut rets = Vec::new(); for (_fd, callback) in self.callbacks.iter_mut() { if let Some(ret) = callback(count)? { rets.push(ret); } } Ok(rets) } /// Process the event queue until a callback returns Some(R) pub fn run(&mut self) -> Result { loop { let mut event = syscall::Event::default(); if self.file.read(&mut event)? > 0 { if let Some(ret) = self.trigger(event.id, event.data)? { return Ok(ret); } } } } }