use salsa for new module map

This commit is contained in:
Aleksey Kladov 2018-09-13 22:58:36 +03:00
parent 60fdfec327
commit 8c737255ff
8 changed files with 319 additions and 396 deletions

View file

@ -14,6 +14,7 @@ fst = "0.3.1"
im = "12.0.0" im = "12.0.0"
libsyntax2 = { path = "../libsyntax2" } libsyntax2 = { path = "../libsyntax2" }
libeditor = { path = "../libeditor" } libeditor = { path = "../libeditor" }
salsa = { path = "../salsa" }
[dev-dependencies] [dev-dependencies]
test_utils = { path = "../test_utils" } test_utils = { path = "../test_utils" }

View file

@ -1,306 +0,0 @@
use std::{
hash::{Hash, Hasher},
sync::Arc,
cell::RefCell,
fmt::Debug,
any::Any,
};
use parking_lot::Mutex;
use libsyntax2::{File};
use im;
use {
FileId,
imp::{FileResolverImp},
};
#[derive(Debug)]
pub(crate) struct DbHost {
db: Arc<Db>,
}
impl DbHost {
pub(crate) fn new() -> DbHost {
let db = Db {
file_resolver: FileResolverImp::default(),
files: im::HashMap::new(),
cache: Mutex::new(Cache::new())
};
DbHost { db: Arc::new(db) }
}
pub(crate) fn change_file(&mut self, file_id: FileId, text: Option<String>) {
let db = self.db_mut();
match text {
None => {
db.files.remove(&file_id);
}
Some(text) => {
db.files.insert(file_id, Arc::new(text));
}
}
}
pub(crate) fn set_file_resolver(&mut self, file_resolver: FileResolverImp) {
let db = self.db_mut();
db.file_resolver = file_resolver
}
pub(crate) fn query_ctx(&self) -> QueryCtx {
QueryCtx {
db: Arc::clone(&self.db),
stack: RefCell::new(Vec::new()),
trace: RefCell::new(Vec::new()),
}
}
fn db_mut(&mut self) -> &mut Db {
// NB: this "forks" the database
let db = Arc::make_mut(&mut self.db);
db.cache.get_mut().gen += 1;
db
}
}
type QueryInvocationId = (u32, u64);
type Gen = u64;
type OutputHash = u64;
fn id<Q: Query>(params: &Q::Params) -> QueryInvocationId {
use std::collections::hash_map::DefaultHasher;
let mut hasher = DefaultHasher::new();
params.hash(&mut hasher);
(Q::ID, hasher.finish())
}
fn output_hash<Q: Query>(output: &Q::Output) -> OutputHash {
use std::collections::hash_map::DefaultHasher;
let mut hasher = DefaultHasher::new();
output.hash(&mut hasher);
hasher.finish()
}
#[derive(Debug)]
pub(crate) struct Db {
file_resolver: FileResolverImp,
files: im::HashMap<FileId, Arc<String>>,
cache: Mutex<Cache>,
}
impl Clone for Db {
fn clone(&self) -> Db {
Db {
file_resolver: self.file_resolver.clone(),
files: self.files.clone(),
cache: Mutex::new(Cache::new()),
}
}
}
type QueryDeps = Vec<(QueryInvocationId, Arc<Any>, OutputHash)>;
#[derive(Default, Debug)]
pub(crate) struct Cache {
gen: Gen,
green: im::HashMap<QueryInvocationId, (Gen, OutputHash)>,
deps: im::HashMap<QueryInvocationId, QueryDeps>,
results: im::HashMap<QueryInvocationId, Arc<Any>>,
}
#[allow(type_alias_bounds)]
pub(crate) type QueryCache<Q: Query> = im::HashMap<
<Q as Query>::Params,
<Q as Query>::Output
>;
impl Cache {
fn new() -> Cache {
Default::default()
}
fn get_result<Q: Query>(&self, id: QueryInvocationId) -> Q::Output
where
Q::Output: Clone
{
let res = &self.results[&id];
let res = res.downcast_ref::<Q::Output>().unwrap();
res.clone()
}
}
pub(crate) struct QueryCtx {
db: Arc<Db>,
stack: RefCell<Vec<(QueryInvocationId, QueryDeps)>>,
pub(crate) trace: RefCell<Vec<TraceEvent>>,
}
#[derive(Clone, Copy, Debug)]
pub(crate) struct TraceEvent {
pub(crate) query_id: u32,
pub(crate) kind: TraceEventKind
}
#[derive(Clone, Copy, Debug, PartialEq, Eq)]
pub(crate) enum TraceEventKind {
Start, Evaluating, Finish
}
impl QueryCtx {
pub(crate) fn get<Q: Get>(&self, params: &Q::Params) -> Q::Output {
let me = id::<Q>(params);
self.trace(TraceEvent { query_id: Q::ID, kind: TraceEventKind::Start });
let res = Q::get(self, params);
self.trace(TraceEvent { query_id: Q::ID, kind: TraceEventKind::Finish });
{
let mut stack = self.stack.borrow_mut();
if let Some((_, ref mut deps)) = stack.last_mut() {
let params = Arc::new(params.clone());
deps.push((me, params, output_hash::<Q>(&res)));
}
}
res
}
fn trace(&self, event: TraceEvent) {
self.trace.borrow_mut().push(event)
}
}
pub(crate) trait Query {
const ID: u32;
type Params: Hash + Eq + Debug + Clone + Any + 'static;
type Output: Hash + Debug + Any + 'static;
}
pub(crate) trait Get: Query {
fn get(ctx: &QueryCtx, params: &Self::Params) -> Self::Output;
}
impl<Q: Eval> Get for Q
where
Q::Params: Clone,
Q::Output: Clone,
{
fn get(ctx: &QueryCtx, params: &Self::Params) -> Self::Output {
if let Some(res) = try_reuse::<Q>(ctx, params) {
return res;
}
let me = id::<Q>(params);
ctx.trace(TraceEvent { query_id: Q::ID, kind: TraceEventKind::Evaluating });
ctx.stack.borrow_mut().push((me, Vec::new()));
let res = Self::eval(ctx, params);
let (also_me, deps) = ctx.stack.borrow_mut().pop().unwrap();
assert_eq!(also_me, me);
let mut cache = ctx.db.cache.lock();
cache.deps.insert(me, deps);
let gen = cache.gen;
let output_hash = output_hash::<Q>(&res);
let id = id::<Q>(params);
cache.green.insert(id, (gen, output_hash));
cache.results.insert(me, Arc::new(res.clone()));
res
}
}
fn try_reuse<Q: Eval>(ctx: &QueryCtx, params: &Q::Params) -> Option<Q::Output>
where
Q::Params: Clone,
Q::Output: Clone,
{
let id = id::<Q>(params);
let mut cache = ctx.db.cache.lock();
let curr_gen = cache.gen;
let old_hash = match *cache.green.get(&id)? {
(gen, _) if gen == curr_gen => {
return Some(cache.get_result::<Q>(id));
}
(_, hash) => hash,
};
let deps_are_fresh = cache.deps[&id]
.iter()
.all(|&(dep_id, _, dep_hash)| {
match cache.green.get(&dep_id) {
//TODO: store the value of parameters, and re-execute the query
Some((gen, hash)) if gen == &curr_gen && hash == &dep_hash => true,
_ => false,
}
});
if !deps_are_fresh {
return None;
}
cache.green.insert(id, (curr_gen, old_hash));
Some(cache.get_result::<Q>(id))
}
pub(crate) trait Eval: Query
where
Self::Params: Clone,
Self::Output: Clone,
{
fn eval(ctx: &QueryCtx, params: &Self::Params) -> Self::Output;
}
#[derive(Debug)]
pub(crate) struct DbFiles {
db: Arc<Db>,
}
impl Hash for DbFiles {
fn hash<H: Hasher>(&self, hasher: &mut H) {
self.db.cache.lock().gen.hash(hasher)
}
}
impl DbFiles {
pub(crate) fn iter<'a>(&'a self) -> impl Iterator<Item=FileId> + 'a {
self.db.files.keys().cloned()
}
pub(crate) fn file_resolver(&self) -> FileResolverImp {
self.db.file_resolver.clone()
}
}
pub(crate) enum Files {}
impl Query for Files {
const ID: u32 = 1;
type Params = ();
type Output = DbFiles;
}
impl Get for Files {
fn get(ctx: &QueryCtx, params: &()) -> DbFiles {
let res = DbFiles { db: Arc::clone(&ctx.db) };
let id = id::<Self>(params);
let hash = output_hash::<Self>(&res);
let mut cache = ctx.db.cache.lock();
let gen = cache.gen;
cache.green.insert(id, (gen, hash));
res
}
}
enum FileText {}
impl Query for FileText {
const ID: u32 = 10;
type Params = FileId;
type Output = Arc<String>;
}
impl Get for FileText {
fn get(ctx: &QueryCtx, file_id: &FileId) -> Arc<String> {
let res = ctx.db.files[file_id].clone();
let id = id::<Self>(file_id);
let hash = output_hash::<Self>(&res);
let mut cache = ctx.db.cache.lock();
let gen = cache.gen;
cache.green.insert(id, (gen, hash));
res
}
}
pub(crate) enum FileSyntax {}
impl Query for FileSyntax {
const ID: u32 = 20;
type Params = FileId;
type Output = File;
}
impl Eval for FileSyntax {
fn eval(ctx: &QueryCtx, file_id: &FileId) -> File {
let text = ctx.get::<FileText>(file_id);
File::parse(&text)
}
}

