| // Copyright 2025 The Pigweed Authors |
| // |
| // Licensed under the Apache License, Version 2.0 (the "License"); you may not |
| // use this file except in compliance with the License. You may obtain a copy of |
| // the License at |
| // |
| // https://www.apache.org/licenses/LICENSE-2.0 |
| // |
| // Unless required by applicable law or agreed to in writing, software |
| // distributed under the License is distributed on an "AS IS" BASIS, WITHOUT |
| // WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the |
| // License for the specific language governing permissions and limitations under |
| // the License. |
| |
| #![no_std] |
| #![cfg_attr(test, no_main)] |
| |
| use core::cell::UnsafeCell; |
| use core::ops::{Deref, DerefMut}; |
| |
| pub use spinlock_backend::BareSpinLock; |
| pub use spinlock_core::*; |
| |
| pub struct SpinLockGuard<'lock, T> { |
| lock: &'lock SpinLock<T>, |
| _inner_guard: <BareSpinLock as BareSpinLockApi>::Guard<'lock>, |
| } |
| |
| impl<T> Deref for SpinLockGuard<'_, T> { |
| type Target = T; |
| |
| fn deref(&self) -> &T { |
| unsafe { &*self.lock.data.get() } |
| } |
| } |
| |
| impl<T> DerefMut for SpinLockGuard<'_, T> { |
| fn deref_mut(&mut self) -> &mut T { |
| unsafe { &mut *self.lock.data.get() } |
| } |
| } |
| |
| pub struct SpinLock<T> { |
| data: UnsafeCell<T>, |
| inner: BareSpinLock, |
| } |
| |
| impl<T> SpinLock<T> { |
| pub fn new(initial_value: T) -> Self { |
| Self { |
| data: UnsafeCell::new(initial_value), |
| inner: BareSpinLock::new(), |
| } |
| } |
| |
| pub fn try_lock(&self) -> Option<SpinLockGuard<'_, T>> { |
| self.inner.try_lock().map(|guard| SpinLockGuard { |
| lock: self, |
| _inner_guard: guard, |
| }) |
| } |
| |
| pub fn lock(&self) -> SpinLockGuard<'_, T> { |
| let inner_guard = self.inner.lock(); |
| SpinLockGuard { |
| lock: self, |
| _inner_guard: inner_guard, |
| } |
| } |
| } |
| |
| #[cfg(test)] |
| mod tests { |
| use super::*; |
| use unittest::test; |
| |
| #[test] |
| fn bare_try_lock_returns_correct_value() -> unittest::Result<()> { |
| let lock = BareSpinLock::new(); |
| |
| { |
| let _sentinel = lock.lock(); |
| unittest::assert_true!(lock.try_lock().is_none()); |
| } |
| |
| unittest::assert_true!(lock.try_lock().is_some()); |
| |
| Ok(()) |
| } |
| |
| #[test] |
| fn try_lock_returns_correct_value() -> unittest::Result<()> { |
| let lock = SpinLock::new(false); |
| |
| { |
| let mut guard = lock.lock(); |
| *guard = true; |
| unittest::assert_true!(lock.try_lock().is_none()); |
| } |
| |
| let guard = lock.lock(); |
| unittest::assert_true!(*guard); |
| |
| Ok(()) |
| } |
| } |