2024-08-04 05:33:58 +00:00
|
|
|
use std::sync::Arc;
|
|
|
|
|
2024-07-24 09:46:19 +00:00
|
|
|
use crossbeam::atomic::AtomicCell;
|
2024-08-04 05:33:58 +00:00
|
|
|
use salsa::Database;
|
2022-08-09 10:06:39 +00:00
|
|
|
|
|
|
|
use crate::signal::Signal;
|
|
|
|
|
|
|
|
/// Various "knobs" and utilities used by tests to force
|
|
|
|
/// a certain behavior.
|
2024-07-27 12:29:41 +00:00
|
|
|
#[salsa::db]
|
|
|
|
pub(crate) trait KnobsDatabase: Database {
|
|
|
|
fn knobs(&self) -> &Knobs;
|
2022-08-09 10:06:39 +00:00
|
|
|
|
|
|
|
fn signal(&self, stage: usize);
|
|
|
|
|
|
|
|
fn wait_for(&self, stage: usize);
|
|
|
|
}
|
|
|
|
|
2024-08-04 05:33:58 +00:00
|
|
|
/// A database containing various "knobs" that can be used to customize how the queries
|
2022-08-09 10:06:39 +00:00
|
|
|
/// behave on one specific thread. Note that this state is
|
|
|
|
/// intentionally thread-local (apart from `signal`).
|
2024-08-04 05:33:58 +00:00
|
|
|
#[salsa::db]
|
2024-07-24 09:46:19 +00:00
|
|
|
#[derive(Default)]
|
2024-07-27 12:29:41 +00:00
|
|
|
pub(crate) struct Knobs {
|
2024-08-04 05:33:58 +00:00
|
|
|
storage: salsa::Storage<Self>,
|
|
|
|
|
2022-08-09 10:06:39 +00:00
|
|
|
/// A kind of flexible barrier used to coordinate execution across
|
|
|
|
/// threads to ensure we reach various weird states.
|
2024-08-04 05:33:58 +00:00
|
|
|
pub(crate) signal: Arc<Signal>,
|
2022-08-09 10:06:39 +00:00
|
|
|
|
2024-07-24 10:36:07 +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>,
|
2024-07-24 10:36:07 +00:00
|
|
|
|
|
|
|
/// 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-08-04 05:33:58 +00:00
|
|
|
impl Clone for Knobs {
|
|
|
|
#[track_caller]
|
|
|
|
fn clone(&self) -> Self {
|
|
|
|
// To avoid mistakes, check that when we clone, we haven't customized this behavior yet
|
|
|
|
assert_eq!(self.signal_on_will_block.load(), 0);
|
|
|
|
assert_eq!(self.signal_on_did_cancel.load(), 0);
|
|
|
|
Self {
|
|
|
|
storage: self.storage.clone(),
|
|
|
|
signal: self.signal.clone(),
|
|
|
|
signal_on_will_block: AtomicCell::new(0),
|
|
|
|
signal_on_did_cancel: AtomicCell::new(0),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[salsa::db]
|
|
|
|
impl salsa::Database for Knobs {
|
|
|
|
fn salsa_event(&self, event: &dyn Fn() -> salsa::Event) {
|
2024-07-27 11:47:58 +00:00
|
|
|
let event = event();
|
2024-07-24 10:36:07 +00:00
|
|
|
match event.kind {
|
|
|
|
salsa::EventKind::WillBlockOn { .. } => {
|
2024-08-04 05:33:58 +00:00
|
|
|
self.signal(self.signal_on_will_block.load());
|
2024-07-24 10:36:07 +00:00
|
|
|
}
|
|
|
|
salsa::EventKind::DidSetCancellationFlag => {
|
2024-08-04 05:33:58 +00:00
|
|
|
self.signal(self.signal_on_did_cancel.load());
|
2024-07-24 10:36:07 +00:00
|
|
|
}
|
|
|
|
_ => {}
|
2022-08-09 10:06:39 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-07-27 12:29:41 +00:00
|
|
|
#[salsa::db]
|
2024-08-04 05:33:58 +00:00
|
|
|
impl KnobsDatabase for Knobs {
|
2024-07-27 12:29:41 +00:00
|
|
|
fn knobs(&self) -> &Knobs {
|
|
|
|
self
|
2022-08-09 10:06:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
fn signal(&self, stage: usize) {
|
2024-07-27 12:29:41 +00:00
|
|
|
self.signal.signal(stage);
|
2022-08-09 10:06:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
fn wait_for(&self, stage: usize) {
|
2024-07-27 12:29:41 +00:00
|
|
|
self.signal.wait_for(stage);
|
2022-08-09 10:06:39 +00:00
|
|
|
}
|
|
|
|
}
|