salsa/tests/parallel/setup.rs

66 lines
1.7 KiB
Rust
Raw Normal View History

2024-07-24 09:46:19 +00:00
use crossbeam::atomic::AtomicCell;
2022-08-09 10:06:39 +00:00
use crate::signal::Signal;
/// Various "knobs" and utilities used by tests to force
/// a certain behavior.
pub(crate) trait Knobs {
fn knobs(&self) -> &KnobsStruct;
fn signal(&self, stage: usize);
fn wait_for(&self, stage: usize);
}
/// Various "knobs" that can be used to customize how the queries
/// behave on one specific thread. Note that this state is
/// intentionally thread-local (apart from `signal`).
2024-07-24 09:46:19 +00:00
#[derive(Default)]
2022-08-09 10:06:39 +00:00
pub(crate) struct KnobsStruct {
/// A kind of flexible barrier used to coordinate execution across
/// threads to ensure we reach various weird states.
2024-07-24 09:46:19 +00:00
pub(crate) signal: Signal,
2022-08-09 10:06:39 +00:00
/// When this database is about to block, send this signal.
2024-07-24 09:46:19 +00:00
pub(crate) signal_on_will_block: AtomicCell<usize>,
/// When this database has set the cancellation flag, send this signal.
pub(crate) signal_on_did_cancel: AtomicCell<usize>,
2022-08-09 10:06:39 +00:00
}
2024-07-17 13:18:43 +00:00
#[salsa::db]
2022-08-09 10:06:39 +00:00
#[derive(Default)]
pub(crate) struct Database {
storage: salsa::Storage<Self>,
2022-08-09 10:33:46 +00:00
knobs: KnobsStruct,
2022-08-09 10:06:39 +00:00
}
2024-07-17 13:18:43 +00:00
#[salsa::db]
2022-08-09 10:06:39 +00:00
impl salsa::Database for Database {
fn salsa_event(&self, event: salsa::Event) {
match event.kind {
salsa::EventKind::WillBlockOn { .. } => {
self.signal(self.knobs().signal_on_will_block.load());
}
salsa::EventKind::DidSetCancellationFlag => {
self.signal(self.knobs().signal_on_did_cancel.load());
}
_ => {}
2022-08-09 10:06:39 +00:00
}
}
}
impl Knobs for Database {
fn knobs(&self) -> &KnobsStruct {
&self.knobs
}
fn signal(&self, stage: usize) {
self.knobs.signal.signal(stage);
}
fn wait_for(&self, stage: usize) {
self.knobs.signal.wait_for(stage);
}
}