use std::ops::{Deref, DerefMut};
use cfg_if::cfg_if;
#[cfg(not(reifydb_single_threaded))]
pub(crate) mod native;
#[cfg(reifydb_single_threaded)]
pub(crate) mod wasm;
cfg_if! {
if #[cfg(not(reifydb_single_threaded))] {
type RwLockInnerImpl<T> = native::RwLockInner<T>;
type RwLockReadGuardInnerImpl<'a, T> = native::RwLockReadGuardInner<'a, T>;
type RwLockWriteGuardInnerImpl<'a, T> = native::RwLockWriteGuardInner<'a, T>;
} else {
type RwLockInnerImpl<T> = wasm::RwLockInner<T>;
type RwLockReadGuardInnerImpl<'a, T> = wasm::RwLockReadGuardInner<'a, T>;
type RwLockWriteGuardInnerImpl<'a, T> = wasm::RwLockWriteGuardInner<'a, T>;
}
}
pub struct RwLock<T> {
inner: RwLockInnerImpl<T>,
}
#[cfg(reifydb_single_threaded)]
unsafe impl<T> Sync for RwLock<T> {}
impl<T> RwLock<T> {
#[inline]
pub fn new(value: T) -> Self {
Self {
inner: RwLockInnerImpl::new(value),
}
}
#[inline]
pub fn read(&self) -> RwLockReadGuard<'_, T> {
RwLockReadGuard {
inner: self.inner.read(),
}
}
#[inline]
pub fn write(&self) -> RwLockWriteGuard<'_, T> {
RwLockWriteGuard {
inner: self.inner.write(),
}
}
#[inline]
pub fn try_read(&self) -> Option<RwLockReadGuard<'_, T>> {
self.inner.try_read().map(|inner| RwLockReadGuard {
inner,
})
}
#[inline]
pub fn try_write(&self) -> Option<RwLockWriteGuard<'_, T>> {
self.inner.try_write().map(|inner| RwLockWriteGuard {
inner,
})
}
}
pub struct RwLockReadGuard<'a, T> {
inner: RwLockReadGuardInnerImpl<'a, T>,
}
impl<'a, T> Deref for RwLockReadGuard<'a, T> {
type Target = T;
#[inline]
fn deref(&self) -> &T {
&self.inner
}
}
pub struct RwLockWriteGuard<'a, T> {
inner: RwLockWriteGuardInnerImpl<'a, T>,
}
impl<'a, T> Deref for RwLockWriteGuard<'a, T> {
type Target = T;
#[inline]
fn deref(&self) -> &T {
&self.inner
}
}
impl<'a, T> DerefMut for RwLockWriteGuard<'a, T> {
#[inline]
fn deref_mut(&mut self) -> &mut T {
&mut self.inner
}
}