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
use core::mem::MaybeUninit;
use core::ops::Deref;
use core::sync::atomic::{AtomicBool, Ordering::Relaxed};
use std::sync::Arc;

const INIT_CODE: u32 = 0xF47E8A5B;

#[derive(Clone)]
pub struct OnDropToken(Arc<AtomicBool>);
impl OnDropToken {
    pub fn is_droped(&self) -> bool {
        self.0.load(Relaxed)
    }
}

pub struct OnDrop<T> {
    data: T,
    is_init: u32,
    drop_time: u8,
    on_drop: MaybeUninit<Box<dyn FnOnce()>>,
}
impl<T> OnDrop<T> {
    pub fn new<F: 'static + FnOnce()>(data: T, on_drop: F) -> Self {
        Self {
            data,
            is_init: INIT_CODE,
            drop_time: 0,
            on_drop: MaybeUninit::new(Box::new(on_drop)),
        }
    }
    pub fn token(data: T) -> (Self, OnDropToken) {
        let droped = Arc::new(AtomicBool::new(false));
        let token = OnDropToken(droped.clone());
        (
            Self {
                data,
                is_init: INIT_CODE,
                drop_time: 0,
                on_drop: MaybeUninit::new(Box::new(move || droped.store(true, Relaxed))),
            },
            token,
        )
    }
}
impl<T> Drop for OnDrop<T> {
    fn drop(&mut self) {
        if self.is_init == INIT_CODE {
            self.drop_time += 1;
            if self.drop_time != 1 {
                panic!("drop {} times", self.drop_time);
            }
            (unsafe { self.on_drop.as_ptr().read() })();
        } else {
            panic!("uninitialized");
        }
    }
}
impl<T> Deref for OnDrop<T> {
    type Target = T;
    fn deref(&self) -> &Self::Target {
        &self.data
    }
}
impl<T: PartialEq> PartialEq for OnDrop<T> {
    fn eq(&self, other: &OnDrop<T>) -> bool {
        self.data == other.data
    }
}