mirror of
https://github.com/rust-lang/rust-analyzer
synced 2025-01-27 12:25:05 +00:00
use salsa for new module map
This commit is contained in:
parent
60fdfec327
commit
8c737255ff
8 changed files with 319 additions and 396 deletions
|
@ -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" }
|
||||||
|
|
|
@ -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)
|
|
||||||
}
|
|
||||||
}
|
|
196
crates/libanalysis/src/db/mod.rs
Normal file
196
crates/libanalysis/src/db/mod.rs
Normal 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(¶ms)),
|
||||||
|
);
|
||||||
|
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(¶ms)),
|
||||||
|
);
|
||||||
|
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(¶ms)),
|
||||||
|
);
|
||||||
|
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()
|
||||||
|
}
|
43
crates/libanalysis/src/db/queries.rs
Normal file
43
crates/libanalysis/src/db/queries.rs
Normal 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)
|
||||||
|
}
|
||||||
|
};
|
|
@ -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;
|
||||||
|
|
|
@ -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, ¶ms.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)]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -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, ¶ms)
|
f(&self.db.ground_data, ¶ms)
|
||||||
} 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, ¶ms)
|
f(self, ¶ms)
|
||||||
} 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,
|
||||||
|
|
|
@ -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]
|
||||||
|
|
Loading…
Reference in a new issue