xref: /linux/rust/pin-init/src/alloc.rs (revision 6315d93541f8a5f77c5ef5c4f25233e66d189603)
1 // SPDX-License-Identifier: Apache-2.0 OR MIT
2 
3 #[cfg(all(feature = "alloc", not(feature = "std")))]
4 use alloc::{boxed::Box, sync::Arc};
5 #[cfg(feature = "alloc")]
6 use core::alloc::AllocError;
7 use core::{mem::MaybeUninit, pin::Pin};
8 #[cfg(feature = "std")]
9 use std::sync::Arc;
10 
11 #[cfg(not(feature = "alloc"))]
12 type AllocError = core::convert::Infallible;
13 
14 use crate::{
15     init_from_closure, pin_init_from_closure, InPlaceWrite, Init, PinInit, ZeroableOption,
16 };
17 
18 pub extern crate alloc;
19 
20 // SAFETY: All zeros is equivalent to `None` (option layout optimization guarantee).
21 //
22 // In this case we are allowed to use `T: ?Sized`, since all zeros is the `None` variant and there
23 // is no problem with a VTABLE pointer being null.
24 unsafe impl<T: ?Sized> ZeroableOption for Box<T> {}
25 
26 /// Smart pointer that can initialize memory in-place.
27 pub trait InPlaceInit<T>: Sized {
28     /// Use the given pin-initializer to pin-initialize a `T` inside of a new smart pointer of this
29     /// type.
30     ///
31     /// If `T: !Unpin` it will not be able to move afterwards.
32     fn try_pin_init<E>(init: impl PinInit<T, E>) -> Result<Pin<Self>, E>
33     where
34         E: From<AllocError>;
35 
36     /// Use the given pin-initializer to pin-initialize a `T` inside of a new smart pointer of this
37     /// type.
38     ///
39     /// If `T: !Unpin` it will not be able to move afterwards.
40     fn pin_init(init: impl PinInit<T>) -> Result<Pin<Self>, AllocError> {
41         // SAFETY: We delegate to `init` and only change the error type.
42         let init = unsafe {
43             pin_init_from_closure(|slot| match init.__pinned_init(slot) {
44                 Ok(()) => Ok(()),
45                 Err(i) => match i {},
46             })
47         };
48         Self::try_pin_init(init)
49     }
50 
51     /// Use the given initializer to in-place initialize a `T`.
52     fn try_init<E>(init: impl Init<T, E>) -> Result<Self, E>
53     where
54         E: From<AllocError>;
55 
56     /// Use the given initializer to in-place initialize a `T`.
57     fn init(init: impl Init<T>) -> Result<Self, AllocError> {
58         // SAFETY: We delegate to `init` and only change the error type.
59         let init = unsafe {
60             init_from_closure(|slot| match init.__init(slot) {
61                 Ok(()) => Ok(()),
62                 Err(i) => match i {},
63             })
64         };
65         Self::try_init(init)
66     }
67 }
68 
69 #[cfg(feature = "alloc")]
70 macro_rules! try_new_uninit {
71     ($type:ident) => {
72         $type::try_new_uninit()?
73     };
74 }
75 #[cfg(all(feature = "std", not(feature = "alloc")))]
76 macro_rules! try_new_uninit {
77     ($type:ident) => {
78         $type::new_uninit()
79     };
80 }
81 
82 impl<T> InPlaceInit<T> for Box<T> {
83     #[inline]
84     fn try_pin_init<E>(init: impl PinInit<T, E>) -> Result<Pin<Self>, E>
85     where
86         E: From<AllocError>,
87     {
88         try_new_uninit!(Box).write_pin_init(init)
89     }
90 
91     #[inline]
92     fn try_init<E>(init: impl Init<T, E>) -> Result<Self, E>
93     where
94         E: From<AllocError>,
95     {
96         try_new_uninit!(Box).write_init(init)
97     }
98 }
99 
100 impl<T> InPlaceInit<T> for Arc<T> {
101     #[inline]
102     fn try_pin_init<E>(init: impl PinInit<T, E>) -> Result<Pin<Self>, E>
103     where
104         E: From<AllocError>,
105     {
106         let mut this = try_new_uninit!(Arc);
107         let Some(slot) = Arc::get_mut(&mut this) else {
108             // SAFETY: the Arc has just been created and has no external references
109             unsafe { core::hint::unreachable_unchecked() }
110         };
111         let slot = slot.as_mut_ptr();
112         // SAFETY: When init errors/panics, slot will get deallocated but not dropped,
113         // slot is valid and will not be moved, because we pin it later.
114         unsafe { init.__pinned_init(slot)? };
115         // SAFETY: All fields have been initialized and this is the only `Arc` to that data.
116         Ok(unsafe { Pin::new_unchecked(this.assume_init()) })
117     }
118 
119     #[inline]
120     fn try_init<E>(init: impl Init<T, E>) -> Result<Self, E>
121     where
122         E: From<AllocError>,
123     {
124         let mut this = try_new_uninit!(Arc);
125         let Some(slot) = Arc::get_mut(&mut this) else {
126             // SAFETY: the Arc has just been created and has no external references
127             unsafe { core::hint::unreachable_unchecked() }
128         };
129         let slot = slot.as_mut_ptr();
130         // SAFETY: When init errors/panics, slot will get deallocated but not dropped,
131         // slot is valid.
132         unsafe { init.__init(slot)? };
133         // SAFETY: All fields have been initialized.
134         Ok(unsafe { this.assume_init() })
135     }
136 }
137 
138 impl<T> InPlaceWrite<T> for Box<MaybeUninit<T>> {
139     type Initialized = Box<T>;
140 
141     fn write_init<E>(mut self, init: impl Init<T, E>) -> Result<Self::Initialized, E> {
142         let slot = self.as_mut_ptr();
143         // SAFETY: When init errors/panics, slot will get deallocated but not dropped,
144         // slot is valid.
145         unsafe { init.__init(slot)? };
146         // SAFETY: All fields have been initialized.
147         Ok(unsafe { self.assume_init() })
148     }
149 
150     fn write_pin_init<E>(mut self, init: impl PinInit<T, E>) -> Result<Pin<Self::Initialized>, E> {
151         let slot = self.as_mut_ptr();
152         // SAFETY: When init errors/panics, slot will get deallocated but not dropped,
153         // slot is valid and will not be moved, because we pin it later.
154         unsafe { init.__pinned_init(slot)? };
155         // SAFETY: All fields have been initialized.
156         Ok(unsafe { self.assume_init() }.into())
157     }
158 }
159