
* Fire up multiple processors * Use IPIs to wake up secondary processors * Much better exception information * Modifications to show more information on fault * WIP: Use real libstd * Add TLS (not complete) * Add random function, export getpid, cleanup * Do not spin APs until new context * Update rust * Update rust * Use rd/wrfsbase * Implement TLS * Implement compiler builtins and update rust * Update rust * Back to Redox libstd * Update rust
84 lines
2.7 KiB
Rust
84 lines
2.7 KiB
Rust
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<R> {
|
|
/// The file to read events from
|
|
file: File,
|
|
/// A map of registered file descriptors to their handler callbacks
|
|
callbacks: BTreeMap<RawFd, Box<FnMut(usize) -> Result<Option<R>>>>
|
|
}
|
|
|
|
impl<R> EventQueue<R> {
|
|
/// Create a new event queue
|
|
pub fn new() -> Result<EventQueue<R>> {
|
|
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<F: FnMut(usize) -> Result<Option<R>> + '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<Option<Box<FnMut(usize) -> Result<Option<R>>>>> {
|
|
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<Option<R>> {
|
|
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<Vec<R>> {
|
|
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<R> {
|
|
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);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|