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
pub struct Entry {
pub ptr: *mut (),
pub drop: unsafe fn(*mut ()),
}
#[cfg(feature = "slab")]
mod slab_impl {
use std::cell::UnsafeCell;
use std::num::NonZeroUsize;
use super::Entry;
pub struct Registry(pub slab::Slab<Entry>);
thread_local!(static REGISTRY: UnsafeCell<Registry> = UnsafeCell::new(Registry(slab::Slab::new())));
pub use usize as ItemId;
pub fn insert(thread_id: NonZeroUsize, entry: Entry) -> ItemId {
let _ = thread_id;
REGISTRY.with(|registry| unsafe { (*registry.get()).0.insert(entry) })
}
pub fn with<R, F: FnOnce(&Entry) -> R>(item_id: ItemId, thread_id: NonZeroUsize, f: F) -> R {
let _ = thread_id;
REGISTRY.with(|registry| f(unsafe { &*registry.get() }.0.get(item_id).unwrap()))
}
pub fn remove(item_id: ItemId, thread_id: NonZeroUsize) -> Entry {
let _ = thread_id;
REGISTRY.with(|registry| unsafe { (*registry.get()).0.remove(item_id) })
}
}
#[cfg(not(feature = "slab"))]
mod map_impl {
use std::cell::UnsafeCell;
use std::num::NonZeroUsize;
use super::Entry;
pub struct Registry(pub std::collections::HashMap<NonZeroUsize, Entry>);
thread_local!(static REGISTRY: UnsafeCell<Registry> = UnsafeCell::new(Registry(Default::default())));
pub type ItemId = ();
pub fn insert(thread_id: NonZeroUsize, entry: Entry) -> ItemId {
REGISTRY.with(|registry| unsafe { (*registry.get()).0.insert(thread_id, entry) });
}
pub fn with<R, F: FnOnce(&Entry) -> R>(item_id: ItemId, thread_id: NonZeroUsize, f: F) -> R {
let _ = item_id;
REGISTRY.with(|registry| f(unsafe { &*registry.get() }.0.get(&thread_id).unwrap()))
}
pub fn remove(item_id: ItemId, thread_id: NonZeroUsize) -> Entry {
let _ = item_id;
REGISTRY.with(|registry| unsafe { (*registry.get()).0.remove(&thread_id).unwrap() })
}
}
#[cfg(feature = "slab")]
pub use self::slab_impl::*;
#[cfg(not(feature = "slab"))]
pub use self::map_impl::*;
impl Drop for Registry {
fn drop(&mut self) {
for (_, value) in self.0.iter() {
unsafe { (value.drop)(value.ptr) };
}
}
}