use std::sync::{Arc, Mutex, Weak};
use zircon;
use {io, poll, Evented, Poll, PollOpt, Ready, Token};
pub struct Awakener {
inner: Mutex<Option<(Token, Weak<zircon::Port>)>>,
}
impl Awakener {
pub fn new() -> io::Result<Awakener> {
Ok(Awakener {
inner: Mutex::new(None),
})
}
pub fn wakeup(&self) -> io::Result<()> {
let inner_locked = self.inner.lock().unwrap();
let &(token, ref weak_port) = inner_locked
.as_ref()
.expect("Called wakeup on unregistered awakener.");
let port = weak_port.upgrade().expect("Tried to wakeup a closed port.");
let status = 0; let packet = zircon::Packet::from_user_packet(
token.0 as u64,
status,
zircon::UserPacket::from_u8_array([0; 32]),
);
Ok(port.queue(&packet)?)
}
pub fn cleanup(&self) {}
}
impl Evented for Awakener {
fn register(
&self,
poll: &Poll,
token: Token,
_events: Ready,
_opts: PollOpt,
) -> io::Result<()> {
let mut inner_locked = self.inner.lock().unwrap();
if inner_locked.is_some() {
panic!("Called register on already-registered Awakener.");
}
*inner_locked = Some((token, Arc::downgrade(poll::selector(poll).port())));
Ok(())
}
fn reregister(
&self,
poll: &Poll,
token: Token,
_events: Ready,
_opts: PollOpt,
) -> io::Result<()> {
let mut inner_locked = self.inner.lock().unwrap();
*inner_locked = Some((token, Arc::downgrade(poll::selector(poll).port())));
Ok(())
}
fn deregister(&self, _poll: &Poll) -> io::Result<()> {
let mut inner_locked = self.inner.lock().unwrap();
*inner_locked = None;
Ok(())
}
}