mirror of
https://github.com/rust-lang/rust-analyzer
synced 2025-01-12 13:18:47 +00:00
Speedup VFS::partition
The task of `partition` function is to bin the flat list of paths into disjoint filesets. Ideally, it should be incremental -- each new file should be added to a specific fileset. However, preliminary measurnments show that it is actually fast enough if we just optimize this to use a binary search instead of a linear scan.
This commit is contained in:
parent
902a9c6da7
commit
b9f3c5d585
4 changed files with 30 additions and 11 deletions
|
@ -2,7 +2,7 @@
|
|||
//! relative paths.
|
||||
use std::{
|
||||
convert::{TryFrom, TryInto},
|
||||
ops,
|
||||
io, ops,
|
||||
path::{Component, Path, PathBuf},
|
||||
};
|
||||
|
||||
|
@ -46,6 +46,9 @@ impl TryFrom<&str> for AbsPathBuf {
|
|||
}
|
||||
|
||||
impl AbsPathBuf {
|
||||
pub fn canonicalized(path: &Path) -> io::Result<AbsPathBuf> {
|
||||
path.canonicalize().map(|it| AbsPathBuf::try_from(it).unwrap())
|
||||
}
|
||||
pub fn as_path(&self) -> &AbsPath {
|
||||
AbsPath::new_unchecked(self.0.as_path())
|
||||
}
|
||||
|
|
|
@ -1,5 +1,4 @@
|
|||
//! Missing batteries for standard libraries.
|
||||
|
||||
use std::{cell::Cell, fmt, time::Instant};
|
||||
|
||||
#[inline(always)]
|
||||
|
|
|
@ -2,7 +2,7 @@
|
|||
//!
|
||||
//! Files which do not belong to any explicitly configured `FileSet` belong to
|
||||
//! the default `FileSet`.
|
||||
use std::{cmp, fmt, iter};
|
||||
use std::{fmt, iter};
|
||||
|
||||
use paths::AbsPathBuf;
|
||||
use rustc_hash::FxHashMap;
|
||||
|
@ -44,6 +44,12 @@ pub struct FileSetConfig {
|
|||
roots: Vec<(AbsPathBuf, usize)>,
|
||||
}
|
||||
|
||||
impl Default for FileSetConfig {
|
||||
fn default() -> Self {
|
||||
FileSetConfig::builder().build()
|
||||
}
|
||||
}
|
||||
|
||||
impl FileSetConfig {
|
||||
pub fn builder() -> FileSetConfigBuilder {
|
||||
FileSetConfigBuilder::default()
|
||||
|
@ -60,16 +66,21 @@ impl FileSetConfig {
|
|||
self.n_file_sets
|
||||
}
|
||||
fn classify(&self, path: &VfsPath) -> usize {
|
||||
for (root, idx) in self.roots.iter() {
|
||||
if let Some(path) = path.as_path() {
|
||||
if path.starts_with(root) {
|
||||
return *idx;
|
||||
}
|
||||
}
|
||||
}
|
||||
let path = match path.as_path() {
|
||||
Some(it) => it,
|
||||
None => return self.len() - 1,
|
||||
};
|
||||
let idx = match self.roots.binary_search_by(|(p, _)| p.as_path().cmp(path)) {
|
||||
Ok(it) => it,
|
||||
Err(it) => it.saturating_sub(1),
|
||||
};
|
||||
if path.starts_with(&self.roots[idx].0) {
|
||||
self.roots[idx].1
|
||||
} else {
|
||||
self.len() - 1
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
pub struct FileSetConfigBuilder {
|
||||
roots: Vec<Vec<AbsPathBuf>>,
|
||||
|
@ -82,6 +93,9 @@ impl Default for FileSetConfigBuilder {
|
|||
}
|
||||
|
||||
impl FileSetConfigBuilder {
|
||||
pub fn len(&self) -> usize {
|
||||
self.roots.len()
|
||||
}
|
||||
pub fn add_file_set(&mut self, roots: Vec<AbsPathBuf>) {
|
||||
self.roots.push(roots)
|
||||
}
|
||||
|
@ -93,7 +107,7 @@ impl FileSetConfigBuilder {
|
|||
.enumerate()
|
||||
.flat_map(|(i, paths)| paths.into_iter().zip(iter::repeat(i)))
|
||||
.collect();
|
||||
roots.sort_by_key(|(path, _)| cmp::Reverse(path.to_string_lossy().len()));
|
||||
roots.sort();
|
||||
FileSetConfig { n_file_sets, roots }
|
||||
}
|
||||
}
|
||||
|
|
|
@ -79,6 +79,9 @@ pub enum ChangeKind {
|
|||
}
|
||||
|
||||
impl Vfs {
|
||||
pub fn len(&self) -> usize {
|
||||
self.data.len()
|
||||
}
|
||||
pub fn file_id(&self, path: &VfsPath) -> Option<FileId> {
|
||||
self.interner.get(path).filter(|&it| self.get(it).is_some())
|
||||
}
|
||||
|
|
Loading…
Reference in a new issue