Skip to content

Commit

Permalink
[WIP] on add semaphore, rwlock, condvar and barrier in axsync, bug in…
Browse files Browse the repository at this point in the history
… condvar?
  • Loading branch information
hky1999 committed Oct 27, 2024
1 parent 7251ab8 commit e8d3ea1
Show file tree
Hide file tree
Showing 13 changed files with 1,861 additions and 32 deletions.
2 changes: 2 additions & 0 deletions Cargo.lock

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

4 changes: 3 additions & 1 deletion modules/axsync/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -10,12 +10,14 @@ repository = "https://github.com/arceos-org/arceos/tree/main/modules/axsync"
documentation = "https://arceos-org.github.io/arceos/axsync/index.html"

[features]
multitask = ["axtask/multitask"]
multitask = ["axtask/multitask", "dep:axhal"]
default = []

[dependencies]
kspin = "0.1"
cfg-if = "1.0"
axtask = { workspace = true }
axhal = { workspace = true, optional = true }

[dev-dependencies]
rand = "0.8"
Expand Down
115 changes: 115 additions & 0 deletions modules/axsync/src/barrier.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,115 @@
//! Synchronization primitive allowing multiple threads to synchronize the
//! beginning of some computation.
//!
//! Implementation adapted from the 'Barrier' type of the standard library. See:
//! <https://doc.rust-lang.org/std/sync/struct.Barrier.html>
//!
//! Note: [`Barrier`] is not available when the `multitask` feature is disabled.
use core::fmt;

use crate::condvar::Condvar;
use crate::mutex::Mutex;

/// A barrier enables multiple threads to synchronize the beginning
/// of some computation.
pub struct Barrier {
lock: Mutex<BarrierState>,
cvar: Condvar,
num_threads: usize,
}

// The inner state of a double barrier
struct BarrierState {
count: usize,
generation_id: usize,
}

/// A `BarrierWaitResult` is returned by [`Barrier::wait()`] when all threads
/// in the [`Barrier`] have rendezvoused.
pub struct BarrierWaitResult(bool);

impl fmt::Debug for Barrier {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
f.debug_struct("Barrier").finish_non_exhaustive()
}
}

impl Barrier {
/// Creates a new barrier that can block a given number of threads.
///
/// A barrier will block `n`-1 threads which call [`wait()`] and then wake
/// up all threads at once when the `n`th thread calls [`wait()`].
///
/// [`wait()`]: Barrier::wait
pub const fn new(n: usize) -> Self {
Self {
lock: Mutex::new(BarrierState {
count: 0,
generation_id: 0,
}),
cvar: Condvar::new(),
num_threads: n,
}
}

/// Blocks the current thread until all threads have rendezvoused here.
///
/// Barriers are re-usable after all threads have rendezvoused once, and can
/// be used continuously.
///
/// A single (arbitrary) thread will receive a [`BarrierWaitResult`] that
/// returns `true` from [`BarrierWaitResult::is_leader()`] when returning
/// from this function, and all other threads will receive a result that
/// will return `false` from [`BarrierWaitResult::is_leader()`].
pub fn wait(&self) -> BarrierWaitResult {
let mut lock = self.lock.lock();
lock.count += 1;

if lock.count < self.num_threads {
// not the leader
let local_gen = lock.generation_id;
let _guard = self
.cvar
.wait_while(lock, |state| local_gen == state.generation_id);
BarrierWaitResult(false)
} else {
// this thread is the leader,
// and is responsible for incrementing the generation
lock.count = 0;
lock.generation_id = lock.generation_id.wrapping_add(1);
self.cvar.notify_all();
BarrierWaitResult(true)
}
}
}

impl fmt::Debug for BarrierWaitResult {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
f.debug_struct("BarrierWaitResult")
.field("is_leader", &self.is_leader())
.finish()
}
}

impl BarrierWaitResult {
/// Returns whether this thread from [`wait`] is the "leader thread".
///
/// Only one thread will have `true` returned from their result, all other
/// threads will have `false` returned.
///
/// [`wait`]: struct.Barrier.html#method.wait
///
/// # Examples
///
/// ```
/// use spin;
///
/// let barrier = spin::Barrier::new(1);
/// let barrier_wait_result = barrier.wait();
/// println!("{:?}", barrier_wait_result.is_leader());
/// ```
pub fn is_leader(&self) -> bool {
self.0
}
}
37 changes: 37 additions & 0 deletions modules/axsync/src/condvar/mod.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,37 @@
//! The Condition Variable
//!
//! Implementation adapted from the 'RwLock' type of the standard library. See:
//! <https://doc.rust-lang.org/stable/std/sync/struct.Condvar.html>
//!
//! Note: [`Condvar`] is not available when the `multitask` feature is disabled.
#[cfg(not(feature = "multitask"))]
mod no_thread;
#[cfg(not(feature = "multitask"))]
pub use no_thread::Condvar;

#[cfg(feature = "multitask")]
mod multitask;
#[cfg(feature = "multitask")]
pub use multitask::Condvar;



/// A type indicating whether a timed wait on a condition variable returned
/// due to a time out or not.
///
/// It is returned by the [`wait_timeout`] method.
///
/// [`wait_timeout`]: Condvar::wait_timeout
#[cfg(feature = "irq")]
#[derive(Debug, PartialEq, Eq, Copy, Clone)]
pub struct WaitTimeoutResult(bool);

#[cfg(feature = "irq")]
impl WaitTimeoutResult {
/// Returns `true` if the wait was known to have timed out.
#[must_use]
pub fn timed_out(&self) -> bool {
self.0
}
}
Loading

0 comments on commit e8d3ea1

Please sign in to comment.