use core::cell::RefCell;
use core::mem::MaybeUninit;
use loom::cell::Cell;
use loom::sync::{Mutex, MutexGuard};
loom::lazy_static! {
static ref GLOBAL_MUTEX: Mutex<()> = Mutex::new(());
static ref GLOBAL_GUARD: RefCell<MaybeUninit<MutexGuard<'static, ()>>> = RefCell::new(MaybeUninit::uninit());
}
loom::thread_local!(static IS_LOCKED: Cell<bool> = Cell::new(false));
struct StdCriticalSection;
critical_section::set_impl!(StdCriticalSection);
unsafe impl critical_section::Impl for StdCriticalSection {
unsafe fn acquire() -> bool {
IS_LOCKED.with(|l| {
if l.get() {
return true;
}
l.set(true);
let guard = match GLOBAL_MUTEX.lock() {
Ok(guard) => guard,
Err(err) => {
err.into_inner()
}
};
GLOBAL_GUARD.borrow_mut().write(guard);
false
})
}
unsafe fn release(nested_cs: bool) {
if !nested_cs {
#[allow(let_underscore_lock)]
let _ = GLOBAL_GUARD.borrow_mut().assume_init_read();
IS_LOCKED.with(|l| l.set(false));
}
}
}