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
use alloc::sync::Arc;
use core::marker::PhantomData;
use core::mem;
use core::sync::atomic::{AtomicUsize, Ordering};
#[derive(Debug)]
pub struct ArcCell<T>(AtomicUsize, PhantomData<Arc<T>>);
impl<T> Drop for ArcCell<T> {
fn drop(&mut self) {
self.take();
}
}
impl<T> ArcCell<T> {
pub fn new(t: Arc<T>) -> ArcCell<T> {
ArcCell(AtomicUsize::new(unsafe { mem::transmute(t) }), PhantomData)
}
fn take(&self) -> Arc<T> {
loop {
match self.0.swap(0, Ordering::Acquire) {
0 => {}
n => return unsafe { mem::transmute(n) },
}
}
}
fn put(&self, t: Arc<T>) {
debug_assert_eq!(self.0.load(Ordering::SeqCst), 0);
self.0
.store(unsafe { mem::transmute(t) }, Ordering::Release);
}
pub fn set(&self, t: Arc<T>) -> Arc<T> {
let old = self.take();
self.put(t);
old
}
pub fn get(&self) -> Arc<T> {
let t = self.take();
let out = t.clone();
self.put(t);
out
}
}
impl<T: Default> Default for ArcCell<T> {
fn default() -> Self {
ArcCell::new(Arc::default())
}
}
impl<T> From<Arc<T>> for ArcCell<T> {
fn from(value: Arc<T>) -> Self {
ArcCell::new(value)
}
}
impl<T> From<T> for ArcCell<T> {
fn from(value: T) -> Self {
ArcCell::new(Arc::new(value))
}
}
#[cfg(test)]
mod test {
use std::sync::atomic::{AtomicUsize, Ordering, ATOMIC_USIZE_INIT};
use std::sync::Arc;
use super::*;
#[test]
fn basic() {
let r = ArcCell::new(Arc::new(0));
assert_eq!(*r.get(), 0);
assert_eq!(*r.set(Arc::new(1)), 0);
assert_eq!(*r.get(), 1);
}
#[test]
fn drop_runs() {
static DROPS: AtomicUsize = ATOMIC_USIZE_INIT;
struct Foo;
impl Drop for Foo {
fn drop(&mut self) {
DROPS.fetch_add(1, Ordering::SeqCst);
}
}
let r = ArcCell::new(Arc::new(Foo));
let _f = r.get();
r.get();
r.set(Arc::new(Foo));
drop(_f);
assert_eq!(DROPS.load(Ordering::SeqCst), 1);
drop(r);
assert_eq!(DROPS.load(Ordering::SeqCst), 2);
}
}