View file

@ -0,0 +1,196 @@
mod queries;
use std::{
hash::{Hash},
sync::Arc,
fmt::Debug,
any::Any,
iter,
};
use im;
use salsa;
use {
FileId,
imp::{FileResolverImp},
};
#[derive(Clone, Default)]
pub(crate) struct State {
pub(crate) resolver: FileResolverImp,
pub(crate) file_map: im::HashMap<FileId, Arc<str>>,
}
type Data = Arc<Any + Send + Sync + 'static>;
pub(crate) struct QueryCtx<'a> {
inner: &'a salsa::QueryCtx<State, Data>
}
pub(crate) struct Db {
inner: salsa::Db<State, Data>
}
struct GroundQuery<T, R> {
id: u16,
f: fn(&State, &T) -> R,
h: fn(&R) -> u64,
}
pub(crate) struct Query<T, R> {
pub(crate) id: u16,
pub(crate) f: fn(QueryCtx, &T) -> R,
}
impl Db {
pub(crate) fn new(state: State) -> Db {
Db { inner: salsa::Db::new(query_config(), state) }
}
pub(crate) fn state(&self) -> &State {
self.inner.ground_data()
}
pub(crate) fn with_state(
&self,
new_state: State,
updated_files: &[FileId],
file_set_changed: bool,
) -> Db {
let mut inv = salsa::Invalidations::new();
if file_set_changed {
inv.invalidate(
salsa::QueryTypeId(queries::FILE_SET.id),
iter::once(salsa::InputFingerprint(hash(&()))),
);
} else {
inv.invalidate(
salsa::QueryTypeId(queries::FILE_SET.id),
iter::empty(),
);
}
inv.invalidate(
salsa::QueryTypeId(queries::FILE_TEXT.id),
updated_files.iter().map(hash).map(salsa::InputFingerprint),
);
Db { inner: self.inner.with_ground_data(new_state, inv) }
}
pub(crate) fn get<T, R>(&self, q: Query<T, R>, params: T) -> (Arc<R>, Vec<u16>)
where
T: Hash + Send + Sync + 'static,
R: Send + Sync + 'static,
{
let query_id = salsa::QueryId(
salsa::QueryTypeId(q.id),
salsa::InputFingerprint(hash(&params)),
);
let params = Arc::new(params);
let (res, events) = self.inner.get(query_id, params);
let res = res.downcast().unwrap();
let events = events.into_iter().map(|it| it.0).collect();
(res, events)
}
}
impl<'a> QueryCtx<'a> {
fn get_g<T, R>(&self, q: GroundQuery<T, R>, params: T) -> Arc<R>
where
T: Hash + Send + Sync + 'static,
R: Send + Sync + 'static,
{
let query_id = salsa::QueryId(
salsa::QueryTypeId(q.id),
salsa::InputFingerprint(hash(&params)),
);
let res = self.inner.get(query_id, Arc::new(params));
res.downcast().unwrap()
}
pub(crate) fn get<T, R>(&self, q: Query<T, R>, params: T) -> Arc<R>
where
T: Hash + Send + Sync + 'static,
R: Send + Sync + 'static,
{
let query_id = salsa::QueryId(
salsa::QueryTypeId(q.id),
salsa::InputFingerprint(hash(&params)),
);
let res = self.inner.get(query_id, Arc::new(params));
res.downcast().unwrap()
}
}
fn query_config() -> salsa::QueryConfig<State, Data> {
let mut res = salsa::QueryConfig::new();
let queries: Vec<SalsaGroundQuery> = vec![
queries::FILE_TEXT.into(),
queries::FILE_SET.into(),
];
for q in queries {
res = res.with_ground_query(q.query_type, q.f)
}
let queries: Vec<SalsaQuery> = vec![
queries::FILE_SYNTAX.into(),
::module_map_db::MODULE_DESCR.into(),
::module_map_db::RESOLVE_SUBMODULE.into(),
::module_map_db::PARENT_MODULE.into(),
];
for q in queries {
res = res.with_query(q.query_type, q.f);
}
res
}
struct SalsaGroundQuery {
query_type: salsa::QueryTypeId,
f: Box<Fn(&State, &Data) -> (Data, salsa::OutputFingerprint) + Send + Sync + 'static>,
}
impl<T, R> From<GroundQuery<T, R>> for SalsaGroundQuery
where
T: Send + Sync + 'static,
R: Send + Sync + 'static,
{
fn from(q: GroundQuery<T, R>) -> SalsaGroundQuery
{
SalsaGroundQuery {
query_type: salsa::QueryTypeId(q.id),
f: Box::new(move |state, data| {
let data: &T = data.downcast_ref().unwrap();
let res = (q.f)(state, data);
let h = (q.h)(&res);
(Arc::new(res), salsa::OutputFingerprint(h))
})
}
}
}
struct SalsaQuery {
query_type: salsa::QueryTypeId,
f: Box<Fn(&salsa::QueryCtx<State, Data>, &Data) -> (Data, salsa::OutputFingerprint) + Send + Sync + 'static>,
}
impl<T, R> From<Query<T, R>> for SalsaQuery
where
T: Hash + Send + Sync + 'static,
R: Hash + Send + Sync + 'static,
{
fn from(q: Query<T, R>) -> SalsaQuery
{
SalsaQuery {
query_type: salsa::QueryTypeId(q.id),
f: Box::new(move |ctx, data| {
let ctx = QueryCtx { inner: ctx };
let data: &T = data.downcast_ref().unwrap();
let res = (q.f)(ctx, data);
let h = hash(&res);
(Arc::new(res), salsa::OutputFingerprint(h))
})
}
}
}
fn hash<T: ::std::hash::Hash>(x: &T) -> u64 {
use std::hash::Hasher;
let mut hasher = ::std::collections::hash_map::DefaultHasher::new();
::std::hash::Hash::hash(x, &mut hasher);
hasher.finish()
}

