Add Loc2IdMap

This commit is contained in:
Aleksey Kladov 2018-11-18 15:44:24 +03:00
parent f1d8558cd5
commit cb22a799d6
6 changed files with 84 additions and 0 deletions

1
Cargo.lock generated
View file

@ -600,6 +600,7 @@ version = "0.1.0"
dependencies = [ dependencies = [
"fst 0.3.2 (registry+https://github.com/rust-lang/crates.io-index)", "fst 0.3.2 (registry+https://github.com/rust-lang/crates.io-index)",
"log 0.4.6 (registry+https://github.com/rust-lang/crates.io-index)", "log 0.4.6 (registry+https://github.com/rust-lang/crates.io-index)",
"parking_lot 0.6.4 (registry+https://github.com/rust-lang/crates.io-index)",
"ra_editor 0.1.0", "ra_editor 0.1.0",
"ra_syntax 0.1.0", "ra_syntax 0.1.0",
"rayon 1.0.3 (registry+https://github.com/rust-lang/crates.io-index)", "rayon 1.0.3 (registry+https://github.com/rust-lang/crates.io-index)",

View file

@ -11,6 +11,7 @@ rayon = "1.0.2"
fst = "0.3.1" fst = "0.3.1"
salsa = "0.8.0" salsa = "0.8.0"
rustc-hash = "1.0" rustc-hash = "1.0"
parking_lot = "0.6.4"
ra_syntax = { path = "../ra_syntax" } ra_syntax = { path = "../ra_syntax" }
ra_editor = { path = "../ra_editor" } ra_editor = { path = "../ra_editor" }
test_utils = { path = "../test_utils" } test_utils = { path = "../test_utils" }

View file

@ -1,5 +1,7 @@
use std::sync::Arc; use std::sync::Arc;
use parking_lot::Mutex;
use ra_editor::LineIndex; use ra_editor::LineIndex;
use ra_syntax::{SourceFileNode, SyntaxNode}; use ra_syntax::{SourceFileNode, SyntaxNode};
use salsa::{self, Database}; use salsa::{self, Database};
@ -9,15 +11,19 @@ use crate::{
descriptors::{ descriptors::{
DescriptorDatabase, FnScopesQuery, FnSyntaxQuery, ModuleScopeQuery, ModuleTreeQuery, DescriptorDatabase, FnScopesQuery, FnSyntaxQuery, ModuleScopeQuery, ModuleTreeQuery,
SubmodulesQuery, SubmodulesQuery,
module::{ModuleSource, ModuleId},
}, },
input::SourceRootId,
symbol_index::SymbolIndex, symbol_index::SymbolIndex,
syntax_ptr::SyntaxPtr, syntax_ptr::SyntaxPtr,
loc2id::Loc2IdMap,
Cancelable, Canceled, FileId, Cancelable, Canceled, FileId,
}; };
#[derive(Debug)] #[derive(Debug)]
pub(crate) struct RootDatabase { pub(crate) struct RootDatabase {
runtime: salsa::Runtime<RootDatabase>, runtime: salsa::Runtime<RootDatabase>,
loc2id: Arc<Mutex<Loc2IdMap<(SourceRootId, ModuleSource), ModuleId>>>,
} }
impl salsa::Database for RootDatabase { impl salsa::Database for RootDatabase {
@ -30,6 +36,7 @@ impl Default for RootDatabase {
fn default() -> RootDatabase { fn default() -> RootDatabase {
let mut db = RootDatabase { let mut db = RootDatabase {
runtime: Default::default(), runtime: Default::default(),
loc2id: Default::default(),
}; };
db.query_mut(crate::input::SourceRootQuery) db.query_mut(crate::input::SourceRootQuery)
.set(crate::input::WORKSPACE, Default::default()); .set(crate::input::WORKSPACE, Default::default());
@ -53,6 +60,7 @@ impl salsa::ParallelDatabase for RootDatabase {
fn snapshot(&self) -> salsa::Snapshot<RootDatabase> { fn snapshot(&self) -> salsa::Snapshot<RootDatabase> {
salsa::Snapshot::new(RootDatabase { salsa::Snapshot::new(RootDatabase {
runtime: self.runtime.snapshot(self), runtime: self.runtime.snapshot(self),
loc2id: Arc::clone(&self.loc2id),
}) })
} }
} }

View file

@ -58,6 +58,15 @@ enum ModuleSourceNode {
#[derive(Clone, Copy, PartialEq, Eq, Hash, PartialOrd, Ord, Debug)] #[derive(Clone, Copy, PartialEq, Eq, Hash, PartialOrd, Ord, Debug)]
pub(crate) struct ModuleId(u32); pub(crate) struct ModuleId(u32);
impl crate::loc2id::NumericId for ModuleId {
fn from_u32(id: u32) -> Self {
ModuleId(id)
}
fn to_u32(self) -> u32 {
self.0
}
}
#[derive(Clone, Copy, PartialEq, Eq, Hash, Debug)] #[derive(Clone, Copy, PartialEq, Eq, Hash, Debug)]
pub(crate) struct LinkId(u32); pub(crate) struct LinkId(u32);

View file

@ -10,6 +10,7 @@ extern crate rustc_hash;
extern crate salsa; extern crate salsa;
mod db; mod db;
mod loc2id;
mod input; mod input;
mod imp; mod imp;
mod completion; mod completion;

View file

@ -0,0 +1,64 @@
use std::hash::Hash;
use rustc_hash::FxHashMap;
/// There are two principle ways to refer to things:
/// - by their locatinon (module in foo/bar/baz.rs at line 42)
/// - by their numeric id (module `ModuleId(42)`)
///
/// The first one is more powerful (you can actually find the thing in question
/// by id), but the second one is so much more compact.
///
/// `Loc2IdMap` allows us to have a cake an eat it as well: by maintaining a
/// bidirectional mapping between positional and numeric ids, we can use compact
/// representation wich still allows us to get the actual item
#[derive(Debug)]
pub(crate) struct Loc2IdMap<L, ID>
where
ID: NumericId,
L: Clone + Eq + Hash,
{
loc2id: FxHashMap<L, ID>,
id2loc: FxHashMap<ID, L>,
}
impl<L, ID> Default for Loc2IdMap<L, ID>
where
ID: NumericId,
L: Clone + Eq + Hash,
{
fn default() -> Self {
Loc2IdMap {
loc2id: FxHashMap::default(),
id2loc: FxHashMap::default(),
}
}
}
impl<L, ID> Loc2IdMap<L, ID>
where
ID: NumericId,
L: Clone + Eq + Hash,
{
pub fn loc2id(&mut self, loc: &L) -> ID {
match self.loc2id.get(loc) {
Some(id) => return id.clone(),
None => (),
}
let id = self.loc2id.len();
assert!(id < u32::max_value() as usize);
let id = ID::from_u32(id as u32);
self.loc2id.insert(loc.clone(), id.clone());
self.id2loc.insert(id.clone(), loc.clone());
id
}
pub fn id2loc(&self, id: &ID) -> L {
self.id2loc[id].clone()
}
}
pub(crate) trait NumericId: Clone + Eq + Hash {
fn from_u32(id: u32) -> Self;
fn to_u32(self) -> u32;
}