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
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
use std::fmt;
use std::hash::{Hash, Hasher};
use crate::boxed::*;
use crate::intern::{AsInterner, InternedSym};
#[repr(C, align(16))]
pub struct Sym {
header: Header,
pub(crate) interned: InternedSym,
}
impl Boxed for Sym {}
impl UniqueTagged for Sym {}
impl Sym {
pub fn new(heap: &mut impl AsHeap, value: &str) -> Gc<Sym> {
let heap = heap.as_heap_mut();
let interned = heap.type_info_mut().interner_mut().intern(value);
Self::from_interned_sym(heap, interned)
}
pub fn from_interned_sym(heap: &mut impl AsHeap, interned: InternedSym) -> Gc<Sym> {
let heap = heap.as_heap_mut();
heap.place_box(Sym {
header: Self::TYPE_TAG.to_heap_header(Self::size()),
interned,
})
}
pub fn size() -> BoxSize {
BoxSize::Size16
}
pub fn name<'a>(&'a self, interner: &'a impl AsInterner) -> &'a str {
interner.as_interner().unintern(&self.interned)
}
pub fn interned(&self) -> InternedSym {
self.interned
}
pub(crate) fn interned_mut(&mut self) -> &mut InternedSym {
&mut self.interned
}
}
impl PartialEq for Sym {
fn eq(&self, other: &Sym) -> bool {
self.interned == other.interned
}
}
impl Hash for Sym {
fn hash<H: Hasher>(&self, state: &mut H) {
Self::TYPE_TAG.hash(state);
self.interned.hash(state);
}
}
impl fmt::Debug for Sym {
fn fmt(&self, formatter: &mut fmt::Formatter<'_>) -> Result<(), fmt::Error> {
write!(formatter, "Sym({:?})", self.interned)
}
}
#[cfg(test)]
mod test {
use super::*;
use crate::boxed::heap::Heap;
use std::mem;
#[test]
fn sizes() {
assert_eq!(16, mem::size_of::<Sym>());
}
#[test]
fn equality() {
let mut heap = Heap::empty();
let boxed_one1 = Sym::new(&mut heap, "one");
let boxed_one2 = Sym::new(&mut heap, "one");
let boxed_two = Sym::new(&mut heap, "two");
assert_ne!(boxed_one1, boxed_two);
assert_eq!(boxed_one1, boxed_one2);
}
#[test]
fn fmt_debug() {
let mut heap = Heap::empty();
let boxed_one = Sym::new(&mut heap, "one");
assert_eq!(r#"Sym('one)"#, format!("{:?}", boxed_one));
}
}