View file

@ -0,0 +1,43 @@
use std::sync::Arc;
use libsyntax2::{File};
use {
FileId, FileResolverImp,
db::{Query, GroundQuery, QueryCtx, hash},
};
impl<'a> QueryCtx<'a> {
pub(crate) fn file_set(&self) -> Arc<(Vec<FileId>, FileResolverImp)> {
self.get_g(FILE_SET, ())
}
pub(crate) fn file_text(&self, file_id: FileId) -> Arc<str> {
Arc::clone(&*self.get_g(FILE_TEXT, file_id))
}
pub(crate) fn file_syntax(&self, file_id: FileId) -> File {
(&*self.get(FILE_SYNTAX, file_id)).clone()
}
}
pub(super) const FILE_TEXT: GroundQuery<FileId, Arc<str>> = GroundQuery {
id: 10,
f: |state, id| state.file_map[&id].clone(),
h: hash,
};
pub(super) const FILE_SET: GroundQuery<(), (Vec<FileId>, FileResolverImp)> = GroundQuery {
id: 11,
f: |state, &()| {
let files = state.file_map.keys().cloned().collect();
let resolver = state.resolver.clone();
(files, resolver)
},
h: |(files, _)| hash(files),
};
pub(super) const FILE_SYNTAX: Query<FileId, File> = Query {
id: 20,
f: |ctx, file_id: &FileId| {
let text = ctx.file_text(*file_id);
File::parse(&*text)
}
};

View file

@ -10,6 +10,7 @@ extern crate relative_path;
#[macro_use] #[macro_use]
extern crate crossbeam_channel; extern crate crossbeam_channel;
extern crate im; extern crate im;
extern crate salsa;
mod symbol_index; mod symbol_index;
mod module_map; mod module_map;

View file

@ -2,66 +2,55 @@ use std::sync::Arc;
use { use {
FileId, FileId,
db::{ db::{
Query, Eval, QueryCtx, FileSyntax, Files, Query, QueryCtx
Cache, QueryCache,
}, },
module_map::resolve_submodule, module_map::resolve_submodule,
}; };
pub(crate) enum ModuleDescr {} impl<'a> QueryCtx<'a> {
impl Query for ModuleDescr { fn module_descr(&self, file_id: FileId) -> Arc<descr::ModuleDescr> {
const ID: u32 = 30; self.get(MODULE_DESCR, file_id)
type Params = FileId;
type Output = Arc<descr::ModuleDescr>;
} }
fn resolve_submodule(&self, file_id: FileId, submod: descr::Submodule) -> Arc<Vec<FileId>> {
enum ResolveSubmodule {} self.get(RESOLVE_SUBMODULE, (file_id, submod))
impl Query for ResolveSubmodule {
const ID: u32 = 31;
type Params = (FileId, descr::Submodule);
type Output = Arc<Vec<FileId>>;
}
enum ParentModule {}
impl Query for ParentModule {
const ID: u32 = 40;
type Params = FileId;
type Output = Arc<Vec<FileId>>;
}
impl Eval for ModuleDescr {
fn eval(ctx: &QueryCtx, file_id: &FileId) -> Arc<descr::ModuleDescr> {
let file = ctx.get::<FileSyntax>(file_id);
Arc::new(descr::ModuleDescr::new(file.ast()))
} }
} }
impl Eval for ResolveSubmodule { pub(crate) const MODULE_DESCR: Query<FileId, descr::ModuleDescr> = Query {
fn eval(ctx: &QueryCtx, &(file_id, ref submodule): &(FileId, descr::Submodule)) -> Arc<Vec<FileId>> { id: 30,
let files = ctx.get::<Files>(&()); f: |ctx, &file_id| {
let res = resolve_submodule(file_id, &submodule.name, &files.file_resolver()).0; let file = ctx.file_syntax(file_id);
Arc::new(res) descr::ModuleDescr::new(file.ast())
}
} }
};
impl Eval for ParentModule { pub(crate) const RESOLVE_SUBMODULE: Query<(FileId, descr::Submodule), Vec<FileId>> = Query {
fn eval(ctx: &QueryCtx, file_id: &FileId) -> Arc<Vec<FileId>> { id: 31,
let files = ctx.get::<Files>(&()); f: |ctx, params| {
let res = files.iter() let files = ctx.file_set();
.map(|parent_id| (parent_id, ctx.get::<ModuleDescr>(&parent_id))) resolve_submodule(params.0, &params.1.name, &files.1).0
}
};
pub(crate) const PARENT_MODULE: Query<FileId, Vec<FileId>> = Query {
id: 40,
f: |ctx, file_id| {
let files = ctx.file_set();
let res = files.0.iter()
.map(|&parent_id| (parent_id, ctx.module_descr(parent_id)))
.filter(|(parent_id, descr)| { .filter(|(parent_id, descr)| {
descr.submodules.iter() descr.submodules.iter()
.any(|subm| { .any(|subm| {
ctx.get::<ResolveSubmodule>(&(*parent_id, subm.clone())) ctx.resolve_submodule(*parent_id, subm.clone())
.iter() .iter()
.any(|it| it == file_id) .any(|it| it == file_id)
}) })
}) })
.map(|(id, _)| id) .map(|(id, _)| id)
.collect(); .collect();
Arc::new(res) res
}
} }
};
mod descr { mod descr {
use libsyntax2::{ use libsyntax2::{
@ -102,7 +91,7 @@ mod tests {
use im; use im;
use relative_path::{RelativePath, RelativePathBuf}; use relative_path::{RelativePath, RelativePathBuf};
use { use {
db::{Query, DbHost, TraceEventKind}, db::{Query, Db, State},
imp::FileResolverImp, imp::FileResolverImp,
FileId, FileResolver, FileId, FileResolver,
}; };
@ -126,7 +115,7 @@ mod tests {
struct Fixture { struct Fixture {
next_file_id: u32, next_file_id: u32,
fm: im::HashMap<FileId, RelativePathBuf>, fm: im::HashMap<FileId, RelativePathBuf>,
db: DbHost, db: Db,
} }
impl Fixture { impl Fixture {
@ -134,7 +123,7 @@ mod tests {
Fixture { Fixture {
next_file_id: 1, next_file_id: 1,
fm: im::HashMap::new(), fm: im::HashMap::new(),
db: DbHost::new(), db: Db::new(State::default()),
} }
} }
fn add_file(&mut self, path: &str, text: &str) -> FileId { fn add_file(&mut self, path: &str, text: &str) -> FileId {
@ -142,36 +131,39 @@ mod tests {
let file_id = FileId(self.next_file_id); let file_id = FileId(self.next_file_id);
self.next_file_id += 1; self.next_file_id += 1;
self.fm.insert(file_id, RelativePathBuf::from(&path[1..])); self.fm.insert(file_id, RelativePathBuf::from(&path[1..]));
self.db.change_file(file_id, Some(text.to_string())); let mut new_state = self.db.state().clone();
self.db.set_file_resolver(FileResolverImp::new( new_state.file_map.insert(file_id, text.to_string().into_boxed_str().into());
new_state.resolver = FileResolverImp::new(
Arc::new(FileMap(self.fm.clone())) Arc::new(FileMap(self.fm.clone()))
)); );
self.db = self.db.with_state(new_state, &[file_id], true);
file_id file_id
} }
fn remove_file(&mut self, file_id: FileId) { fn remove_file(&mut self, file_id: FileId) {
self.fm.remove(&file_id); self.fm.remove(&file_id);
self.db.change_file(file_id, None); let mut new_state = self.db.state().clone();
self.db.set_file_resolver(FileResolverImp::new( new_state.file_map.remove(&file_id);
new_state.resolver = FileResolverImp::new(
Arc::new(FileMap(self.fm.clone())) Arc::new(FileMap(self.fm.clone()))
)) );
self.db = self.db.with_state(new_state, &[file_id], true);
} }
fn change_file(&mut self, file_id: FileId, new_text: &str) { fn change_file(&mut self, file_id: FileId, new_text: &str) {
self.db.change_file(file_id, Some(new_text.to_string())); let mut new_state = self.db.state().clone();
new_state.file_map.insert(file_id, new_text.to_string().into_boxed_str().into());
self.db = self.db.with_state(new_state, &[file_id], false);
} }
fn check_parent_modules( fn check_parent_modules(
&self, &self,
file_id: FileId, file_id: FileId,
expected: &[FileId], expected: &[FileId],
queries: &[(u32, u64)] queries: &[(u16, u64)]
) { ) {
let ctx = self.db.query_ctx(); let (actual, events) = self.db.get(PARENT_MODULE, file_id);
let actual = ctx.get::<ParentModule>(&file_id);
assert_eq!(actual.as_slice(), expected); assert_eq!(actual.as_slice(), expected);
let mut counts = HashMap::new(); let mut counts = HashMap::new();
ctx.trace.borrow().iter() events.into_iter()
.filter(|event| event.kind == TraceEventKind::Evaluating) .for_each(|event| *counts.entry(event).or_insert(0) += 1);
.for_each(|event| *counts.entry(event.query_id).or_insert(0) += 1);
for &(query_id, expected_count) in queries.iter() { for &(query_id, expected_count) in queries.iter() {
let actual_count = *counts.get(&query_id).unwrap_or(&0); let actual_count = *counts.get(&query_id).unwrap_or(&0);
assert_eq!( assert_eq!(
@ -189,25 +181,25 @@ mod tests {
fn test_parent_module() { fn test_parent_module() {
let mut f = Fixture::new(); let mut f = Fixture::new();
let foo = f.add_file("/foo.rs", ""); let foo = f.add_file("/foo.rs", "");
// f.check_parent_modules(foo, &[], &[(ModuleDescr::ID, 1)]); f.check_parent_modules(foo, &[], &[(MODULE_DESCR.id, 1)]);
let lib = f.add_file("/lib.rs", "mod foo;"); let lib = f.add_file("/lib.rs", "mod foo;");
f.check_parent_modules(foo, &[lib], &[(ModuleDescr::ID, 2)]); f.check_parent_modules(foo, &[lib], &[(MODULE_DESCR.id, 1)]);
f.check_parent_modules(foo, &[lib], &[(ModuleDescr::ID, 0)]); f.check_parent_modules(foo, &[lib], &[(MODULE_DESCR.id, 0)]);
f.change_file(lib, ""); f.change_file(lib, "");
f.check_parent_modules(foo, &[], &[(ModuleDescr::ID, 2)]); f.check_parent_modules(foo, &[], &[(MODULE_DESCR.id, 1)]);
// f.change_file(lib, "mod foo;"); f.change_file(lib, "mod foo;");
// f.check_parent_modules(foo, &[lib], &[(ModuleDescr::ID, 2)]); f.check_parent_modules(foo, &[lib], &[(MODULE_DESCR.id, 1)]);
// f.change_file(lib, "mod bar;"); f.change_file(lib, "mod bar;");
// f.check_parent_modules(foo, &[], &[(ModuleDescr::ID, 2)]); f.check_parent_modules(foo, &[], &[(MODULE_DESCR.id, 1)]);
// f.change_file(lib, "mod foo;"); f.change_file(lib, "mod foo;");
// f.check_parent_modules(foo, &[lib], &[(ModuleDescr::ID, 2)]); f.check_parent_modules(foo, &[lib], &[(MODULE_DESCR.id, 1)]);
// f.remove_file(lib); f.remove_file(lib);
// f.check_parent_modules(foo, &[], &[(ModuleDescr::ID, 1)]); f.check_parent_modules(foo, &[], &[(MODULE_DESCR.id, 0)]);
} }
} }

View file

@ -8,8 +8,8 @@ use std::{
}; };
use parking_lot::Mutex; use parking_lot::Mutex;
type GroundQueryFn<T, D> = fn(&T, &D) -> (D, OutputFingerprint); type GroundQueryFn<T, D> = Box<Fn(&T, &D) -> (D, OutputFingerprint) + Send + Sync + 'static>;
type QueryFn<T, D> = fn(&QueryCtx<T, D>, &D) -> (D, OutputFingerprint); type QueryFn<T, D> = Box<Fn(&QueryCtx<T, D>, &D) -> (D, OutputFingerprint) + Send + Sync + 'static>;
#[derive(Debug)] #[derive(Debug)]
pub struct Db<T, D> { pub struct Db<T, D> {
@ -119,7 +119,7 @@ where
res res
} }
pub fn get_inner( fn get_inner(
&self, &self,
query_id: QueryId, query_id: QueryId,
params: D, params: D,
@ -176,9 +176,9 @@ where
self.executed.borrow_mut().push(query_id.0); self.executed.borrow_mut().push(query_id.0);
self.stack.borrow_mut().push(Vec::new()); self.stack.borrow_mut().push(Vec::new());
let (res, output_fingerprint) = if let Some(f) = self.ground_query_fn_by_type(query_id.0) { let (res, output_fingerprint) = if let Some(f) = self.query_config.ground_fn.get(&query_id.0) {
f(&self.db.ground_data, &params) f(&self.db.ground_data, &params)
} else if let Some(f) = self.query_fn_by_type(query_id.0) { } else if let Some(f) = self.query_config.query_fn.get(&query_id.0) {
f(self, &params) f(self, &params)
} else { } else {
panic!("unknown query type: {:?}", query_id.0); panic!("unknown query type: {:?}", query_id.0);
@ -190,12 +190,6 @@ where
self.db.record(query_id, params, res.clone(), output_fingerprint, deps); self.db.record(query_id, params, res.clone(), output_fingerprint, deps);
(res, output_fingerprint) (res, output_fingerprint)
} }
fn ground_query_fn_by_type(&self, query_type: QueryTypeId) -> Option<GroundQueryFn<T, D>> {
self.query_config.ground_fn.get(&query_type).map(|&it| it)
}
fn query_fn_by_type(&self, query_type: QueryTypeId) -> Option<QueryFn<T, D>> {
self.query_config.query_fn.get(&query_type).map(|&it| it)
}
fn record_dep( fn record_dep(
&self, &self,
query_id: QueryId, query_id: QueryId,
@ -239,7 +233,9 @@ where
query_config: Arc::new(query_config), query_config: Arc::new(query_config),
} }
} }
pub fn ground_data(&self) -> &T {
&self.db.ground_data
}
pub fn with_ground_data( pub fn with_ground_data(
&self, &self,
ground_data: T, ground_data: T,

View file

@ -79,19 +79,19 @@ where
fn mk_queries() -> salsa::QueryConfig<State, Data> { fn mk_queries() -> salsa::QueryConfig<State, Data> {
salsa::QueryConfig::<State, Data>::new() salsa::QueryConfig::<State, Data>::new()
.with_ground_query(GET_TEXT, |state, id| { .with_ground_query(GET_TEXT, Box::new(|state, id| {
mk_ground_query::<u32, String>(state, id, |state, id| state[id].clone()) mk_ground_query::<u32, String>(state, id, |state, id| state[id].clone())
}) }))
.with_ground_query(GET_FILES, |state, id| { .with_ground_query(GET_FILES, Box::new(|state, id| {
mk_ground_query::<(), Vec<u32>>(state, id, |state, &()| state.keys().cloned().collect()) mk_ground_query::<(), Vec<u32>>(state, id, |state, &()| state.keys().cloned().collect())
}) }))
.with_query(FILE_NEWLINES, |query_ctx, id| { .with_query(FILE_NEWLINES, Box::new(|query_ctx, id| {
mk_query(query_ctx, id, |query_ctx, &id| { mk_query(query_ctx, id, |query_ctx, &id| {
let text = query_ctx.get_text(id); let text = query_ctx.get_text(id);
text.lines().count() text.lines().count()
}) })
}) }))
.with_query(TOTAL_NEWLINES, |query_ctx, id| { .with_query(TOTAL_NEWLINES, Box::new(|query_ctx, id| {
mk_query(query_ctx, id, |query_ctx, &()| { mk_query(query_ctx, id, |query_ctx, &()| {
let mut total = 0; let mut total = 0;
for &id in query_ctx.get_files().iter() { for &id in query_ctx.get_files().iter() {
@ -99,7 +99,7 @@ fn mk_queries() -> salsa::QueryConfig<State, Data> {
} }
total total
}) })
}) }))
} }
#[test] #[test]