Rename the 'cretonne' crate to 'cretonne-codegen'.
This fixes the next part of #287.
This commit is contained in:
222
lib/codegen/src/scoped_hash_map.rs
Normal file
222
lib/codegen/src/scoped_hash_map.rs
Normal file
@@ -0,0 +1,222 @@
|
||||
//! `ScopedHashMap`
|
||||
//!
|
||||
//! This module defines a struct `ScopedHashMap<K, V>` which defines a `HashMap`-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 std::collections::{hash_map, HashMap};
|
||||
use std::hash::Hash;
|
||||
use std::mem;
|
||||
|
||||
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> {
|
||||
entry: hash_map::OccupiedEntry<'a, K, Val<K, V>>,
|
||||
}
|
||||
|
||||
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> {
|
||||
entry: hash_map::VacantEntry<'a, K, Val<K, V>>,
|
||||
next_key: Option<K>,
|
||||
depth: usize,
|
||||
}
|
||||
|
||||
impl<'a, K, 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 `HashMap` 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: HashMap<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: HashMap::new(),
|
||||
last_insert: None,
|
||||
current_depth: 0,
|
||||
}
|
||||
}
|
||||
|
||||
/// Similar to `HashMap::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 self::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 self::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),
|
||||
}
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user