arc_swap/debt/mod.rs
1//! Debt handling.
2//!
3//! A debt is a reference count of a smart pointer that is owed. This module provides a lock-free
4//! storage for debts.
5//!
6//! Each thread has its own node with bunch of slots. Only that thread can allocate debts in there,
7//! but others are allowed to inspect and pay them. The nodes form a linked list for the reason of
8//! inspection. The nodes are never removed (even after the thread terminates), but if the thread
9//! gives it up, another (new) thread can claim it.
10//!
11//! The writers walk the whole chain and pay the debts (by bumping the ref counts) of the just
12//! removed pointer.
13//!
14//! Each node has some fast (but fallible) nodes and a fallback node, with different algorithms to
15//! claim them (see the relevant submodules).
16
17use core::sync::atomic::AtomicUsize;
18use core::sync::atomic::Ordering::*;
19
20pub(crate) use self::list::{LocalNode, Node};
21use super::RefCnt;
22
23mod fast;
24mod helping;
25mod list;
26
27/// One debt slot.
28///
29/// It may contain an „owed“ reference count.
30#[derive(Debug)]
31pub(crate) struct Debt(pub(crate) AtomicUsize);
32
33impl Debt {
34 /// The value of pointer `3` should be pretty safe, for two reasons:
35 ///
36 /// * It's an odd number, but the pointers we have are likely aligned at least to the word size,
37 /// because the data at the end of the `Arc` has the counters.
38 /// * It's in the very first page where NULL lives, so it's not mapped.
39 pub(crate) const NONE: usize = 0b11;
40}
41
42impl Default for Debt {
43 fn default() -> Self {
44 Debt(AtomicUsize::new(Self::NONE))
45 }
46}
47
48impl Debt {
49 /// Tries to pay the given debt.
50 ///
51 /// If the debt is still there, for the given pointer, it is paid and `true` is returned. If it
52 /// is empty or if there's some other pointer, it is not paid and `false` is returned, meaning
53 /// the debt was paid previously by someone else.
54 ///
55 /// # Notes
56 ///
57 /// * It is possible that someone paid the debt and then someone else put a debt for the same
58 /// pointer in there. This is fine, as we'll just pay the debt for that someone else.
59 /// * This relies on the fact that the same pointer must point to the same object and
60 /// specifically to the same type ‒ the caller provides the type, it's destructor, etc.
61 /// * It also relies on the fact the same thing is not stuffed both inside an `Arc` and `Rc` or
62 /// something like that, but that sounds like a reasonable assumption. Someone storing it
63 /// through `ArcSwap<T>` and someone else with `ArcSwapOption<T>` will work.
64 #[inline]
65 pub(crate) fn pay<T: RefCnt>(&self, ptr: *const T::Base) -> bool {
66 self.0
67 // If we don't change anything because there's something else, Relaxed is fine.
68 //
69 // The Release works as kind of Mutex. We make sure nothing from the debt-protected
70 // sections leaks below this point.
71 //
72 // Note that if it got paid already, it is inside the reference count. We don't
73 // necessarily observe that increment, but whoever destroys the pointer *must* see the
74 // up to date value, with all increments already counted in (the Arc takes care of that
75 // part).
76 .compare_exchange(ptr as usize, Self::NONE, Release, Relaxed)
77 .is_ok()
78 }
79
80 /// Pays all the debts on the given pointer and the storage.
81 pub(crate) fn pay_all<T, R>(ptr: *const T::Base, storage_addr: usize, replacement: R)
82 where
83 T: RefCnt,
84 R: Fn() -> T,
85 {
86 LocalNode::with(|local| {
87 let val = unsafe { T::from_ptr(ptr) };
88 // Pre-pay one ref count that can be safely put into a debt slot to pay it.
89 T::inc(&val);
90
91 Node::traverse::<(), _>(|node| {
92 // Make the cooldown trick know we are poking into this node.
93 let _reservation = node.reserve_writer();
94
95 local.help(node, storage_addr, &replacement);
96
97 let all_slots = node
98 .fast_slots()
99 .chain(core::iter::once(node.helping_slot()));
100 for slot in all_slots {
101 // Note: Release is enough even here. That makes sure the increment is
102 // visible to whoever might acquire on this slot and can't leak below this.
103 // And we are the ones doing decrements anyway.
104 if slot.pay::<T>(ptr) {
105 // Pre-pay one more, for another future slot
106 T::inc(&val);
107 }
108 }
109
110 None
111 });
112 // Implicit dec by dropping val in here, pair for the above
113 })
114 }
115}
116
117#[cfg(test)]
118mod tests {
119 use alloc::sync::Arc;
120
121 /// Checks the assumption that arcs to ZSTs have different pointer values.
122 #[test]
123 fn arc_zst() {
124 struct A;
125 struct B;
126
127 let a = Arc::new(A);
128 let b = Arc::new(B);
129
130 let aref: &A = &a;
131 let bref: &B = &b;
132
133 let aptr = aref as *const _ as usize;
134 let bptr = bref as *const _ as usize;
135 assert_ne!(aptr, bptr);
136 }
137}