2017-01-18 20:39:46 +00:00
|
|
|
macro_rules! component {
|
|
|
|
($name:ident : $inner:ident { $($key:ident : $ty:ty = $value:expr,)* }) => {
|
|
|
|
#[derive(Clone)]
|
|
|
|
pub struct $name(::std::sync::Arc<($crate::session::SessionWeak, ::std::sync::Mutex<$inner>)>);
|
|
|
|
impl $name {
|
|
|
|
#[allow(dead_code)]
|
|
|
|
pub fn new(session: $crate::session::SessionWeak) -> $name {
|
|
|
|
$name(::std::sync::Arc::new((session, ::std::sync::Mutex::new($inner {
|
|
|
|
$($key : $value,)*
|
|
|
|
}))))
|
|
|
|
}
|
|
|
|
|
|
|
|
#[allow(dead_code)]
|
|
|
|
fn lock<F: FnOnce(&mut $inner) -> R, R>(&self, f: F) -> R {
|
|
|
|
let mut inner = (self.0).1.lock().expect("Mutex poisoned");
|
|
|
|
f(&mut inner)
|
|
|
|
}
|
|
|
|
|
|
|
|
#[allow(dead_code)]
|
|
|
|
fn session(&self) -> $crate::session::Session {
|
|
|
|
(self.0).0.upgrade()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
struct $inner {
|
|
|
|
$($key : $ty,)*
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2017-01-19 22:45:24 +00:00
|
|
|
|
|
|
|
use std::sync::Mutex;
|
|
|
|
use std::cell::UnsafeCell;
|
|
|
|
|
|
|
|
pub struct Lazy<T>(Mutex<bool>, UnsafeCell<Option<T>>);
|
|
|
|
unsafe impl <T: Sync> Sync for Lazy<T> {}
|
|
|
|
unsafe impl <T: Send> Send for Lazy<T> {}
|
|
|
|
|
2017-01-29 16:25:09 +00:00
|
|
|
#[cfg_attr(feature = "cargo-clippy", allow(mutex_atomic))]
|
2017-01-19 22:45:24 +00:00
|
|
|
impl <T> Lazy<T> {
|
|
|
|
pub fn new() -> Lazy<T> {
|
|
|
|
Lazy(Mutex::new(false), UnsafeCell::new(None))
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn get<F: FnOnce() -> T>(&self, f: F) -> &T {
|
|
|
|
let mut inner = self.0.lock().unwrap();
|
|
|
|
if !*inner {
|
|
|
|
unsafe {
|
|
|
|
*self.1.get() = Some(f());
|
|
|
|
}
|
|
|
|
*inner = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
unsafe { &*self.1.get() }.as_ref().unwrap()
|
|
|
|
}
|
|
|
|
}
|