synchronoise::event

Struct SignalEvent

Source
pub struct SignalEvent { /* private fields */ }
Expand description

A synchronization primitive that allows one or more threads to wait on a signal from another thread.

With a SignalEvent, it’s possible to have one or more threads gate on a signal from another thread. The behavior for what happens when an event is signaled depends on the value of the signal_kind parameter given to new, or whether auto or manual is used to construct the SignalEvent:

  • A value of SignalKind::Auto (or a SignalEvent created via SignalEvent::auto()) will automatically reset the signal when a thread is resumed by this event. If more than one thread is waiting on the event when it is signaled, only one will be resumed.
  • A value of SignalKind::Manual (or a SignalEvent created via SignalEvent::manual()) will remain signaled until it is manually reset. If more than one thread is waiting on the event when it is signaled, all of them will be resumed. Any other thread that tries to wait on the signal before it is reset will not be blocked at all.

SignalEvent is a port of System.Threading.EventWaitHandle from .NET.

§Example

The following example uses two SignalEvents:

  • start_signal is used as a kind of std::sync::Barrier, that keeps all the threads inside the loop from starting until they all have been spawned. All the start.wait() calls resume when start_signal.signal() is called after the initial loop.
    • Note that because the “coordinator” doesn’t wait for each thread to be scheduled before signaling, it’s possible that some later threads may not have had a chance to enter start.wait() before the signal is set. In this case they won’t block in the first place, and immediately return.
  • stop_signal is used to wake up the “coordinator” thread when each “worker” thread is finished with its work. This allows it to keep a count of the number of threads yet to finish, so it can exit its final loop when all the threads have stopped.
use synchronoise::SignalEvent;
use std::sync::Arc;
use std::thread;
use std::time::Duration;

let start_signal = Arc::new(SignalEvent::manual(false));
let stop_signal = Arc::new(SignalEvent::auto(false));
let mut thread_count = 5;

for i in 0..thread_count {
    let start = start_signal.clone();
    let stop = stop_signal.clone();
    thread::spawn(move || {
        // as a Manual-reset signal, all the threads will start at the same time
        start.wait();
        thread::sleep(Duration::from_secs(i));
        println!("thread {} activated!", i);
        stop.signal();
    });
}

start_signal.signal();

while thread_count > 0 {
    // as an Auto-reset signal, this will automatically reset when resuming
    // so when the loop comes back, we don't have to reset before blocking again
    stop_signal.wait();
    thread_count -= 1;
}

println!("all done!");

Implementations§

Source§

impl SignalEvent

Source

pub fn new(init_state: bool, signal_kind: SignalKind) -> SignalEvent

Creates a new SignalEvent with the given starting state and reset behavior.

If init_state is true, then this SignalEvent will start with the signal already set, so that threads that wait will immediately unblock.

Source

pub fn auto(init_state: bool) -> SignalEvent

Creates a new automatically-resetting SignalEvent with the given starting state.

If init_state is true, then this SignalEvent will start with the signal already set, so that the first thread that tries to wait will immediately unblock.

Source

pub fn manual(init_state: bool) -> SignalEvent

Creates a new manually-resetting SignalEvent with the given starting state.

If init_state is true, then this SignalEvent will start with the signal alraedy set, so that threads that wait will immediately unblock until reset is called.

Source

pub fn status(&self) -> bool

Returns the current signal status of the SignalEvent.

Source

pub fn signal(&self)

Sets the signal on this SignalEvent, potentially waking up one or all threads waiting on it.

If more than one thread is waiting on the event, the behavior is different depending on the SignalKind passed to the event when it was created. For a value of Auto, one thread will be resumed. For a value of Manual, all waiting threads will be resumed.

If no thread is currently waiting on the event, its state will be set regardless. Any future attempts to wait on the event will unblock immediately, except for a SignalKind of Auto, which will immediately unblock the first thread only.

Source

pub fn reset(&self)

Resets the signal on this SignalEvent, allowing threads that wait on it to block.

Source

pub fn wait(&self)

Blocks this thread until another thread calls signal.

If this event is already set, then this function will immediately return without blocking. For events with a SignalKind of Auto, this will reset the signal so that the next thread to wait will block.

Source

pub fn wait_timeout(&self, timeout: Duration) -> bool

Blocks this thread until either another thread calls signal, or until the timeout elapses.

This function returns the status of the signal when it woke up. If this function exits because the signal was set, and this event has a SignalKind of Auto, the signal will be reset so that the next thread to wait will block.

Auto Trait Implementations§

Blanket Implementations§

Source§

impl<T> Any for T
where T: 'static + ?Sized,

Source§

fn type_id(&self) -> TypeId

Gets the TypeId of self. Read more
Source§

impl<T> Borrow<T> for T
where T: ?Sized,

Source§

fn borrow(&self) -> &T

Immutably borrows from an owned value. Read more
Source§

impl<T> BorrowMut<T> for T
where T: ?Sized,

Source§

fn borrow_mut(&mut self) -> &mut T

Mutably borrows from an owned value. Read more
Source§

impl<T> From<T> for T

Source§

fn from(t: T) -> T

Returns the argument unchanged.

Source§

impl<T, U> Into<U> for T
where U: From<T>,

Source§

fn into(self) -> U

Calls U::from(self).

That is, this conversion is whatever the implementation of From<T> for U chooses to do.

Source§

impl<T, U> TryFrom<U> for T
where U: Into<T>,

Source§

type Error = Infallible

The type returned in the event of a conversion error.
Source§

fn try_from(value: U) -> Result<T, <T as TryFrom<U>>::Error>

Performs the conversion.
Source§

impl<T, U> TryInto<U> for T
where U: TryFrom<T>,

Source§

type Error = <U as TryFrom<T>>::Error

The type returned in the event of a conversion error.
Source§

fn try_into(self) -> Result<U, <U as TryFrom<T>>::Error>

Performs the conversion.

Layout§

Note: Most layout information is completely unstable and may even differ between compilations. The only exception is types with certain repr(...) attributes. Please see the Rust Reference's “Type Layout” chapter for details on type layout guarantees.

Size: 384 bytes