mirror of
https://github.com/asterinas/asterinas.git
synced 2025-06-29 16:13:27 +00:00
Support T:?Sized as type parameter for Mutex, SpinLock, RwLock, and RwMutex
This commit is contained in:
committed by
Tate, Hongliang Tian
parent
92a9f5616e
commit
dac41e9a2f
@ -1,6 +1,6 @@
|
||||
// SPDX-License-Identifier: MPL-2.0
|
||||
|
||||
use alloc::{boxed::Box, sync::Arc};
|
||||
use alloc::sync::Arc;
|
||||
|
||||
use log::info;
|
||||
use spin::Once;
|
||||
@ -11,7 +11,7 @@ pub mod ioapic;
|
||||
pub mod x2apic;
|
||||
pub mod xapic;
|
||||
|
||||
pub static APIC_INSTANCE: Once<Arc<Mutex<Box<dyn Apic + 'static>>>> = Once::new();
|
||||
pub static APIC_INSTANCE: Once<Arc<Mutex<dyn Apic + 'static>>> = Once::new();
|
||||
|
||||
pub trait Apic: ApicTimer + Sync + Send {
|
||||
fn id(&self) -> u32;
|
||||
@ -71,7 +71,7 @@ pub fn init() -> Result<(), ApicInitError> {
|
||||
version & 0xff,
|
||||
(version >> 16) & 0xff
|
||||
);
|
||||
APIC_INSTANCE.call_once(|| Arc::new(Mutex::new(Box::new(x2apic))));
|
||||
APIC_INSTANCE.call_once(|| Arc::new(Mutex::new(x2apic)));
|
||||
Ok(())
|
||||
} else if let Some(mut xapic) = xapic::XApic::new() {
|
||||
xapic.enable();
|
||||
@ -82,7 +82,7 @@ pub fn init() -> Result<(), ApicInitError> {
|
||||
version & 0xff,
|
||||
(version >> 16) & 0xff
|
||||
);
|
||||
APIC_INSTANCE.call_once(|| Arc::new(Mutex::new(Box::new(xapic))));
|
||||
APIC_INSTANCE.call_once(|| Arc::new(Mutex::new(xapic)));
|
||||
Ok(())
|
||||
} else {
|
||||
log::warn!("Not found x2APIC or xAPIC");
|
||||
|
@ -10,22 +10,24 @@ use core::{
|
||||
use super::WaitQueue;
|
||||
|
||||
/// A mutex with waitqueue.
|
||||
pub struct Mutex<T> {
|
||||
val: UnsafeCell<T>,
|
||||
pub struct Mutex<T: ?Sized> {
|
||||
lock: AtomicBool,
|
||||
queue: WaitQueue,
|
||||
val: UnsafeCell<T>,
|
||||
}
|
||||
|
||||
impl<T> Mutex<T> {
|
||||
/// Create a new mutex.
|
||||
pub const fn new(val: T) -> Self {
|
||||
Self {
|
||||
val: UnsafeCell::new(val),
|
||||
lock: AtomicBool::new(false),
|
||||
queue: WaitQueue::new(),
|
||||
val: UnsafeCell::new(val),
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
impl<T: ?Sized> Mutex<T> {
|
||||
/// Acquire the mutex.
|
||||
///
|
||||
/// This method runs in a block way until the mutex can be acquired.
|
||||
@ -57,27 +59,27 @@ impl<T> Mutex<T> {
|
||||
}
|
||||
}
|
||||
|
||||
impl<T: fmt::Debug> fmt::Debug for Mutex<T> {
|
||||
impl<T: ?Sized + fmt::Debug> fmt::Debug for Mutex<T> {
|
||||
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
||||
fmt::Debug::fmt(&self.val, f)
|
||||
}
|
||||
}
|
||||
|
||||
unsafe impl<T: Send> Send for Mutex<T> {}
|
||||
unsafe impl<T: Send> Sync for Mutex<T> {}
|
||||
unsafe impl<T: ?Sized + Send> Send for Mutex<T> {}
|
||||
unsafe impl<T: ?Sized + Send> Sync for Mutex<T> {}
|
||||
|
||||
#[clippy::has_significant_drop]
|
||||
pub struct MutexGuard<'a, T> {
|
||||
pub struct MutexGuard<'a, T: ?Sized> {
|
||||
mutex: &'a Mutex<T>,
|
||||
}
|
||||
|
||||
impl<'a, T> MutexGuard<'a, T> {
|
||||
impl<'a, T: ?Sized> MutexGuard<'a, T> {
|
||||
fn new(mutex: &'a Mutex<T>) -> MutexGuard<'a, T> {
|
||||
MutexGuard { mutex }
|
||||
}
|
||||
}
|
||||
|
||||
impl<'a, T> Deref for MutexGuard<'a, T> {
|
||||
impl<'a, T: ?Sized> Deref for MutexGuard<'a, T> {
|
||||
type Target = T;
|
||||
|
||||
fn deref(&self) -> &Self::Target {
|
||||
@ -85,24 +87,24 @@ impl<'a, T> Deref for MutexGuard<'a, T> {
|
||||
}
|
||||
}
|
||||
|
||||
impl<'a, T> DerefMut for MutexGuard<'a, T> {
|
||||
impl<'a, T: ?Sized> DerefMut for MutexGuard<'a, T> {
|
||||
fn deref_mut(&mut self) -> &mut Self::Target {
|
||||
unsafe { &mut *self.mutex.val.get() }
|
||||
}
|
||||
}
|
||||
|
||||
impl<'a, T> Drop for MutexGuard<'a, T> {
|
||||
impl<'a, T: ?Sized> Drop for MutexGuard<'a, T> {
|
||||
fn drop(&mut self) {
|
||||
self.mutex.unlock();
|
||||
}
|
||||
}
|
||||
|
||||
impl<'a, T: fmt::Debug> fmt::Debug for MutexGuard<'a, T> {
|
||||
impl<'a, T: ?Sized + fmt::Debug> fmt::Debug for MutexGuard<'a, T> {
|
||||
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
||||
fmt::Debug::fmt(&**self, f)
|
||||
}
|
||||
}
|
||||
|
||||
impl<'a, T> !Send for MutexGuard<'a, T> {}
|
||||
impl<'a, T: ?Sized> !Send for MutexGuard<'a, T> {}
|
||||
|
||||
unsafe impl<T: Sync> Sync for MutexGuard<'_, T> {}
|
||||
unsafe impl<T: ?Sized + Sync> Sync for MutexGuard<'_, T> {}
|
||||
|
@ -96,14 +96,14 @@ use crate::{
|
||||
/// assert_eq!(*w2, 7);
|
||||
/// } // write lock is dropped at this point
|
||||
/// ```
|
||||
pub struct RwLock<T> {
|
||||
val: UnsafeCell<T>,
|
||||
pub struct RwLock<T: ?Sized> {
|
||||
/// The internal representation of the lock state is as follows:
|
||||
/// - **Bit 63:** Writer lock.
|
||||
/// - **Bit 62:** Upgradeable reader lock.
|
||||
/// - **Bit 61:** Indicates if an upgradeable reader is being upgraded.
|
||||
/// - **Bits 60-0:** Reader lock count.
|
||||
lock: AtomicUsize,
|
||||
val: UnsafeCell<T>,
|
||||
}
|
||||
|
||||
const READER: usize = 1;
|
||||
@ -120,7 +120,9 @@ impl<T> RwLock<T> {
|
||||
lock: AtomicUsize::new(0),
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
impl<T: ?Sized> RwLock<T> {
|
||||
/// Acquire a read lock while disabling the local IRQs and spin-wait
|
||||
/// until it can be acquired.
|
||||
///
|
||||
@ -386,7 +388,7 @@ impl<T> RwLock<T> {
|
||||
}
|
||||
}
|
||||
|
||||
impl<T: fmt::Debug> fmt::Debug for RwLock<T> {
|
||||
impl<T: ?Sized + fmt::Debug> fmt::Debug for RwLock<T> {
|
||||
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
||||
fmt::Debug::fmt(&self.val, f)
|
||||
}
|
||||
@ -394,17 +396,17 @@ impl<T: fmt::Debug> fmt::Debug for RwLock<T> {
|
||||
|
||||
/// Because there can be more than one readers to get the T's immutable ref,
|
||||
/// so T must be Sync to guarantee the sharing safety.
|
||||
unsafe impl<T: Send> Send for RwLock<T> {}
|
||||
unsafe impl<T: Send + Sync> Sync for RwLock<T> {}
|
||||
unsafe impl<T: ?Sized + Send> Send for RwLock<T> {}
|
||||
unsafe impl<T: ?Sized + Send + Sync> Sync for RwLock<T> {}
|
||||
|
||||
impl<'a, T> !Send for RwLockWriteGuard<'a, T> {}
|
||||
unsafe impl<T: Sync> Sync for RwLockWriteGuard<'_, T> {}
|
||||
impl<'a, T: ?Sized> !Send for RwLockWriteGuard<'a, T> {}
|
||||
unsafe impl<T: ?Sized + Sync> Sync for RwLockWriteGuard<'_, T> {}
|
||||
|
||||
impl<'a, T> !Send for RwLockReadGuard<'a, T> {}
|
||||
unsafe impl<T: Sync> Sync for RwLockReadGuard<'_, T> {}
|
||||
impl<'a, T: ?Sized> !Send for RwLockReadGuard<'a, T> {}
|
||||
unsafe impl<T: ?Sized + Sync> Sync for RwLockReadGuard<'_, T> {}
|
||||
|
||||
impl<'a, T> !Send for RwLockUpgradeableGuard<'a, T> {}
|
||||
unsafe impl<T: Sync> Sync for RwLockUpgradeableGuard<'_, T> {}
|
||||
impl<'a, T: ?Sized> !Send for RwLockUpgradeableGuard<'a, T> {}
|
||||
unsafe impl<T: ?Sized + Sync> Sync for RwLockUpgradeableGuard<'_, T> {}
|
||||
|
||||
enum InnerGuard {
|
||||
IrqGuard(DisabledLocalIrqGuard),
|
||||
@ -412,12 +414,12 @@ enum InnerGuard {
|
||||
}
|
||||
|
||||
/// A guard that provides immutable data access.
|
||||
pub struct RwLockReadGuard<'a, T> {
|
||||
inner: &'a RwLock<T>,
|
||||
pub struct RwLockReadGuard<'a, T: ?Sized> {
|
||||
inner_guard: InnerGuard,
|
||||
inner: &'a RwLock<T>,
|
||||
}
|
||||
|
||||
impl<'a, T> Deref for RwLockReadGuard<'a, T> {
|
||||
impl<'a, T: ?Sized> Deref for RwLockReadGuard<'a, T> {
|
||||
type Target = T;
|
||||
|
||||
fn deref(&self) -> &T {
|
||||
@ -425,25 +427,25 @@ impl<'a, T> Deref for RwLockReadGuard<'a, T> {
|
||||
}
|
||||
}
|
||||
|
||||
impl<'a, T> Drop for RwLockReadGuard<'a, T> {
|
||||
impl<'a, T: ?Sized> Drop for RwLockReadGuard<'a, T> {
|
||||
fn drop(&mut self) {
|
||||
self.inner.lock.fetch_sub(READER, Release);
|
||||
}
|
||||
}
|
||||
|
||||
impl<'a, T: fmt::Debug> fmt::Debug for RwLockReadGuard<'a, T> {
|
||||
impl<'a, T: ?Sized + fmt::Debug> fmt::Debug for RwLockReadGuard<'a, T> {
|
||||
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
||||
fmt::Debug::fmt(&**self, f)
|
||||
}
|
||||
}
|
||||
|
||||
/// A guard that provides mutable data access.
|
||||
pub struct RwLockWriteGuard<'a, T> {
|
||||
inner: &'a RwLock<T>,
|
||||
pub struct RwLockWriteGuard<'a, T: ?Sized> {
|
||||
inner_guard: InnerGuard,
|
||||
inner: &'a RwLock<T>,
|
||||
}
|
||||
|
||||
impl<'a, T> Deref for RwLockWriteGuard<'a, T> {
|
||||
impl<'a, T: ?Sized> Deref for RwLockWriteGuard<'a, T> {
|
||||
type Target = T;
|
||||
|
||||
fn deref(&self) -> &T {
|
||||
@ -451,19 +453,19 @@ impl<'a, T> Deref for RwLockWriteGuard<'a, T> {
|
||||
}
|
||||
}
|
||||
|
||||
impl<'a, T> DerefMut for RwLockWriteGuard<'a, T> {
|
||||
impl<'a, T: ?Sized> DerefMut for RwLockWriteGuard<'a, T> {
|
||||
fn deref_mut(&mut self) -> &mut Self::Target {
|
||||
unsafe { &mut *self.inner.val.get() }
|
||||
}
|
||||
}
|
||||
|
||||
impl<'a, T> Drop for RwLockWriteGuard<'a, T> {
|
||||
impl<'a, T: ?Sized> Drop for RwLockWriteGuard<'a, T> {
|
||||
fn drop(&mut self) {
|
||||
self.inner.lock.fetch_and(!WRITER, Release);
|
||||
}
|
||||
}
|
||||
|
||||
impl<'a, T: fmt::Debug> fmt::Debug for RwLockWriteGuard<'a, T> {
|
||||
impl<'a, T: ?Sized + fmt::Debug> fmt::Debug for RwLockWriteGuard<'a, T> {
|
||||
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
||||
fmt::Debug::fmt(&**self, f)
|
||||
}
|
||||
@ -471,12 +473,12 @@ impl<'a, T: fmt::Debug> fmt::Debug for RwLockWriteGuard<'a, T> {
|
||||
|
||||
/// A guard that provides immutable data access but can be atomically
|
||||
/// upgraded to `RwLockWriteGuard`.
|
||||
pub struct RwLockUpgradeableGuard<'a, T> {
|
||||
inner: &'a RwLock<T>,
|
||||
pub struct RwLockUpgradeableGuard<'a, T: ?Sized> {
|
||||
inner_guard: InnerGuard,
|
||||
inner: &'a RwLock<T>,
|
||||
}
|
||||
|
||||
impl<'a, T> RwLockUpgradeableGuard<'a, T> {
|
||||
impl<'a, T: ?Sized> RwLockUpgradeableGuard<'a, T> {
|
||||
/// Upgrade this upread guard to a write guard atomically.
|
||||
///
|
||||
/// After calling this method, subsequent readers will be blocked
|
||||
@ -517,7 +519,7 @@ impl<'a, T> RwLockUpgradeableGuard<'a, T> {
|
||||
}
|
||||
}
|
||||
|
||||
impl<'a, T> Deref for RwLockUpgradeableGuard<'a, T> {
|
||||
impl<'a, T: ?Sized> Deref for RwLockUpgradeableGuard<'a, T> {
|
||||
type Target = T;
|
||||
|
||||
fn deref(&self) -> &T {
|
||||
@ -525,13 +527,13 @@ impl<'a, T> Deref for RwLockUpgradeableGuard<'a, T> {
|
||||
}
|
||||
}
|
||||
|
||||
impl<'a, T> Drop for RwLockUpgradeableGuard<'a, T> {
|
||||
impl<'a, T: ?Sized> Drop for RwLockUpgradeableGuard<'a, T> {
|
||||
fn drop(&mut self) {
|
||||
self.inner.lock.fetch_sub(UPGRADEABLE_READER, Release);
|
||||
}
|
||||
}
|
||||
|
||||
impl<'a, T: fmt::Debug> fmt::Debug for RwLockUpgradeableGuard<'a, T> {
|
||||
impl<'a, T: ?Sized + fmt::Debug> fmt::Debug for RwLockUpgradeableGuard<'a, T> {
|
||||
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
||||
fmt::Debug::fmt(&**self, f)
|
||||
}
|
||||
|
@ -84,8 +84,7 @@ use super::WaitQueue;
|
||||
/// assert_eq!(*w2, 7);
|
||||
/// } // write mutex is dropped at this point
|
||||
/// ```
|
||||
pub struct RwMutex<T> {
|
||||
val: UnsafeCell<T>,
|
||||
pub struct RwMutex<T: ?Sized> {
|
||||
/// The internal representation of the mutex state is as follows:
|
||||
/// - **Bit 63:** Writer mutex.
|
||||
/// - **Bit 62:** Upgradeable reader mutex.
|
||||
@ -94,6 +93,7 @@ pub struct RwMutex<T> {
|
||||
lock: AtomicUsize,
|
||||
/// Threads that fail to acquire the mutex will sleep on this waitqueue.
|
||||
queue: WaitQueue,
|
||||
val: UnsafeCell<T>,
|
||||
}
|
||||
|
||||
const READER: usize = 1;
|
||||
@ -111,7 +111,9 @@ impl<T> RwMutex<T> {
|
||||
queue: WaitQueue::new(),
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
impl<T: ?Sized> RwMutex<T> {
|
||||
/// Acquire a read mutex and sleep until it can be acquired.
|
||||
///
|
||||
/// The calling thread will sleep until there are no writers or upgrading
|
||||
@ -188,7 +190,7 @@ impl<T> RwMutex<T> {
|
||||
}
|
||||
}
|
||||
|
||||
impl<T: fmt::Debug> fmt::Debug for RwMutex<T> {
|
||||
impl<T: ?Sized + fmt::Debug> fmt::Debug for RwMutex<T> {
|
||||
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
||||
fmt::Debug::fmt(&self.val, f)
|
||||
}
|
||||
@ -196,24 +198,24 @@ impl<T: fmt::Debug> fmt::Debug for RwMutex<T> {
|
||||
|
||||
/// Because there can be more than one readers to get the T's immutable ref,
|
||||
/// so T must be Sync to guarantee the sharing safety.
|
||||
unsafe impl<T: Send> Send for RwMutex<T> {}
|
||||
unsafe impl<T: Send + Sync> Sync for RwMutex<T> {}
|
||||
unsafe impl<T: ?Sized + Send> Send for RwMutex<T> {}
|
||||
unsafe impl<T: ?Sized + Send + Sync> Sync for RwMutex<T> {}
|
||||
|
||||
impl<'a, T> !Send for RwMutexWriteGuard<'a, T> {}
|
||||
unsafe impl<T: Sync> Sync for RwMutexWriteGuard<'_, T> {}
|
||||
impl<'a, T: ?Sized> !Send for RwMutexWriteGuard<'a, T> {}
|
||||
unsafe impl<T: ?Sized + Sync> Sync for RwMutexWriteGuard<'_, T> {}
|
||||
|
||||
impl<'a, T> !Send for RwMutexReadGuard<'a, T> {}
|
||||
unsafe impl<T: Sync> Sync for RwMutexReadGuard<'_, T> {}
|
||||
impl<'a, T: ?Sized> !Send for RwMutexReadGuard<'a, T> {}
|
||||
unsafe impl<T: ?Sized + Sync> Sync for RwMutexReadGuard<'_, T> {}
|
||||
|
||||
impl<'a, T> !Send for RwMutexUpgradeableGuard<'a, T> {}
|
||||
unsafe impl<T: Sync> Sync for RwMutexUpgradeableGuard<'_, T> {}
|
||||
impl<'a, T: ?Sized> !Send for RwMutexUpgradeableGuard<'a, T> {}
|
||||
unsafe impl<T: ?Sized + Sync> Sync for RwMutexUpgradeableGuard<'_, T> {}
|
||||
|
||||
/// A guard that provides immutable data access.
|
||||
pub struct RwMutexReadGuard<'a, T> {
|
||||
pub struct RwMutexReadGuard<'a, T: ?Sized> {
|
||||
inner: &'a RwMutex<T>,
|
||||
}
|
||||
|
||||
impl<'a, T> Deref for RwMutexReadGuard<'a, T> {
|
||||
impl<'a, T: ?Sized> Deref for RwMutexReadGuard<'a, T> {
|
||||
type Target = T;
|
||||
|
||||
fn deref(&self) -> &T {
|
||||
@ -221,7 +223,7 @@ impl<'a, T> Deref for RwMutexReadGuard<'a, T> {
|
||||
}
|
||||
}
|
||||
|
||||
impl<'a, T> Drop for RwMutexReadGuard<'a, T> {
|
||||
impl<'a, T: ?Sized> Drop for RwMutexReadGuard<'a, T> {
|
||||
fn drop(&mut self) {
|
||||
// When there are no readers, wake up a waiting writer.
|
||||
if self.inner.lock.fetch_sub(READER, Release) == READER {
|
||||
@ -231,11 +233,11 @@ impl<'a, T> Drop for RwMutexReadGuard<'a, T> {
|
||||
}
|
||||
|
||||
/// A guard that provides mutable data access.
|
||||
pub struct RwMutexWriteGuard<'a, T> {
|
||||
pub struct RwMutexWriteGuard<'a, T: ?Sized> {
|
||||
inner: &'a RwMutex<T>,
|
||||
}
|
||||
|
||||
impl<'a, T> Deref for RwMutexWriteGuard<'a, T> {
|
||||
impl<'a, T: ?Sized> Deref for RwMutexWriteGuard<'a, T> {
|
||||
type Target = T;
|
||||
|
||||
fn deref(&self) -> &T {
|
||||
@ -243,13 +245,13 @@ impl<'a, T> Deref for RwMutexWriteGuard<'a, T> {
|
||||
}
|
||||
}
|
||||
|
||||
impl<'a, T> DerefMut for RwMutexWriteGuard<'a, T> {
|
||||
impl<'a, T: ?Sized> DerefMut for RwMutexWriteGuard<'a, T> {
|
||||
fn deref_mut(&mut self) -> &mut Self::Target {
|
||||
unsafe { &mut *self.inner.val.get() }
|
||||
}
|
||||
}
|
||||
|
||||
impl<'a, T> Drop for RwMutexWriteGuard<'a, T> {
|
||||
impl<'a, T: ?Sized> Drop for RwMutexWriteGuard<'a, T> {
|
||||
fn drop(&mut self) {
|
||||
self.inner.lock.fetch_and(!WRITER, Release);
|
||||
|
||||
@ -263,11 +265,11 @@ impl<'a, T> Drop for RwMutexWriteGuard<'a, T> {
|
||||
|
||||
/// A guard that provides immutable data access but can be atomically
|
||||
/// upgraded to `RwMutexWriteGuard`.
|
||||
pub struct RwMutexUpgradeableGuard<'a, T> {
|
||||
pub struct RwMutexUpgradeableGuard<'a, T: ?Sized> {
|
||||
inner: &'a RwMutex<T>,
|
||||
}
|
||||
|
||||
impl<'a, T> RwMutexUpgradeableGuard<'a, T> {
|
||||
impl<'a, T: ?Sized> RwMutexUpgradeableGuard<'a, T> {
|
||||
/// Upgrade this upread guard to a write guard atomically.
|
||||
///
|
||||
/// After calling this method, subsequent readers will be blocked
|
||||
@ -307,7 +309,7 @@ impl<'a, T> RwMutexUpgradeableGuard<'a, T> {
|
||||
}
|
||||
}
|
||||
|
||||
impl<'a, T> Deref for RwMutexUpgradeableGuard<'a, T> {
|
||||
impl<'a, T: ?Sized> Deref for RwMutexUpgradeableGuard<'a, T> {
|
||||
type Target = T;
|
||||
|
||||
fn deref(&self) -> &T {
|
||||
@ -315,7 +317,7 @@ impl<'a, T> Deref for RwMutexUpgradeableGuard<'a, T> {
|
||||
}
|
||||
}
|
||||
|
||||
impl<'a, T> Drop for RwMutexUpgradeableGuard<'a, T> {
|
||||
impl<'a, T: ?Sized> Drop for RwMutexUpgradeableGuard<'a, T> {
|
||||
fn drop(&mut self) {
|
||||
let res = self.inner.lock.fetch_sub(UPGRADEABLE_READER, Release);
|
||||
if res == 0 {
|
||||
|
@ -13,20 +13,22 @@ use crate::{
|
||||
};
|
||||
|
||||
/// A spin lock.
|
||||
pub struct SpinLock<T> {
|
||||
val: UnsafeCell<T>,
|
||||
pub struct SpinLock<T: ?Sized> {
|
||||
lock: AtomicBool,
|
||||
val: UnsafeCell<T>,
|
||||
}
|
||||
|
||||
impl<T> SpinLock<T> {
|
||||
/// Creates a new spin lock.
|
||||
pub const fn new(val: T) -> Self {
|
||||
Self {
|
||||
val: UnsafeCell::new(val),
|
||||
lock: AtomicBool::new(false),
|
||||
val: UnsafeCell::new(val),
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
impl<T: ?Sized> SpinLock<T> {
|
||||
/// Acquire the spin lock with disabling the local IRQs. This is the most secure
|
||||
/// locking way.
|
||||
///
|
||||
@ -102,15 +104,15 @@ impl<T> SpinLock<T> {
|
||||
}
|
||||
}
|
||||
|
||||
impl<T: fmt::Debug> fmt::Debug for SpinLock<T> {
|
||||
impl<T: ?Sized + fmt::Debug> fmt::Debug for SpinLock<T> {
|
||||
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
||||
fmt::Debug::fmt(&self.val, f)
|
||||
}
|
||||
}
|
||||
|
||||
// Safety. Only a single lock holder is permitted to access the inner data of Spinlock.
|
||||
unsafe impl<T: Send> Send for SpinLock<T> {}
|
||||
unsafe impl<T: Send> Sync for SpinLock<T> {}
|
||||
unsafe impl<T: ?Sized + Send> Send for SpinLock<T> {}
|
||||
unsafe impl<T: ?Sized + Send> Sync for SpinLock<T> {}
|
||||
|
||||
enum InnerGuard {
|
||||
IrqGuard(DisabledLocalIrqGuard),
|
||||
@ -118,12 +120,12 @@ enum InnerGuard {
|
||||
}
|
||||
|
||||
/// The guard of a spin lock that disables the local IRQs.
|
||||
pub struct SpinLockGuard<'a, T> {
|
||||
lock: &'a SpinLock<T>,
|
||||
pub struct SpinLockGuard<'a, T: ?Sized> {
|
||||
inner_guard: InnerGuard,
|
||||
lock: &'a SpinLock<T>,
|
||||
}
|
||||
|
||||
impl<'a, T> Deref for SpinLockGuard<'a, T> {
|
||||
impl<'a, T: ?Sized> Deref for SpinLockGuard<'a, T> {
|
||||
type Target = T;
|
||||
|
||||
fn deref(&self) -> &T {
|
||||
@ -131,26 +133,26 @@ impl<'a, T> Deref for SpinLockGuard<'a, T> {
|
||||
}
|
||||
}
|
||||
|
||||
impl<'a, T> DerefMut for SpinLockGuard<'a, T> {
|
||||
impl<'a, T: ?Sized> DerefMut for SpinLockGuard<'a, T> {
|
||||
fn deref_mut(&mut self) -> &mut Self::Target {
|
||||
unsafe { &mut *self.lock.val.get() }
|
||||
}
|
||||
}
|
||||
|
||||
impl<'a, T> Drop for SpinLockGuard<'a, T> {
|
||||
impl<'a, T: ?Sized> Drop for SpinLockGuard<'a, T> {
|
||||
fn drop(&mut self) {
|
||||
self.lock.release_lock();
|
||||
}
|
||||
}
|
||||
|
||||
impl<'a, T: fmt::Debug> fmt::Debug for SpinLockGuard<'a, T> {
|
||||
impl<'a, T: ?Sized + fmt::Debug> fmt::Debug for SpinLockGuard<'a, T> {
|
||||
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
||||
fmt::Debug::fmt(&**self, f)
|
||||
}
|
||||
}
|
||||
|
||||
impl<'a, T> !Send for SpinLockGuard<'a, T> {}
|
||||
impl<'a, T: ?Sized> !Send for SpinLockGuard<'a, T> {}
|
||||
|
||||
// Safety. `SpinLockGuard` can be shared between tasks/threads in same CPU.
|
||||
// As `lock()` is only called when there are no race conditions caused by interrupts.
|
||||
unsafe impl<T: Sync> Sync for SpinLockGuard<'_, T> {}
|
||||
unsafe impl<T: ?Sized + Sync> Sync for SpinLockGuard<'_, T> {}
|
||||
|
Reference in New Issue
Block a user