use std::cell::Cell;
use std::mem::MaybeUninit;
use std::sync::{Mutex, MutexGuard};
static GLOBAL_MUTEX: Mutex<()> = Mutex::new(());
static mut GLOBAL_GUARD: MaybeUninit<MutexGuard<'static, ()>> = MaybeUninit::uninit();
std::thread_local!(static IS_LOCKED: Cell<bool> = Cell::new(false));
struct StdCriticalSection;
crate::set_impl!(StdCriticalSection);
unsafe impl crate::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.write(guard);
false
})
}
unsafe fn release(nested_cs: bool) {
if !nested_cs {
#[allow(let_underscore_lock)]
let _ = GLOBAL_GUARD.assume_init_read();
IS_LOCKED.with(|l| l.set(false));
}
}
}
#[cfg(test)]
mod tests {
use std::thread;
use crate as critical_section;
#[cfg(feature = "std")]
#[test]
#[should_panic(expected = "Not a PoisonError!")]
fn reusable_after_panic() {
let _ = thread::spawn(|| {
critical_section::with(|_| {
panic!("Boom!");
})
})
.join();
critical_section::with(|_| {
panic!("Not a PoisonError!");
})
}
}