mirror of
https://github.com/rust-lang/rust-analyzer
synced 2025-01-12 21:28:51 +00:00
Use rayon for parallel indexing
This commit is contained in:
parent
439e0fd32e
commit
d9e86e574a
2 changed files with 35 additions and 2 deletions
|
@ -8,6 +8,7 @@ log = "0.4.2"
|
||||||
failure = "0.1.2"
|
failure = "0.1.2"
|
||||||
parking_lot = "0.6.3"
|
parking_lot = "0.6.3"
|
||||||
once_cell = "0.1.4"
|
once_cell = "0.1.4"
|
||||||
|
rayon = "1.0.2"
|
||||||
fst = { git = "https://github.com/matklad/fst", branch = "subsequence"}
|
fst = { git = "https://github.com/matklad/fst", branch = "subsequence"}
|
||||||
libsyntax2 = { path = "../libsyntax2" }
|
libsyntax2 = { path = "../libsyntax2" }
|
||||||
libeditor = { path = "../libeditor" }
|
libeditor = { path = "../libeditor" }
|
||||||
|
|
|
@ -7,15 +7,21 @@ extern crate once_cell;
|
||||||
extern crate libsyntax2;
|
extern crate libsyntax2;
|
||||||
extern crate libeditor;
|
extern crate libeditor;
|
||||||
extern crate fst;
|
extern crate fst;
|
||||||
|
extern crate rayon;
|
||||||
|
|
||||||
mod symbol_index;
|
mod symbol_index;
|
||||||
|
|
||||||
use once_cell::sync::OnceCell;
|
use once_cell::sync::OnceCell;
|
||||||
|
use rayon::prelude::*;
|
||||||
|
|
||||||
use std::{
|
use std::{
|
||||||
sync::Arc,
|
sync::{
|
||||||
|
Arc,
|
||||||
|
atomic::{AtomicUsize, Ordering::SeqCst},
|
||||||
|
},
|
||||||
collections::hash_map::HashMap,
|
collections::hash_map::HashMap,
|
||||||
path::{PathBuf, Path},
|
path::{PathBuf, Path},
|
||||||
|
time::Instant,
|
||||||
};
|
};
|
||||||
|
|
||||||
use libsyntax2::{
|
use libsyntax2::{
|
||||||
|
@ -29,6 +35,7 @@ use self::symbol_index::FileSymbols;
|
||||||
pub use self::symbol_index::Query;
|
pub use self::symbol_index::Query;
|
||||||
|
|
||||||
pub type Result<T> = ::std::result::Result<T, ::failure::Error>;
|
pub type Result<T> = ::std::result::Result<T, ::failure::Error>;
|
||||||
|
const INDEXING_THRESHOLD: usize = 128;
|
||||||
|
|
||||||
pub struct WorldState {
|
pub struct WorldState {
|
||||||
data: Arc<WorldData>
|
data: Arc<WorldData>
|
||||||
|
@ -56,7 +63,9 @@ impl WorldState {
|
||||||
|
|
||||||
pub fn change_files(&mut self, changes: impl Iterator<Item=(PathBuf, Option<String>)>) {
|
pub fn change_files(&mut self, changes: impl Iterator<Item=(PathBuf, Option<String>)>) {
|
||||||
let data = self.data_mut();
|
let data = self.data_mut();
|
||||||
|
let mut cnt = 0;
|
||||||
for (path, text) in changes {
|
for (path, text) in changes {
|
||||||
|
cnt += 1;
|
||||||
data.file_map.remove(&path);
|
data.file_map.remove(&path);
|
||||||
if let Some(text) = text {
|
if let Some(text) = text {
|
||||||
let file_data = FileData::new(text);
|
let file_data = FileData::new(text);
|
||||||
|
@ -65,11 +74,15 @@ impl WorldState {
|
||||||
data.file_map.remove(&path);
|
data.file_map.remove(&path);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
*data.unindexed.get_mut() += cnt;
|
||||||
}
|
}
|
||||||
|
|
||||||
fn data_mut(&mut self) -> &mut WorldData {
|
fn data_mut(&mut self) -> &mut WorldData {
|
||||||
if Arc::get_mut(&mut self.data).is_none() {
|
if Arc::get_mut(&mut self.data).is_none() {
|
||||||
self.data = Arc::new(WorldData {
|
self.data = Arc::new(WorldData {
|
||||||
|
unindexed: AtomicUsize::new(
|
||||||
|
self.data.unindexed.load(SeqCst)
|
||||||
|
),
|
||||||
file_map: self.data.file_map.clone(),
|
file_map: self.data.file_map.clone(),
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
@ -95,10 +108,11 @@ impl World {
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn world_symbols<'a>(&'a self, mut query: Query) -> impl Iterator<Item=(&'a Path, &'a FileSymbol)> + 'a {
|
pub fn world_symbols<'a>(&'a self, mut query: Query) -> impl Iterator<Item=(&'a Path, &'a FileSymbol)> + 'a {
|
||||||
|
self.reindex();
|
||||||
self.data.file_map.iter()
|
self.data.file_map.iter()
|
||||||
.flat_map(move |(path, data)| {
|
.flat_map(move |(path, data)| {
|
||||||
let path: &'a Path = path.as_path();
|
|
||||||
let symbols = data.symbols();
|
let symbols = data.symbols();
|
||||||
|
let path: &'a Path = path.as_path();
|
||||||
query.process(symbols).into_iter().map(move |s| (path, s))
|
query.process(symbols).into_iter().map(move |s| (path, s))
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
@ -129,6 +143,23 @@ impl World {
|
||||||
Ok(self.world_symbols(query).collect())
|
Ok(self.world_symbols(query).collect())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn reindex(&self) {
|
||||||
|
let data = &*self.data;
|
||||||
|
let unindexed = data.unindexed.load(SeqCst);
|
||||||
|
if unindexed < INDEXING_THRESHOLD {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if unindexed == data.unindexed.compare_and_swap(unindexed, 0, SeqCst) {
|
||||||
|
let now = Instant::now();
|
||||||
|
data.file_map
|
||||||
|
.par_iter()
|
||||||
|
.for_each(|(_, data)| {
|
||||||
|
data.symbols();
|
||||||
|
});
|
||||||
|
info!("parallel indexing took {:?}", now.elapsed());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
fn file_data(&self, path: &Path) -> Result<Arc<FileData>> {
|
fn file_data(&self, path: &Path) -> Result<Arc<FileData>> {
|
||||||
match self.data.file_map.get(path) {
|
match self.data.file_map.get(path) {
|
||||||
Some(data) => Ok(data.clone()),
|
Some(data) => Ok(data.clone()),
|
||||||
|
@ -150,6 +181,7 @@ pub const BREAK: SearchResult = Err(Break);
|
||||||
|
|
||||||
#[derive(Default, Debug)]
|
#[derive(Default, Debug)]
|
||||||
struct WorldData {
|
struct WorldData {
|
||||||
|
unindexed: AtomicUsize,
|
||||||
file_map: HashMap<PathBuf, Arc<FileData>>,
|
file_map: HashMap<PathBuf, Arc<FileData>>,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue