mirror of
https://github.com/rust-lang/rust-analyzer
synced 2024-12-26 04:53:34 +00:00
start salsa migration
This commit is contained in:
parent
93d77e9b22
commit
d8aee31a60
8 changed files with 226 additions and 105 deletions
8
Cargo.lock
generated
8
Cargo.lock
generated
|
@ -842,14 +842,6 @@ name = "safemem"
|
||||||
version = "0.3.0"
|
version = "0.3.0"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "salsa"
|
|
||||||
version = "0.1.0"
|
|
||||||
dependencies = [
|
|
||||||
"im 12.2.0 (registry+https://github.com/rust-lang/crates.io-index)",
|
|
||||||
"parking_lot 0.6.4 (registry+https://github.com/rust-lang/crates.io-index)",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "salsa"
|
name = "salsa"
|
||||||
version = "0.4.1"
|
version = "0.4.1"
|
||||||
|
|
|
@ -1,85 +1,193 @@
|
||||||
mod imp;
|
|
||||||
|
|
||||||
use std::{
|
use std::{
|
||||||
|
fmt,
|
||||||
sync::Arc,
|
sync::Arc,
|
||||||
|
hash::{Hash, Hasher},
|
||||||
|
collections::HashSet,
|
||||||
};
|
};
|
||||||
use im;
|
|
||||||
use salsa;
|
use salsa;
|
||||||
use crate::{FileId, imp::FileResolverImp};
|
use ra_syntax::File;
|
||||||
|
use ra_editor::{LineIndex};
|
||||||
|
use crate::{
|
||||||
|
symbol_index::SymbolIndex,
|
||||||
|
FileId, FileResolverImp
|
||||||
|
};
|
||||||
|
|
||||||
#[derive(Debug, Default, Clone)]
|
#[derive(Default)]
|
||||||
pub(crate) struct State {
|
pub(crate) struct RootDatabase {
|
||||||
pub(crate) file_map: im::HashMap<FileId, Arc<String>>,
|
runtime: salsa::runtime::Runtime<RootDatabase>,
|
||||||
pub(crate) file_resolver: FileResolverImp
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug)]
|
impl fmt::Debug for RootDatabase {
|
||||||
pub(crate) struct Db {
|
fn fmt(&self, fmt: &mut fmt::Formatter) -> fmt::Result {
|
||||||
imp: imp::Db,
|
fmt.write_str("RootDatabase { ... }")
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Clone, Copy)]
|
|
||||||
pub(crate) struct QueryCtx<'a> {
|
|
||||||
imp: &'a salsa::QueryCtx<State, imp::Data>,
|
|
||||||
}
|
|
||||||
|
|
||||||
pub(crate) struct Query<T, R>(pub(crate) u16, pub(crate) fn(QueryCtx, &T) -> R);
|
|
||||||
|
|
||||||
pub(crate) struct QueryRegistry {
|
|
||||||
imp: imp::QueryRegistry,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Default for Db {
|
|
||||||
fn default() -> Db {
|
|
||||||
Db::new()
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Db {
|
impl salsa::Database for RootDatabase {
|
||||||
pub(crate) fn new() -> Db {
|
fn salsa_runtime(&self) -> &salsa::runtime::Runtime<RootDatabase> {
|
||||||
let reg = QueryRegistry::new();
|
&self.runtime
|
||||||
Db { imp: imp::Db::new(reg.imp) }
|
|
||||||
}
|
|
||||||
pub(crate) fn state(&self) -> &State {
|
|
||||||
self.imp.imp.ground_data()
|
|
||||||
}
|
|
||||||
pub(crate) fn with_changes(&self, new_state: State, changed_files: &[FileId], resolver_changed: bool) -> Db {
|
|
||||||
Db { imp: self.imp.with_changes(new_state, changed_files, resolver_changed) }
|
|
||||||
}
|
|
||||||
pub(crate) fn make_query<F: FnOnce(QueryCtx) -> R, R>(&self, f: F) -> R {
|
|
||||||
let ctx = QueryCtx { imp: &self.imp.imp.query_ctx() };
|
|
||||||
f(ctx)
|
|
||||||
}
|
|
||||||
#[allow(unused)]
|
|
||||||
pub(crate) fn trace_query<F: FnOnce(QueryCtx) -> R, R>(&self, f: F) -> (R, Vec<&'static str>) {
|
|
||||||
let ctx = QueryCtx { imp: &self.imp.imp.query_ctx() };
|
|
||||||
let res = f(ctx);
|
|
||||||
let trace = self.imp.extract_trace(ctx.imp);
|
|
||||||
(res, trace)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<'a> QueryCtx<'a> {
|
salsa::database_storage! {
|
||||||
pub(crate) fn get<Q: imp::EvalQuery>(&self, q: Q, params: Q::Params) -> Arc<Q::Output> {
|
pub(crate) struct RootDatabaseStorage for RootDatabase {
|
||||||
q.get(self, params)
|
impl FilesDatabase {
|
||||||
|
fn file_text() for FileTextQuery;
|
||||||
|
fn file_set() for FileSetQuery;
|
||||||
|
}
|
||||||
|
impl SyntaxDatabase {
|
||||||
|
fn file_syntax() for FileSyntaxQuery;
|
||||||
|
fn file_lines() for FileLinesQuery;
|
||||||
|
fn file_symbols() for FileSymbolsQuery;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub(crate) fn file_text(ctx: QueryCtx, file_id: FileId) -> Arc<String> {
|
salsa::query_group! {
|
||||||
imp::file_text(ctx, file_id)
|
pub(crate) trait FilesDatabase: salsa::Database {
|
||||||
|
fn file_text(file_id: FileId) -> Arc<String> {
|
||||||
|
type FileTextQuery;
|
||||||
|
storage input;
|
||||||
|
}
|
||||||
|
fn file_set(key: ()) -> Arc<FileSet> {
|
||||||
|
type FileSetQuery;
|
||||||
|
storage input;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub(crate) fn file_set(ctx: QueryCtx) -> Arc<(Vec<FileId>, FileResolverImp)> {
|
#[derive(Default, Debug)]
|
||||||
imp::file_set(ctx)
|
pub(crate) struct FileSet {
|
||||||
|
pub(crate) files: HashSet<FileId>,
|
||||||
|
pub(crate) resolver: FileResolverImp,
|
||||||
}
|
}
|
||||||
impl QueryRegistry {
|
|
||||||
fn new() -> QueryRegistry {
|
impl PartialEq for FileSet {
|
||||||
let mut reg = QueryRegistry { imp: imp::QueryRegistry::new() };
|
fn eq(&self, other: &FileSet) -> bool {
|
||||||
crate::queries::register_queries(&mut reg);
|
self.files == other.files
|
||||||
crate::module_map::register_queries(&mut reg);
|
|
||||||
reg
|
|
||||||
}
|
|
||||||
pub(crate) fn add<Q: imp::EvalQuery>(&mut self, q: Q, name: &'static str) {
|
|
||||||
self.imp.add(q, name)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl Eq for FileSet {
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Hash for FileSet {
|
||||||
|
fn hash<H: Hasher>(&self, hasher: &mut H) {
|
||||||
|
let mut files = self.files.iter().cloned().collect::<Vec<_>>();
|
||||||
|
files.sort();
|
||||||
|
files.hash(hasher);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
salsa::query_group! {
|
||||||
|
pub(crate) trait SyntaxDatabase: FilesDatabase {
|
||||||
|
fn file_syntax(file_id: FileId) -> File {
|
||||||
|
type FileSyntaxQuery;
|
||||||
|
}
|
||||||
|
fn file_lines(file_id: FileId) -> Arc<LineIndex> {
|
||||||
|
type FileLinesQuery;
|
||||||
|
}
|
||||||
|
fn file_symbols(file_id: FileId) -> Arc<SymbolIndex> {
|
||||||
|
type FileSymbolsQuery;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn file_syntax(db: &impl SyntaxDatabase, file_id: FileId) -> File {
|
||||||
|
let text = db.file_text(file_id);
|
||||||
|
File::parse(&*text)
|
||||||
|
}
|
||||||
|
fn file_lines(db: &impl SyntaxDatabase, file_id: FileId) -> Arc<LineIndex> {
|
||||||
|
let text = db.file_text(file_id);
|
||||||
|
Arc::new(LineIndex::new(&*text))
|
||||||
|
}
|
||||||
|
fn file_symbols(db: &impl SyntaxDatabase, file_id: FileId) -> Arc<SymbolIndex> {
|
||||||
|
let syntax = db.file_syntax(file_id);
|
||||||
|
Arc::new(SymbolIndex::for_file(file_id, syntax))
|
||||||
|
}
|
||||||
|
|
||||||
|
// mod imp;
|
||||||
|
|
||||||
|
// use std::{
|
||||||
|
// sync::Arc,
|
||||||
|
// };
|
||||||
|
// use im;
|
||||||
|
// use salsa;
|
||||||
|
// use {FileId, imp::FileResolverImp};
|
||||||
|
|
||||||
|
// #[derive(Debug, Default, Clone)]
|
||||||
|
// pub(crate) struct State {
|
||||||
|
// pub(crate) file_map: im::HashMap<FileId, Arc<String>>,
|
||||||
|
// pub(crate) file_resolver: FileResolverImp
|
||||||
|
// }
|
||||||
|
|
||||||
|
// #[derive(Debug)]
|
||||||
|
// pub(crate) struct Db {
|
||||||
|
// imp: imp::Db,
|
||||||
|
// }
|
||||||
|
|
||||||
|
// #[derive(Clone, Copy)]
|
||||||
|
// pub(crate) struct QueryCtx<'a> {
|
||||||
|
// imp: &'a salsa::QueryCtx<State, imp::Data>,
|
||||||
|
// }
|
||||||
|
|
||||||
|
// pub(crate) struct Query<T, R>(pub(crate) u16, pub(crate) fn(QueryCtx, &T) -> R);
|
||||||
|
|
||||||
|
// pub(crate) struct QueryRegistry {
|
||||||
|
// imp: imp::QueryRegistry,
|
||||||
|
// }
|
||||||
|
|
||||||
|
// impl Default for Db {
|
||||||
|
// fn default() -> Db {
|
||||||
|
// Db::new()
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
|
||||||
|
// impl Db {
|
||||||
|
// pub(crate) fn new() -> Db {
|
||||||
|
// let reg = QueryRegistry::new();
|
||||||
|
// Db { imp: imp::Db::new(reg.imp) }
|
||||||
|
// }
|
||||||
|
// pub(crate) fn state(&self) -> &State {
|
||||||
|
// self.imp.imp.ground_data()
|
||||||
|
// }
|
||||||
|
// pub(crate) fn with_changes(&self, new_state: State, changed_files: &[FileId], resolver_changed: bool) -> Db {
|
||||||
|
// Db { imp: self.imp.with_changes(new_state, changed_files, resolver_changed) }
|
||||||
|
// }
|
||||||
|
// pub(crate) fn make_query<F: FnOnce(QueryCtx) -> R, R>(&self, f: F) -> R {
|
||||||
|
// let ctx = QueryCtx { imp: &self.imp.imp.query_ctx() };
|
||||||
|
// f(ctx)
|
||||||
|
// }
|
||||||
|
// #[allow(unused)]
|
||||||
|
// pub(crate) fn trace_query<F: FnOnce(QueryCtx) -> R, R>(&self, f: F) -> (R, Vec<&'static str>) {
|
||||||
|
// let ctx = QueryCtx { imp: &self.imp.imp.query_ctx() };
|
||||||
|
// let res = f(ctx);
|
||||||
|
// let trace = self.imp.extract_trace(ctx.imp);
|
||||||
|
// (res, trace)
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
|
||||||
|
// impl<'a> QueryCtx<'a> {
|
||||||
|
// pub(crate) fn get<Q: imp::EvalQuery>(&self, q: Q, params: Q::Params) -> Arc<Q::Output> {
|
||||||
|
// q.get(self, params)
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
|
||||||
|
// pub(crate) fn file_text(ctx: QueryCtx, file_id: FileId) -> Arc<String> {
|
||||||
|
// imp::file_text(ctx, file_id)
|
||||||
|
// }
|
||||||
|
|
||||||
|
// pub(crate) fn file_set(ctx: QueryCtx) -> Arc<(Vec<FileId>, FileResolverImp)> {
|
||||||
|
// imp::file_set(ctx)
|
||||||
|
// }
|
||||||
|
// impl QueryRegistry {
|
||||||
|
// fn new() -> QueryRegistry {
|
||||||
|
// let mut reg = QueryRegistry { imp: imp::QueryRegistry::new() };
|
||||||
|
// ::queries::register_queries(&mut reg);
|
||||||
|
// ::module_map::register_queries(&mut reg);
|
||||||
|
// reg
|
||||||
|
// }
|
||||||
|
// pub(crate) fn add<Q: imp::EvalQuery>(&mut self, q: Q, name: &'static str) {
|
||||||
|
// self.imp.add(q, name)
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
|
|
@ -14,12 +14,12 @@ extern crate salsa;
|
||||||
extern crate rustc_hash;
|
extern crate rustc_hash;
|
||||||
|
|
||||||
mod symbol_index;
|
mod symbol_index;
|
||||||
mod module_map;
|
// mod module_map;
|
||||||
mod imp;
|
mod imp;
|
||||||
mod job;
|
mod job;
|
||||||
mod roots;
|
mod roots;
|
||||||
mod db;
|
mod db;
|
||||||
mod queries;
|
// mod queries;
|
||||||
mod descriptors;
|
mod descriptors;
|
||||||
|
|
||||||
use std::{
|
use std::{
|
||||||
|
|
|
@ -1,11 +1,13 @@
|
||||||
use std::{
|
use std::{
|
||||||
|
collections::{HashMap, HashSet},
|
||||||
sync::Arc,
|
sync::Arc,
|
||||||
panic,
|
panic,
|
||||||
};
|
};
|
||||||
|
use parking_lot::RwLock;
|
||||||
|
|
||||||
use once_cell::sync::OnceCell;
|
use once_cell::sync::OnceCell;
|
||||||
use rayon::prelude::*;
|
use rayon::prelude::*;
|
||||||
use rustc_hash::FxHashMap;
|
use salsa::Database;
|
||||||
use ra_editor::LineIndex;
|
use ra_editor::LineIndex;
|
||||||
use ra_syntax::File;
|
use ra_syntax::File;
|
||||||
|
|
||||||
|
@ -14,7 +16,7 @@ use crate::{
|
||||||
imp::FileResolverImp,
|
imp::FileResolverImp,
|
||||||
symbol_index::SymbolIndex,
|
symbol_index::SymbolIndex,
|
||||||
descriptors::{ModuleDescriptor, ModuleTreeDescriptor},
|
descriptors::{ModuleDescriptor, ModuleTreeDescriptor},
|
||||||
db::Db,
|
db::{self, FilesDatabase, SyntaxDatabase}
|
||||||
};
|
};
|
||||||
|
|
||||||
pub(crate) trait SourceRoot {
|
pub(crate) trait SourceRoot {
|
||||||
|
@ -25,9 +27,9 @@ pub(crate) trait SourceRoot {
|
||||||
fn symbols(&self, acc: &mut Vec<Arc<SymbolIndex>>);
|
fn symbols(&self, acc: &mut Vec<Arc<SymbolIndex>>);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Default, Debug)]
|
#[derive(Default, Debug, Clone)]
|
||||||
pub(crate) struct WritableSourceRoot {
|
pub(crate) struct WritableSourceRoot {
|
||||||
db: Db,
|
db: Arc<RwLock<db::RootDatabase>>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl WritableSourceRoot {
|
impl WritableSourceRoot {
|
||||||
|
@ -36,51 +38,61 @@ impl WritableSourceRoot {
|
||||||
changes: &mut dyn Iterator<Item=(FileId, Option<String>)>,
|
changes: &mut dyn Iterator<Item=(FileId, Option<String>)>,
|
||||||
file_resolver: Option<FileResolverImp>,
|
file_resolver: Option<FileResolverImp>,
|
||||||
) -> WritableSourceRoot {
|
) -> WritableSourceRoot {
|
||||||
let resolver_changed = file_resolver.is_some();
|
let db = self.db.write();
|
||||||
let mut changed_files = Vec::new();
|
let mut changed = HashSet::new();
|
||||||
let mut new_state = self.db.state().clone();
|
let mut removed = HashSet::new();
|
||||||
|
|
||||||
for (file_id, text) in changes {
|
for (file_id, text) in changes {
|
||||||
changed_files.push(file_id);
|
|
||||||
match text {
|
match text {
|
||||||
Some(text) => {
|
|
||||||
new_state.file_map.insert(file_id, Arc::new(text));
|
|
||||||
},
|
|
||||||
None => {
|
None => {
|
||||||
new_state.file_map.remove(&file_id);
|
removed.insert(file_id);
|
||||||
|
}
|
||||||
|
Some(text) => {
|
||||||
|
db.query(db::FileTextQuery)
|
||||||
|
.set(file_id, Arc::new(text));
|
||||||
|
changed.insert(file_id);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if let Some(file_resolver) = file_resolver {
|
if let Some(resolver) = file_resolver {
|
||||||
new_state.file_resolver = file_resolver
|
let mut files: HashSet<FileId> = db.file_set(())
|
||||||
}
|
.files
|
||||||
WritableSourceRoot {
|
.clone();
|
||||||
db: self.db.with_changes(new_state, &changed_files, resolver_changed)
|
for file_id in removed {
|
||||||
|
files.remove(&file_id);
|
||||||
|
}
|
||||||
|
files.extend(changed);
|
||||||
|
db.query(db::FileSetQuery)
|
||||||
|
.set((), Arc::new(db::FileSet { files, resolver }))
|
||||||
}
|
}
|
||||||
|
// TODO: reconcile sasla's API with our needs
|
||||||
|
// https://github.com/salsa-rs/salsa/issues/12
|
||||||
|
self.clone()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl SourceRoot for WritableSourceRoot {
|
impl SourceRoot for WritableSourceRoot {
|
||||||
fn module_tree(&self) -> Arc<ModuleTreeDescriptor> {
|
fn module_tree(&self) -> Arc<ModuleTreeDescriptor> {
|
||||||
self.db.make_query(crate::module_map::module_tree)
|
unimplemented!()
|
||||||
|
//self.db.make_query(::module_map::module_tree)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn contains(&self, file_id: FileId) -> bool {
|
fn contains(&self, file_id: FileId) -> bool {
|
||||||
self.db.state().file_map.contains_key(&file_id)
|
self.db.read().file_set(()).files.contains(&file_id)
|
||||||
}
|
}
|
||||||
fn lines(&self, file_id: FileId) -> Arc<LineIndex> {
|
fn lines(&self, file_id: FileId) -> Arc<LineIndex> {
|
||||||
self.db.make_query(|ctx| crate::queries::file_lines(ctx, file_id))
|
self.db.read().file_lines(file_id)
|
||||||
}
|
}
|
||||||
fn syntax(&self, file_id: FileId) -> File {
|
fn syntax(&self, file_id: FileId) -> File {
|
||||||
self.db.make_query(|ctx| crate::queries::file_syntax(ctx, file_id))
|
self.db.read().file_syntax(file_id)
|
||||||
}
|
}
|
||||||
fn symbols<'a>(&'a self, acc: &mut Vec<Arc<SymbolIndex>>) {
|
fn symbols<'a>(&'a self, acc: &mut Vec<Arc<SymbolIndex>>) {
|
||||||
self.db.make_query(|ctx| {
|
let db = self.db.read();
|
||||||
let file_set = crate::queries::file_set(ctx);
|
let symbols = db.file_set(());
|
||||||
let syms = file_set.0.iter()
|
let symbols = symbols
|
||||||
.map(|file_id| crate::queries::file_symbols(ctx, *file_id));
|
.files
|
||||||
acc.extend(syms);
|
.iter()
|
||||||
});
|
.map(|&file_id| db.file_symbols(file_id));
|
||||||
|
acc.extend(symbols);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -17,6 +17,15 @@ pub(crate) struct SymbolIndex {
|
||||||
map: fst::Map,
|
map: fst::Map,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl PartialEq for SymbolIndex {
|
||||||
|
fn eq(&self, other: &SymbolIndex) -> bool {
|
||||||
|
self.symbols == other.symbols
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Eq for SymbolIndex {
|
||||||
|
}
|
||||||
|
|
||||||
impl Hash for SymbolIndex {
|
impl Hash for SymbolIndex {
|
||||||
fn hash<H: Hasher>(&self, hasher: &mut H) {
|
fn hash<H: Hasher>(&self, hasher: &mut H) {
|
||||||
self.symbols.hash(hasher)
|
self.symbols.hash(hasher)
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
use superslice::Ext;
|
use superslice::Ext;
|
||||||
use crate::TextUnit;
|
use crate::TextUnit;
|
||||||
|
|
||||||
#[derive(Clone, Debug, Hash)]
|
#[derive(Clone, Debug, Hash, PartialEq, Eq)]
|
||||||
pub struct LineIndex {
|
pub struct LineIndex {
|
||||||
newlines: Vec<TextUnit>,
|
newlines: Vec<TextUnit>,
|
||||||
}
|
}
|
||||||
|
|
|
@ -17,7 +17,7 @@ pub struct StructureNode {
|
||||||
pub kind: SyntaxKind,
|
pub kind: SyntaxKind,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, Clone, Hash)]
|
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
|
||||||
pub struct FileSymbol {
|
pub struct FileSymbol {
|
||||||
pub name: SmolStr,
|
pub name: SmolStr,
|
||||||
pub node_range: TextRange,
|
pub node_range: TextRange,
|
||||||
|
|
|
@ -59,7 +59,7 @@ use crate::{
|
||||||
yellow::{GreenNode},
|
yellow::{GreenNode},
|
||||||
};
|
};
|
||||||
|
|
||||||
#[derive(Clone, Debug, Hash)]
|
#[derive(Clone, Debug, Hash, PartialEq, Eq)]
|
||||||
pub struct File {
|
pub struct File {
|
||||||
root: SyntaxNode
|
root: SyntaxNode
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue