//! 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. use crate::setup::Knobs; use crate::setup::KnobsDatabase; use expect_test::expect; use salsa::Database; #[salsa::input] pub(crate) struct MyInput { field: i32, } #[salsa::tracked] pub(crate) fn a(db: &dyn KnobsDatabase, input: MyInput) -> i32 { // Wait to create the cycle until both threads have entered db.signal(1); db.wait_for(2); b(db, input) } #[salsa::tracked] pub(crate) fn b(db: &dyn KnobsDatabase, input: MyInput) -> i32 { // Wait to create the cycle until both threads have entered db.wait_for(1); db.signal(2); // Wait for thread A to block on this thread db.wait_for(3); // Now try to execute A a(db, input) } #[test] fn execute() { let db = Knobs::default(); let input = MyInput::new(&db, -1); let thread_a = std::thread::spawn({ let db = db.clone(); db.knobs().signal_on_will_block.store(3); move || a(&db, input) }); let thread_b = std::thread::spawn({ let db = db.clone(); move || b(&db, input) }); // 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. let err_b = thread_b.join().unwrap_err(); db.attach(|_| { if let Some(c) = err_b.downcast_ref::() { let expected = expect![[r#" [ a(Id(0)), b(Id(0)), ] "#]]; expected.assert_debug_eq(&c.all_participants(&db)); } else { panic!("b failed in an unexpected way: {:?}", err_b); } }); // We expect A to propagate a panic, which causes us to use the sentinel // type `Canceled`. assert!(thread_a .join() .unwrap_err() .downcast_ref::() .is_some()); }