2022-08-09 10:33:46 +00:00
|
|
|
//! Test a cycle where no queries recover that occurs across threads.
|
|
|
|
//! See the `../cycles.rs` for a complete listing of cycle tests,
|
|
|
|
//! both intra and cross thread.
|
|
|
|
|
2022-08-09 10:06:39 +00:00
|
|
|
use crate::setup::Knobs;
|
2024-07-27 12:29:41 +00:00
|
|
|
use crate::setup::KnobsDatabase;
|
2022-08-09 10:06:39 +00:00
|
|
|
use expect_test::expect;
|
2024-07-24 09:53:24 +00:00
|
|
|
use salsa::Database as _;
|
2024-07-27 12:29:41 +00:00
|
|
|
use salsa::DatabaseImpl;
|
2024-07-24 09:53:24 +00:00
|
|
|
use salsa::Handle;
|
2022-08-09 10:33:46 +00:00
|
|
|
|
2024-07-16 10:04:01 +00:00
|
|
|
#[salsa::input]
|
2022-08-09 10:06:39 +00:00
|
|
|
pub(crate) struct MyInput {
|
2022-08-09 10:33:46 +00:00
|
|
|
field: i32,
|
2022-08-09 10:06:39 +00:00
|
|
|
}
|
|
|
|
|
2024-07-16 10:04:01 +00:00
|
|
|
#[salsa::tracked]
|
2024-07-27 12:29:41 +00:00
|
|
|
pub(crate) fn a(db: &dyn KnobsDatabase, input: MyInput) -> i32 {
|
2022-08-09 10:33:46 +00:00
|
|
|
// Wait to create the cycle until both threads have entered
|
2022-08-09 10:06:39 +00:00
|
|
|
db.signal(1);
|
|
|
|
db.wait_for(2);
|
|
|
|
|
|
|
|
b(db, input)
|
|
|
|
}
|
|
|
|
|
2024-07-16 10:04:01 +00:00
|
|
|
#[salsa::tracked]
|
2024-07-27 12:29:41 +00:00
|
|
|
pub(crate) fn b(db: &dyn KnobsDatabase, input: MyInput) -> i32 {
|
2022-08-09 10:33:46 +00:00
|
|
|
// Wait to create the cycle until both threads have entered
|
2022-08-09 10:06:39 +00:00
|
|
|
db.wait_for(1);
|
|
|
|
db.signal(2);
|
|
|
|
|
2022-08-09 10:33:46 +00:00
|
|
|
// Wait for thread A to block on this thread
|
2022-08-09 10:06:39 +00:00
|
|
|
db.wait_for(3);
|
2022-08-09 10:33:46 +00:00
|
|
|
|
|
|
|
// Now try to execute A
|
2022-08-09 10:06:39 +00:00
|
|
|
a(db, input)
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn execute() {
|
2024-07-27 12:29:41 +00:00
|
|
|
let db = Handle::new(<DatabaseImpl<Knobs>>::default());
|
2024-07-24 09:53:24 +00:00
|
|
|
db.knobs().signal_on_will_block.store(3);
|
2022-08-09 10:06:39 +00:00
|
|
|
|
2024-07-24 09:53:24 +00:00
|
|
|
let input = MyInput::new(&*db, -1);
|
2022-08-09 10:06:39 +00:00
|
|
|
|
2022-08-09 10:33:46 +00:00
|
|
|
let thread_a = std::thread::spawn({
|
2024-07-24 09:53:24 +00:00
|
|
|
let db = db.clone();
|
2022-08-09 10:06:39 +00:00
|
|
|
move || a(&*db, input)
|
|
|
|
});
|
|
|
|
|
|
|
|
let thread_b = std::thread::spawn({
|
2024-07-24 09:53:24 +00:00
|
|
|
let db = db.clone();
|
2022-08-09 10:06:39 +00:00
|
|
|
move || b(&*db, input)
|
|
|
|
});
|
2022-08-09 10:33:46 +00:00
|
|
|
|
|
|
|
// We expect B to panic because it detects a cycle (it is the one that calls A, ultimately).
|
|
|
|
// Right now, it panics with a string.
|
2022-08-09 10:06:39 +00:00
|
|
|
let err_b = thread_b.join().unwrap_err();
|
2024-07-24 09:53:24 +00:00
|
|
|
db.attach(|_| {
|
|
|
|
if let Some(c) = err_b.downcast_ref::<salsa::Cycle>() {
|
|
|
|
let expected = expect![[r#"
|
|
|
|
[
|
|
|
|
a(0),
|
|
|
|
b(0),
|
|
|
|
]
|
|
|
|
"#]];
|
|
|
|
expected.assert_debug_eq(&c.all_participants(&*db));
|
|
|
|
} else {
|
|
|
|
panic!("b failed in an unexpected way: {:?}", err_b);
|
|
|
|
}
|
|
|
|
});
|
2022-08-09 10:33:46 +00:00
|
|
|
|
|
|
|
// We expect A to propagate a panic, which causes us to use the sentinel
|
|
|
|
// type `Canceled`.
|
|
|
|
assert!(thread_a
|
|
|
|
.join()
|
|
|
|
.unwrap_err()
|
|
|
|
.downcast_ref::<salsa::Cancelled>()
|
|
|
|
.is_some());
|
|
|
|
}
|