rust-analyzer/crates/vfs/src/path_interner.rs

44 lines
1.3 KiB
Rust
Raw Normal View History

2020-06-15 11:29:07 +00:00
//! Maps paths to compact integer ids. We don't care about clearings paths which
//! no longer exist -- the assumption is total size of paths we ever look at is
//! not too big.
use std::hash::BuildHasherDefault;
use indexmap::IndexSet;
use rustc_hash::FxHasher;
2020-06-15 11:29:07 +00:00
use crate::{FileId, VfsPath};
2021-01-12 16:41:45 +00:00
/// Structure to map between [`VfsPath`] and [`FileId`].
#[derive(Default)]
2020-06-15 11:29:07 +00:00
pub(crate) struct PathInterner {
map: IndexSet<VfsPath, BuildHasherDefault<FxHasher>>,
}
2020-06-15 11:29:07 +00:00
impl PathInterner {
2021-01-12 16:41:45 +00:00
/// Get the id corresponding to `path`.
///
/// If `path` does not exists in `self`, returns [`None`].
2020-06-15 11:29:07 +00:00
pub(crate) fn get(&self, path: &VfsPath) -> Option<FileId> {
self.map.get_index_of(path).map(|i| FileId(i as u32))
2020-06-15 11:29:07 +00:00
}
2021-01-12 16:41:45 +00:00
/// Insert `path` in `self`.
///
/// - If `path` already exists in `self`, returns its associated id;
/// - Else, returns a newly allocated id.
2020-06-15 11:29:07 +00:00
pub(crate) fn intern(&mut self, path: VfsPath) -> FileId {
let (id, _added) = self.map.insert_full(path);
assert!(id < u32::MAX as usize);
FileId(id as u32)
2020-06-15 11:29:07 +00:00
}
2021-01-12 16:41:45 +00:00
/// Returns the path corresponding to `id`.
///
/// # Panics
///
/// Panics if `id` does not exists in `self`.
2020-06-15 11:29:07 +00:00
pub(crate) fn lookup(&self, id: FileId) -> &VfsPath {
self.map.get_index(id.0 as usize).unwrap()
2020-06-15 11:29:07 +00:00
}
}