use crate::implementation::{TestContext, TestContextImpl}; #[salsa::query_group(MemoizedDepInputs)] pub(crate) trait MemoizedDepInputsContext: TestContext { fn dep_memoized2(&self) -> usize; fn dep_memoized1(&self) -> usize; #[salsa::dependencies] fn dep_derived1(&self) -> usize; #[salsa::input] fn dep_input1(&self) -> usize; #[salsa::input] fn dep_input2(&self) -> usize; } fn dep_memoized2(db: &dyn MemoizedDepInputsContext) -> usize { db.log().add("Memoized2 invoked"); db.dep_memoized1() } fn dep_memoized1(db: &dyn MemoizedDepInputsContext) -> usize { db.log().add("Memoized1 invoked"); db.dep_derived1() * 2 } fn dep_derived1(db: &dyn MemoizedDepInputsContext) -> usize { db.log().add("Derived1 invoked"); db.dep_input1() / 2 } #[test] fn revalidate() { let db = &mut TestContextImpl::default(); db.set_dep_input1(0); // Initial run starts from Memoized2: let v = db.dep_memoized2(); assert_eq!(v, 0); db.assert_log(&["Memoized2 invoked", "Memoized1 invoked", "Derived1 invoked"]); // After that, we first try to validate Memoized1 but wind up // running Memoized2. Note that we don't try to validate // Derived1, so it is invoked by Memoized1. db.set_dep_input1(44); let v = db.dep_memoized2(); assert_eq!(v, 44); db.assert_log(&["Memoized1 invoked", "Derived1 invoked", "Memoized2 invoked"]); // Here validation of Memoized1 succeeds so Memoized2 never runs. let value = db.remove_dep_input1() + 1; db.set_dep_input1(value); let v = db.dep_memoized2(); assert_eq!(v, 44); db.assert_log(&["Memoized1 invoked", "Derived1 invoked"]); // Here, a change to input2 doesn't affect us, so nothing runs. db.set_dep_input2(45); let v = db.dep_memoized2(); assert_eq!(v, 44); db.assert_log(&[]); }