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
//! A simple wrapper that prevents the inner object from being dropped.

#![no_std]

use core::{
    fmt::{self, Debug},
    mem::ManuallyDrop,
    ops::{Deref, DerefMut}
};

/// A wrapper for an inner object that ensures the inner object is never dropped.
///
/// This is effectively a safe version of `ManuallyDrop` with a restricted interface.
/// 
/// Auto-derefs to the inner object type `T`.
///
/// To re-take ownership of the object, call [`Self::into_inner()`].
#[repr(transparent)]
pub struct NoDrop<T>(ManuallyDrop<T>);

impl<T> NoDrop<T> {
    /// Wraps the given `obj` in a `NoDrop` wrapper.
    pub const fn new(obj: T) -> NoDrop<T> {
        NoDrop(ManuallyDrop::new(obj))
    }

    /// Consumes this `NoDrop` wrapper and returns the inner object.
    pub const fn into_inner(self) -> T {
        ManuallyDrop::into_inner(self.0)
    }
}

impl<T: Debug> Debug for NoDrop<T> {
    fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
        Debug::fmt(&self.0.deref(), f)
    }
}
impl<T> Deref for NoDrop<T> {
    type Target = T;
    fn deref(&self) -> &Self::Target {
        &self.0
    }
}
impl<T> DerefMut for NoDrop<T> {
    fn deref_mut(&mut self) -> &mut Self::Target {
        &mut self.0
    }
}