1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
//! An extremely simple, non-threadsafe string interner to reduce the memory usage and the
//! allocation pressure of ParserDatabase.
//!
//! The StringIds returned by `intern` are only valid for this specific instance of the interner
//! they were interned with.

use indexmap::IndexSet;

#[derive(Clone, Copy, PartialEq, Eq, PartialOrd, Ord, Debug, Hash)]
pub(crate) struct StringId(usize);

#[derive(Default)]
pub(crate) struct StringInterner {
    map: IndexSet<String>,
}

impl StringInterner {
    pub(crate) fn get(&self, id: StringId) -> Option<&str> {
        self.map.get_index(id.0).map(|s| s.as_str())
    }

    /// Get an already-interned string.
    pub(crate) fn lookup(&self, s: &str) -> Option<StringId> {
        self.map.get_index_of(s).map(StringId)
    }

    pub(crate) fn intern(&mut self, s: &str) -> StringId {
        if let Some(id) = self.lookup(s) {
            id
        } else {
            let (idx, is_new) = self.map.insert_full(s.to_owned());
            debug_assert!(is_new);
            StringId(idx)
        }
    }
}