mirror of
https://github.com/salsa-rs/salsa.git
synced 2025-01-12 08:30:51 +00:00
add debugging, get incremental test working
This commit is contained in:
parent
2d2bdfe7f1
commit
9bfd8ebbfa
7 changed files with 135 additions and 38 deletions
|
@ -13,6 +13,8 @@ derive-new = "0.5.5"
|
||||||
rustc-hash = "1.0"
|
rustc-hash = "1.0"
|
||||||
parking_lot = "0.6.4"
|
parking_lot = "0.6.4"
|
||||||
indexmap = "1.0.1"
|
indexmap = "1.0.1"
|
||||||
|
log = "0.4.5"
|
||||||
|
|
||||||
[dev-dependencies]
|
[dev-dependencies]
|
||||||
text-diff = "0.4.0"
|
difference = "2.0"
|
||||||
|
env_logger = "0.5.13"
|
|
@ -76,7 +76,7 @@ where
|
||||||
&self,
|
&self,
|
||||||
query: &'q QC,
|
query: &'q QC,
|
||||||
key: &Q::Key,
|
key: &Q::Key,
|
||||||
descriptor: impl FnOnce() -> QC::QueryDescriptor,
|
descriptor: &QC::QueryDescriptor,
|
||||||
) -> Result<Q::Value, CycleDetected>;
|
) -> Result<Q::Value, CycleDetected>;
|
||||||
|
|
||||||
/// True if the query **may** have changed since the given
|
/// True if the query **may** have changed since the given
|
||||||
|
@ -125,12 +125,13 @@ where
|
||||||
Q: Query<QC>,
|
Q: Query<QC>,
|
||||||
{
|
{
|
||||||
pub fn of(&self, key: Q::Key) -> Q::Value {
|
pub fn of(&self, key: Q::Key) -> Q::Value {
|
||||||
|
let descriptor = self.descriptor(&key);
|
||||||
self.storage
|
self.storage
|
||||||
.try_fetch(self.query, &key, || self.descriptor(&key))
|
.try_fetch(self.query, &key, &descriptor)
|
||||||
.unwrap_or_else(|CycleDetected| {
|
.unwrap_or_else(|CycleDetected| {
|
||||||
self.query
|
self.query
|
||||||
.salsa_runtime()
|
.salsa_runtime()
|
||||||
.report_unexpected_cycle(self.descriptor(&key))
|
.report_unexpected_cycle(descriptor)
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -6,6 +6,7 @@ use crate::QueryContext;
|
||||||
use crate::QueryDescriptor;
|
use crate::QueryDescriptor;
|
||||||
use crate::QueryStorageOps;
|
use crate::QueryStorageOps;
|
||||||
use crate::QueryTable;
|
use crate::QueryTable;
|
||||||
|
use log::debug;
|
||||||
use parking_lot::{RwLock, RwLockUpgradableReadGuard};
|
use parking_lot::{RwLock, RwLockUpgradableReadGuard};
|
||||||
use rustc_hash::FxHashMap;
|
use rustc_hash::FxHashMap;
|
||||||
use std::any::Any;
|
use std::any::Any;
|
||||||
|
@ -28,7 +29,6 @@ where
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Defines the "current state" of query's memoized results.
|
/// Defines the "current state" of query's memoized results.
|
||||||
#[derive(Debug)]
|
|
||||||
enum QueryState<QC, Q>
|
enum QueryState<QC, Q>
|
||||||
where
|
where
|
||||||
Q: Query<QC>,
|
Q: Query<QC>,
|
||||||
|
@ -42,7 +42,6 @@ where
|
||||||
Memoized(Memo<QC, Q>),
|
Memoized(Memo<QC, Q>),
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug)]
|
|
||||||
struct Memo<QC, Q>
|
struct Memo<QC, Q>
|
||||||
where
|
where
|
||||||
Q: Query<QC>,
|
Q: Query<QC>,
|
||||||
|
@ -76,7 +75,6 @@ where
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug)]
|
|
||||||
struct StampedValue<V> {
|
struct StampedValue<V> {
|
||||||
value: V,
|
value: V,
|
||||||
changed_at: Revision,
|
changed_at: Revision,
|
||||||
|
@ -103,17 +101,38 @@ where
|
||||||
&self,
|
&self,
|
||||||
query: &'q QC,
|
query: &'q QC,
|
||||||
key: &Q::Key,
|
key: &Q::Key,
|
||||||
descriptor: impl FnOnce() -> QC::QueryDescriptor,
|
descriptor: &QC::QueryDescriptor,
|
||||||
) -> Result<StampedValue<Q::Value>, CycleDetected> {
|
) -> Result<StampedValue<Q::Value>, CycleDetected> {
|
||||||
let revision_now = query.salsa_runtime().current_revision();
|
let revision_now = query.salsa_runtime().current_revision();
|
||||||
|
|
||||||
|
debug!(
|
||||||
|
"{:?}({:?}): invoked at {:?}",
|
||||||
|
Q::default(),
|
||||||
|
key,
|
||||||
|
revision_now,
|
||||||
|
);
|
||||||
|
|
||||||
let mut old_value = {
|
let mut old_value = {
|
||||||
let map_read = self.map.upgradable_read();
|
let map_read = self.map.upgradable_read();
|
||||||
if let Some(value) = map_read.get(key) {
|
if let Some(value) = map_read.get(key) {
|
||||||
match value {
|
match value {
|
||||||
QueryState::InProgress => return Err(CycleDetected),
|
QueryState::InProgress => return Err(CycleDetected),
|
||||||
QueryState::Memoized(m) => {
|
QueryState::Memoized(m) => {
|
||||||
|
debug!(
|
||||||
|
"{:?}({:?}): found memoized value verified_at={:?}",
|
||||||
|
Q::default(),
|
||||||
|
key,
|
||||||
|
m.verified_at,
|
||||||
|
);
|
||||||
|
|
||||||
if m.verified_at == revision_now {
|
if m.verified_at == revision_now {
|
||||||
|
debug!(
|
||||||
|
"{:?}({:?}): returning memoized value (changed_at={:?})",
|
||||||
|
Q::default(),
|
||||||
|
key,
|
||||||
|
m.changed_at,
|
||||||
|
);
|
||||||
|
|
||||||
return Ok(m.stamped_value());
|
return Ok(m.stamped_value());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -134,6 +153,8 @@ where
|
||||||
.iter()
|
.iter()
|
||||||
.all(|old_input| !old_input.maybe_changed_since(query, old_memo.verified_at))
|
.all(|old_input| !old_input.maybe_changed_since(query, old_memo.verified_at))
|
||||||
{
|
{
|
||||||
|
debug!("{:?}({:?}): inputs still valid", Q::default(), key);
|
||||||
|
|
||||||
// If none of out inputs have changed since the last time we refreshed
|
// If none of out inputs have changed since the last time we refreshed
|
||||||
// our value, then our value must still be good. We'll just patch
|
// our value, then our value must still be good. We'll just patch
|
||||||
// the verified-at date and re-use it.
|
// the verified-at date and re-use it.
|
||||||
|
@ -155,7 +176,6 @@ where
|
||||||
|
|
||||||
// Query was not previously executed or value is potentially
|
// Query was not previously executed or value is potentially
|
||||||
// stale. Let's execute!
|
// stale. Let's execute!
|
||||||
let descriptor = descriptor();
|
|
||||||
let (value, inputs) = query
|
let (value, inputs) = query
|
||||||
.salsa_runtime()
|
.salsa_runtime()
|
||||||
.execute_query_implementation::<Q>(query, descriptor, key);
|
.execute_query_implementation::<Q>(query, descriptor, key);
|
||||||
|
@ -214,9 +234,16 @@ where
|
||||||
&self,
|
&self,
|
||||||
query: &'q QC,
|
query: &'q QC,
|
||||||
key: &Q::Key,
|
key: &Q::Key,
|
||||||
descriptor: impl FnOnce() -> QC::QueryDescriptor,
|
descriptor: &QC::QueryDescriptor,
|
||||||
) -> Result<Q::Value, CycleDetected> {
|
) -> Result<Q::Value, CycleDetected> {
|
||||||
Ok(self.read(query, key, descriptor)?.value)
|
let StampedValue {
|
||||||
|
value,
|
||||||
|
changed_at: _,
|
||||||
|
} = self.read(query, key, &descriptor)?;
|
||||||
|
|
||||||
|
query.salsa_runtime().report_query_read(descriptor);
|
||||||
|
|
||||||
|
Ok(value)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn maybe_changed_since(
|
fn maybe_changed_since(
|
||||||
|
@ -226,10 +253,19 @@ where
|
||||||
key: &Q::Key,
|
key: &Q::Key,
|
||||||
descriptor: &QC::QueryDescriptor,
|
descriptor: &QC::QueryDescriptor,
|
||||||
) -> bool {
|
) -> bool {
|
||||||
|
let revision_now = query.salsa_runtime().current_revision();
|
||||||
|
|
||||||
|
debug!(
|
||||||
|
"{:?}({:?})::maybe_changed_since(revision={:?}, revision_now={:?})",
|
||||||
|
Q::default(),
|
||||||
|
key,
|
||||||
|
revision,
|
||||||
|
revision_now,
|
||||||
|
);
|
||||||
|
|
||||||
// Check for the case where we have no cache entry, or our cache
|
// Check for the case where we have no cache entry, or our cache
|
||||||
// entry is up to date (common case):
|
// entry is up to date (common case):
|
||||||
{
|
{
|
||||||
let revision_now = query.salsa_runtime().current_revision();
|
|
||||||
let map_read = self.map.read();
|
let map_read = self.map.read();
|
||||||
match map_read.get(key) {
|
match map_read.get(key) {
|
||||||
None | Some(QueryState::InProgress) => return true,
|
None | Some(QueryState::InProgress) => return true,
|
||||||
|
@ -242,7 +278,7 @@ where
|
||||||
}
|
}
|
||||||
|
|
||||||
// Otherwise fall back to the full read to compute the result.
|
// Otherwise fall back to the full read to compute the result.
|
||||||
match self.read(query, key, || descriptor.clone()) {
|
match self.read(query, key, descriptor) {
|
||||||
Ok(v) => v.changed_at > revision,
|
Ok(v) => v.changed_at > revision,
|
||||||
Err(CycleDetected) => true,
|
Err(CycleDetected) => true,
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,5 +1,6 @@
|
||||||
use crate::Query;
|
use crate::Query;
|
||||||
use crate::QueryContext;
|
use crate::QueryContext;
|
||||||
|
use log::debug;
|
||||||
use rustc_hash::FxHasher;
|
use rustc_hash::FxHasher;
|
||||||
use std::cell::RefCell;
|
use std::cell::RefCell;
|
||||||
use std::fmt::Write;
|
use std::fmt::Write;
|
||||||
|
@ -77,24 +78,32 @@ where
|
||||||
|
|
||||||
/// Increments the current revision counter and returns the new value.
|
/// Increments the current revision counter and returns the new value.
|
||||||
crate fn increment_revision(&self) -> Revision {
|
crate fn increment_revision(&self) -> Revision {
|
||||||
Revision {
|
let result = Revision {
|
||||||
generation: 1 + self.shared_state.revision.fetch_add(1, Ordering::SeqCst),
|
generation: 1 + self.shared_state.revision.fetch_add(1, Ordering::SeqCst),
|
||||||
}
|
};
|
||||||
|
|
||||||
|
debug!("increment_revision: incremented to {:?}", result);
|
||||||
|
|
||||||
|
result
|
||||||
}
|
}
|
||||||
|
|
||||||
crate fn execute_query_implementation<Q>(
|
crate fn execute_query_implementation<Q>(
|
||||||
&self,
|
&self,
|
||||||
query: &QC,
|
query: &QC,
|
||||||
descriptor: QC::QueryDescriptor,
|
descriptor: &QC::QueryDescriptor,
|
||||||
key: &Q::Key,
|
key: &Q::Key,
|
||||||
) -> (Q::Value, QueryDescriptorSet<QC>)
|
) -> (Q::Value, QueryDescriptorSet<QC>)
|
||||||
where
|
where
|
||||||
Q: Query<QC>,
|
Q: Query<QC>,
|
||||||
{
|
{
|
||||||
|
debug!("{:?}({:?}): executing query", Q::default(), key);
|
||||||
|
|
||||||
// Push the active query onto the stack.
|
// Push the active query onto the stack.
|
||||||
let push_len = {
|
let push_len = {
|
||||||
let mut local_state = self.local_state.borrow_mut();
|
let mut local_state = self.local_state.borrow_mut();
|
||||||
local_state.query_stack.push(ActiveQuery::new(descriptor));
|
local_state
|
||||||
|
.query_stack
|
||||||
|
.push(ActiveQuery::new(descriptor.clone()));
|
||||||
local_state.query_stack.len()
|
local_state.query_stack.len()
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -122,13 +131,10 @@ where
|
||||||
/// - `descriptor`: the query whose result was read
|
/// - `descriptor`: the query whose result was read
|
||||||
/// - `changed_revision`: the last revision in which the result of that
|
/// - `changed_revision`: the last revision in which the result of that
|
||||||
/// query had changed
|
/// query had changed
|
||||||
crate fn report_query_read(&self, descriptor: QC::QueryDescriptor, changed_revision: Revision) {
|
crate fn report_query_read(&self, descriptor: &QC::QueryDescriptor) {
|
||||||
self.local_state
|
if let Some(top_query) = self.local_state.borrow_mut().query_stack.last_mut() {
|
||||||
.borrow_mut()
|
top_query.add_read(descriptor);
|
||||||
.query_stack
|
}
|
||||||
.last_mut()
|
|
||||||
.unwrap()
|
|
||||||
.add_read(descriptor, changed_revision);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Obviously, this should be user configurable at some point.
|
/// Obviously, this should be user configurable at some point.
|
||||||
|
@ -178,12 +184,12 @@ impl<QC: QueryContext> ActiveQuery<QC> {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn add_read(&mut self, subquery: QC::QueryDescriptor, _changed_revision: Revision) {
|
fn add_read(&mut self, subquery: &QC::QueryDescriptor) {
|
||||||
self.subqueries.insert(subquery);
|
self.subqueries.insert(subquery.clone());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Copy, Clone, Debug, PartialEq, Eq, Hash, PartialOrd, Ord)]
|
#[derive(Copy, Clone, PartialEq, Eq, Hash, PartialOrd, Ord)]
|
||||||
pub struct Revision {
|
pub struct Revision {
|
||||||
generation: u64,
|
generation: u64,
|
||||||
}
|
}
|
||||||
|
@ -194,6 +200,12 @@ impl Revision {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl std::fmt::Debug for Revision {
|
||||||
|
fn fmt(&self, fmt: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
||||||
|
write!(fmt, "R({})", self.generation)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// An insertion-order-preserving set of queries. Used to track the
|
/// An insertion-order-preserving set of queries. Used to track the
|
||||||
/// inputs accessed during query execution.
|
/// inputs accessed during query execution.
|
||||||
crate struct QueryDescriptorSet<QC: QueryContext> {
|
crate struct QueryDescriptorSet<QC: QueryContext> {
|
||||||
|
|
|
@ -4,6 +4,7 @@ use crate::Query;
|
||||||
use crate::QueryContext;
|
use crate::QueryContext;
|
||||||
use crate::QueryStorageOps;
|
use crate::QueryStorageOps;
|
||||||
use crate::QueryTable;
|
use crate::QueryTable;
|
||||||
|
use log::debug;
|
||||||
use parking_lot::{RwLock, RwLockUpgradableReadGuard};
|
use parking_lot::{RwLock, RwLockUpgradableReadGuard};
|
||||||
use rustc_hash::FxHashMap;
|
use rustc_hash::FxHashMap;
|
||||||
use std::any::Any;
|
use std::any::Any;
|
||||||
|
@ -28,25 +29,34 @@ where
|
||||||
&self,
|
&self,
|
||||||
query: &'q QC,
|
query: &'q QC,
|
||||||
key: &Q::Key,
|
key: &Q::Key,
|
||||||
descriptor: impl FnOnce() -> QC::QueryDescriptor,
|
descriptor: &QC::QueryDescriptor,
|
||||||
) -> Result<Q::Value, CycleDetected> {
|
) -> Result<Q::Value, CycleDetected> {
|
||||||
// FIXME: Should we even call `execute_query_implementation`
|
// FIXME: Should we even call `execute_query_implementation`
|
||||||
// here? Or should we just call `Q::execute`, and maybe
|
// here? Or should we just call `Q::execute`, and maybe
|
||||||
// separate out the `push`/`pop` operations.
|
// separate out the `push`/`pop` operations.
|
||||||
let descriptor = descriptor();
|
|
||||||
let (value, _inputs) = query
|
let (value, _inputs) = query
|
||||||
.salsa_runtime()
|
.salsa_runtime()
|
||||||
.execute_query_implementation::<Q>(query, descriptor, key);
|
.execute_query_implementation::<Q>(query, descriptor, key);
|
||||||
|
|
||||||
|
query.salsa_runtime().report_query_read(descriptor);
|
||||||
|
|
||||||
Ok(value)
|
Ok(value)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn maybe_changed_since(
|
fn maybe_changed_since(
|
||||||
&self,
|
&self,
|
||||||
_query: &'q QC,
|
_query: &'q QC,
|
||||||
_revision: Revision,
|
revision: Revision,
|
||||||
_key: &Q::Key,
|
key: &Q::Key,
|
||||||
_descriptor: &QC::QueryDescriptor,
|
_descriptor: &QC::QueryDescriptor,
|
||||||
) -> bool {
|
) -> bool {
|
||||||
|
debug!(
|
||||||
|
"{:?}({:?})::maybe_changed_since(revision={:?}) ==> true (volatile)",
|
||||||
|
Q::default(),
|
||||||
|
key,
|
||||||
|
revision,
|
||||||
|
);
|
||||||
|
|
||||||
true
|
true
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -10,7 +10,7 @@ impl Log {
|
||||||
self.data.borrow_mut().push(text.into());
|
self.data.borrow_mut().push(text.into());
|
||||||
}
|
}
|
||||||
|
|
||||||
crate fn read(&self) -> Vec<String> {
|
crate fn take(&self) -> Vec<String> {
|
||||||
self.data.borrow().clone()
|
std::mem::replace(&mut *self.data.borrow_mut(), vec![])
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -2,18 +2,36 @@
|
||||||
|
|
||||||
use crate::implementation::QueryContextImpl;
|
use crate::implementation::QueryContextImpl;
|
||||||
use crate::queries::CounterContext;
|
use crate::queries::CounterContext;
|
||||||
use crate::queries::QueryContext;
|
use crate::queries::QueryContext as _;
|
||||||
|
use salsa::QueryContext as _;
|
||||||
|
|
||||||
impl QueryContextImpl {
|
impl QueryContextImpl {
|
||||||
fn assert_log(&self, expected_log: &[&str]) {
|
fn assert_log(&self, expected_log: &[&str]) {
|
||||||
let expected_text = format!("{:#?}", expected_log);
|
use difference::{Changeset, Difference};
|
||||||
let actual_text = format!("{:#?}", self.log().read());
|
|
||||||
text_diff::assert_diff(&expected_text, &actual_text, "", 0);
|
let expected_text = &format!("{:#?}", expected_log);
|
||||||
|
let actual_text = &format!("{:#?}", self.log().take());
|
||||||
|
|
||||||
|
if expected_text == actual_text {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
let Changeset { diffs, .. } = Changeset::new(expected_text, actual_text, "\n");
|
||||||
|
|
||||||
|
for i in 0..diffs.len() {
|
||||||
|
match &diffs[i] {
|
||||||
|
Difference::Same(x) => println!(" {}", x),
|
||||||
|
Difference::Add(x) => println!("+{}", x),
|
||||||
|
Difference::Rem(x) => println!("-{}", x),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
panic!("incorrect log results");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn foo() {
|
fn volatile_x2() {
|
||||||
let query = QueryContextImpl::default();
|
let query = QueryContextImpl::default();
|
||||||
|
|
||||||
// Invoking volatile twice will simply execute twice.
|
// Invoking volatile twice will simply execute twice.
|
||||||
|
@ -21,3 +39,21 @@ fn foo() {
|
||||||
query.volatile().of(());
|
query.volatile().of(());
|
||||||
query.assert_log(&["Volatile invoked", "Volatile invoked"]);
|
query.assert_log(&["Volatile invoked", "Volatile invoked"]);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn foo() {
|
||||||
|
env_logger::init();
|
||||||
|
|
||||||
|
let query = QueryContextImpl::default();
|
||||||
|
|
||||||
|
query.memoized2().of(());
|
||||||
|
query.assert_log(&["Memoized2 invoked", "Memoized1 invoked", "Volatile invoked"]);
|
||||||
|
|
||||||
|
query.memoized2().of(());
|
||||||
|
query.assert_log(&[]);
|
||||||
|
|
||||||
|
query.salsa_runtime().next_revision();
|
||||||
|
|
||||||
|
query.memoized2().of(());
|
||||||
|
query.assert_log(&["Memoized1 invoked", "Volatile invoked"]);
|
||||||
|
}
|
||||||
|
|
Loading…
Reference in a new issue