add a public next_revision method and start on incremental test

This commit is contained in:
Niko Matsakis 2018-09-30 06:59:28 -04:00
parent 1929b26505
commit e134ffcdf4
8 changed files with 158 additions and 0 deletions

View file

@ -13,3 +13,6 @@ derive-new = "0.5.5"
rustc-hash = "1.0"
parking_lot = "0.6.4"
indexmap = "1.0.1"
[dev-dependencies]
text-diff = "0.4.0"

View file

@ -0,0 +1,14 @@
use std::cell::Cell;
#[derive(Default)]
crate struct Counter {
value: Cell<usize>,
}
impl Counter {
crate fn increment(&self) -> usize {
let v = self.value.get();
self.value.set(v + 1);
v
}
}

View file

@ -0,0 +1,36 @@
use crate::counter::Counter;
use crate::log::Log;
use crate::queries;
#[derive(Default)]
pub struct QueryContextImpl {
runtime: salsa::runtime::Runtime<QueryContextImpl>,
clock: Counter,
log: Log,
}
salsa::query_context_storage! {
pub struct QueryContextImplStorage for QueryContextImpl {
impl queries::QueryContext {
fn memoized2() for queries::Memoized2;
fn memoized1() for queries::Memoized1;
fn volatile() for queries::Volatile;
}
}
}
impl queries::CounterContext for QueryContextImpl {
fn clock(&self) -> &Counter {
&self.clock
}
fn log(&self) -> &Log {
&self.log
}
}
impl salsa::QueryContext for QueryContextImpl {
fn salsa_runtime(&self) -> &salsa::runtime::Runtime<QueryContextImpl> {
&self.runtime
}
}

View file

@ -0,0 +1,16 @@
use std::cell::RefCell;
#[derive(Default)]
crate struct Log {
data: RefCell<Vec<String>>,
}
impl Log {
crate fn add(&self, text: impl Into<String>) {
self.data.borrow_mut().push(text.into());
}
crate fn read(&self) -> Vec<String> {
self.data.borrow().clone()
}
}

View file

@ -0,0 +1,10 @@
#![feature(crate_visibility_modifier)]
#![feature(underscore_imports)]
mod counter;
mod implementation;
mod log;
mod queries;
mod tests;
fn main() {}

View file

@ -0,0 +1,38 @@
use crate::counter::Counter;
use crate::log::Log;
crate trait CounterContext: salsa::QueryContext {
fn clock(&self) -> &Counter;
fn log(&self) -> &Log;
}
crate trait QueryContext: CounterContext {
salsa::query_prototype! {
fn memoized2() for Memoized2;
fn memoized1() for Memoized1;
fn volatile() for Volatile;
}
}
salsa::query_definition! {
crate Memoized2(query: &impl QueryContext, (): ()) -> usize {
query.log().add("Memoized2 invoked");
query.memoized1().of(())
}
}
salsa::query_definition! {
crate Memoized1(query: &impl QueryContext, (): ()) -> usize {
query.log().add("Memoized1 invoked");
let v = query.volatile().of(());
v / 3
}
}
salsa::query_definition! {
#[storage(volatile)]
crate Volatile(query: &impl QueryContext, (): ()) -> usize {
query.log().add("Volatile invoked");
query.clock().increment()
}
}

View file

@ -0,0 +1,23 @@
#![cfg(test)]
use crate::implementation::QueryContextImpl;
use crate::queries::CounterContext;
use crate::queries::QueryContext;
impl QueryContextImpl {
fn assert_log(&self, expected_log: &[&str]) {
let expected_text = format!("{:#?}", expected_log);
let actual_text = format!("{:#?}", self.log().read());
text_diff::assert_diff(&expected_text, &actual_text, "", 0);
}
}
#[test]
fn foo() {
let query = QueryContextImpl::default();
// Invoking volatile twice will simply execute twice.
query.volatile().of(());
query.volatile().of(());
query.assert_log(&["Volatile invoked", "Volatile invoked"]);
}

View file

@ -50,6 +50,24 @@ where
&self.shared_state.storage
}
/// Indicates that some input to the system has changed and hence
/// that memoized values **may** be invalidated. This cannot be
/// invoked while query computation is in progress.
///
/// As a user of the system, you would not normally invoke this
/// method directly. Instead, you would use "input" queries and
/// invoke their `set` method. But it can be useful if you have a
/// "volatile" input that you must poll from time to time; in that
/// case, you can wrap the input with a "no-storage" query and
/// invoke this method from time to time.
pub fn next_revision(&self) {
if !self.local_state.borrow().query_stack.is_empty() {
panic!("next_revision invoked during a query computation");
}
self.increment_revision();
}
/// Read current value of the revision counter.
crate fn current_revision(&self) -> Revision {
Revision {