aboutsummaryrefslogtreecommitdiff
path: root/crates/arena/src/lib.rs
diff options
context:
space:
mode:
authorIgor Aleksanov <[email protected]>2020-08-14 05:34:07 +0100
committerIgor Aleksanov <[email protected]>2020-08-14 05:34:07 +0100
commitc26c911ec1e6c2ad1dcb7d155a6a1d528839ad1a (patch)
tree7cff36c38234be0afb65273146d8247083a5cfeb /crates/arena/src/lib.rs
parent3c018bf84de5c693b5ee1c6bec0fed3b201c2060 (diff)
parentf1f73649a686dc6e6449afc35e0fa6fed00e225d (diff)
Merge branch 'master' into add-disable-diagnostics
Diffstat (limited to 'crates/arena/src/lib.rs')
-rw-r--r--crates/arena/src/lib.rs152
1 files changed, 152 insertions, 0 deletions
diff --git a/crates/arena/src/lib.rs b/crates/arena/src/lib.rs
new file mode 100644
index 000000000..3169aa5b8
--- /dev/null
+++ b/crates/arena/src/lib.rs
@@ -0,0 +1,152 @@
1//! Yet another index-based arena.
2
3use std::{
4 fmt,
5 hash::{Hash, Hasher},
6 iter::FromIterator,
7 marker::PhantomData,
8 ops::{Index, IndexMut},
9};
10
11pub mod map;
12
13#[derive(Clone, Copy, PartialEq, Eq, PartialOrd, Ord, Hash)]
14pub struct RawId(u32);
15
16impl From<RawId> for u32 {
17 fn from(raw: RawId) -> u32 {
18 raw.0
19 }
20}
21
22impl From<u32> for RawId {
23 fn from(id: u32) -> RawId {
24 RawId(id)
25 }
26}
27
28impl fmt::Debug for RawId {
29 fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
30 self.0.fmt(f)
31 }
32}
33
34impl fmt::Display for RawId {
35 fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
36 self.0.fmt(f)
37 }
38}
39
40pub struct Idx<T> {
41 raw: RawId,
42 _ty: PhantomData<fn() -> T>,
43}
44
45impl<T> Clone for Idx<T> {
46 fn clone(&self) -> Self {
47 *self
48 }
49}
50impl<T> Copy for Idx<T> {}
51
52impl<T> PartialEq for Idx<T> {
53 fn eq(&self, other: &Idx<T>) -> bool {
54 self.raw == other.raw
55 }
56}
57impl<T> Eq for Idx<T> {}
58
59impl<T> Hash for Idx<T> {
60 fn hash<H: Hasher>(&self, state: &mut H) {
61 self.raw.hash(state)
62 }
63}
64
65impl<T> fmt::Debug for Idx<T> {
66 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
67 let mut type_name = std::any::type_name::<T>();
68 if let Some(idx) = type_name.rfind(':') {
69 type_name = &type_name[idx + 1..]
70 }
71 write!(f, "Idx::<{}>({})", type_name, self.raw)
72 }
73}
74
75impl<T> Idx<T> {
76 pub fn from_raw(raw: RawId) -> Self {
77 Idx { raw, _ty: PhantomData }
78 }
79 pub fn into_raw(self) -> RawId {
80 self.raw
81 }
82}
83
84#[derive(Clone, PartialEq, Eq)]
85pub struct Arena<T> {
86 data: Vec<T>,
87}
88
89impl<T: fmt::Debug> fmt::Debug for Arena<T> {
90 fn fmt(&self, fmt: &mut fmt::Formatter) -> fmt::Result {
91 fmt.debug_struct("Arena").field("len", &self.len()).field("data", &self.data).finish()
92 }
93}
94
95impl<T> Arena<T> {
96 pub const fn new() -> Arena<T> {
97 Arena { data: Vec::new() }
98 }
99 pub fn clear(&mut self) {
100 self.data.clear();
101 }
102
103 pub fn len(&self) -> usize {
104 self.data.len()
105 }
106 pub fn is_empty(&self) -> bool {
107 self.data.is_empty()
108 }
109 pub fn alloc(&mut self, value: T) -> Idx<T> {
110 let id = RawId(self.data.len() as u32);
111 self.data.push(value);
112 Idx::from_raw(id)
113 }
114 pub fn iter(
115 &self,
116 ) -> impl Iterator<Item = (Idx<T>, &T)> + ExactSizeIterator + DoubleEndedIterator {
117 self.data.iter().enumerate().map(|(idx, value)| (Idx::from_raw(RawId(idx as u32)), value))
118 }
119 pub fn shrink_to_fit(&mut self) {
120 self.data.shrink_to_fit();
121 }
122}
123
124impl<T> Default for Arena<T> {
125 fn default() -> Arena<T> {
126 Arena { data: Vec::new() }
127 }
128}
129
130impl<T> Index<Idx<T>> for Arena<T> {
131 type Output = T;
132 fn index(&self, idx: Idx<T>) -> &T {
133 let idx = idx.into_raw().0 as usize;
134 &self.data[idx]
135 }
136}
137
138impl<T> IndexMut<Idx<T>> for Arena<T> {
139 fn index_mut(&mut self, idx: Idx<T>) -> &mut T {
140 let idx = idx.into_raw().0 as usize;
141 &mut self.data[idx]
142 }
143}
144
145impl<T> FromIterator<T> for Arena<T> {
146 fn from_iter<I>(iter: I) -> Self
147 where
148 I: IntoIterator<Item = T>,
149 {
150 Arena { data: Vec::from_iter(iter) }
151 }
152}