Files
wasmtime/cranelift/codegen/src/scoped_hash_map.rs

234 lines
7.4 KiB
Rust

//! `ScopedHashMap`
//!
//! This module defines a struct `ScopedHashMap<K, V>` which defines a `FxHashMap`-like
//! container that has a concept of scopes that can be entered and exited, such that
//! values inserted while inside a scope aren't visible outside the scope.
use crate::fx::FxHashMap;
use core::hash::Hash;
use core::mem;
#[cfg(not(feature = "std"))]
use crate::fx::FxHasher;
#[cfg(not(feature = "std"))]
type Hasher = core::hash::BuildHasherDefault<FxHasher>;
struct Val<K, V> {
value: V,
next_key: Option<K>,
depth: usize,
}
/// A view into an occupied entry in a `ScopedHashMap`. It is part of the `Entry` enum.
pub struct OccupiedEntry<'a, K: 'a, V: 'a> {
#[cfg(feature = "std")]
entry: super::hash_map::OccupiedEntry<'a, K, Val<K, V>>,
#[cfg(not(feature = "std"))]
entry: super::hash_map::OccupiedEntry<'a, K, Val<K, V>, Hasher>,
}
impl<'a, K, V> OccupiedEntry<'a, K, V> {
/// Gets a reference to the value in the entry.
pub fn get(&self) -> &V {
&self.entry.get().value
}
}
/// A view into a vacant entry in a `ScopedHashMap`. It is part of the `Entry` enum.
pub struct VacantEntry<'a, K: 'a, V: 'a> {
#[cfg(feature = "std")]
entry: super::hash_map::VacantEntry<'a, K, Val<K, V>>,
#[cfg(not(feature = "std"))]
entry: super::hash_map::VacantEntry<'a, K, Val<K, V>, Hasher>,
next_key: Option<K>,
depth: usize,
}
impl<'a, K: Hash, V> VacantEntry<'a, K, V> {
/// Sets the value of the entry with the `VacantEntry`'s key.
pub fn insert(self, value: V) {
self.entry.insert(Val {
value,
next_key: self.next_key,
depth: self.depth,
});
}
}
/// A view into a single entry in a map, which may either be vacant or occupied.
///
/// This enum is constructed from the `entry` method on `ScopedHashMap`.
pub enum Entry<'a, K: 'a, V: 'a> {
Occupied(OccupiedEntry<'a, K, V>),
Vacant(VacantEntry<'a, K, V>),
}
/// A wrapper around a `FxHashMap` which adds the concept of scopes. Items inserted
/// within a scope are removed when the scope is exited.
///
/// Shadowing, where one scope has entries with the same keys as a containing scope,
/// is not supported in this implementation.
pub struct ScopedHashMap<K, V> {
map: FxHashMap<K, Val<K, V>>,
last_insert: Option<K>,
current_depth: usize,
}
impl<K, V> ScopedHashMap<K, V>
where
K: PartialEq + Eq + Hash + Clone,
{
/// Creates an empty `ScopedHashMap`.
pub fn new() -> Self {
Self {
map: FxHashMap(),
last_insert: None,
current_depth: 0,
}
}
/// Similar to `FxHashMap::entry`, gets the given key's corresponding entry in the map for
/// in-place manipulation.
pub fn entry(&mut self, key: K) -> Entry<K, V> {
use super::hash_map::Entry::*;
match self.map.entry(key) {
Occupied(entry) => Entry::Occupied(OccupiedEntry { entry }),
Vacant(entry) => {
let clone_key = entry.key().clone();
Entry::Vacant(VacantEntry {
entry,
next_key: mem::replace(&mut self.last_insert, Some(clone_key)),
depth: self.current_depth,
})
}
}
}
/// Enter a new scope.
pub fn increment_depth(&mut self) {
// Increment the depth.
self.current_depth = self.current_depth.checked_add(1).unwrap();
}
/// Exit the current scope.
pub fn decrement_depth(&mut self) {
// Remove all elements inserted at the current depth.
while let Some(key) = self.last_insert.clone() {
use crate::hash_map::Entry::*;
match self.map.entry(key) {
Occupied(entry) => {
if entry.get().depth != self.current_depth {
break;
}
self.last_insert = entry.remove_entry().1.next_key;
}
Vacant(_) => panic!(),
}
}
// Decrement the depth.
self.current_depth = self.current_depth.checked_sub(1).unwrap();
}
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn basic() {
let mut map: ScopedHashMap<i32, i32> = ScopedHashMap::new();
match map.entry(0) {
Entry::Occupied(_entry) => panic!(),
Entry::Vacant(entry) => entry.insert(1),
}
match map.entry(2) {
Entry::Occupied(_entry) => panic!(),
Entry::Vacant(entry) => entry.insert(8),
}
match map.entry(2) {
Entry::Occupied(entry) => assert!(*entry.get() == 8),
Entry::Vacant(_entry) => panic!(),
}
map.increment_depth();
match map.entry(2) {
Entry::Occupied(entry) => assert!(*entry.get() == 8),
Entry::Vacant(_entry) => panic!(),
}
match map.entry(1) {
Entry::Occupied(_entry) => panic!(),
Entry::Vacant(entry) => entry.insert(3),
}
match map.entry(1) {
Entry::Occupied(entry) => assert!(*entry.get() == 3),
Entry::Vacant(_entry) => panic!(),
}
match map.entry(0) {
Entry::Occupied(entry) => assert!(*entry.get() == 1),
Entry::Vacant(_entry) => panic!(),
}
match map.entry(2) {
Entry::Occupied(entry) => assert!(*entry.get() == 8),
Entry::Vacant(_entry) => panic!(),
}
map.decrement_depth();
match map.entry(0) {
Entry::Occupied(entry) => assert!(*entry.get() == 1),
Entry::Vacant(_entry) => panic!(),
}
match map.entry(2) {
Entry::Occupied(entry) => assert!(*entry.get() == 8),
Entry::Vacant(_entry) => panic!(),
}
map.increment_depth();
match map.entry(2) {
Entry::Occupied(entry) => assert!(*entry.get() == 8),
Entry::Vacant(_entry) => panic!(),
}
match map.entry(1) {
Entry::Occupied(_entry) => panic!(),
Entry::Vacant(entry) => entry.insert(4),
}
match map.entry(1) {
Entry::Occupied(entry) => assert!(*entry.get() == 4),
Entry::Vacant(_entry) => panic!(),
}
match map.entry(2) {
Entry::Occupied(entry) => assert!(*entry.get() == 8),
Entry::Vacant(_entry) => panic!(),
}
map.decrement_depth();
map.increment_depth();
map.increment_depth();
map.increment_depth();
match map.entry(2) {
Entry::Occupied(entry) => assert!(*entry.get() == 8),
Entry::Vacant(_entry) => panic!(),
}
match map.entry(1) {
Entry::Occupied(_entry) => panic!(),
Entry::Vacant(entry) => entry.insert(5),
}
match map.entry(1) {
Entry::Occupied(entry) => assert!(*entry.get() == 5),
Entry::Vacant(_entry) => panic!(),
}
match map.entry(2) {
Entry::Occupied(entry) => assert!(*entry.get() == 8),
Entry::Vacant(_entry) => panic!(),
}
map.decrement_depth();
map.decrement_depth();
map.decrement_depth();
match map.entry(2) {
Entry::Occupied(entry) => assert!(*entry.get() == 8),
Entry::Vacant(_entry) => panic!(),
}
match map.entry(1) {
Entry::Occupied(_entry) => panic!(),
Entry::Vacant(entry) => entry.insert(3),
}
}
}