1#[cfg(all(feature = "alloc", not(feature = "std")))]
4use alloc::{boxed::Box, sync::Arc};
5#[cfg(feature = "alloc")]
6use core::alloc::AllocError;
7use core::{mem::MaybeUninit, pin::Pin};
8#[cfg(feature = "std")]
9use std::sync::Arc;
10
11#[cfg(not(feature = "alloc"))]
12type AllocError = core::convert::Infallible;
13
14use crate::{
15    init_from_closure, pin_init_from_closure, InPlaceWrite, Init, PinInit, ZeroableOption,
16};
17
18pub extern crate alloc;
19
20unsafe impl<T> ZeroableOption for Box<T> {}
23
24pub trait InPlaceInit<T>: Sized {
26    fn try_pin_init<E>(init: impl PinInit<T, E>) -> Result<Pin<Self>, E>
31    where
32        E: From<AllocError>;
33
34    fn pin_init(init: impl PinInit<T>) -> Result<Pin<Self>, AllocError> {
39        let init = unsafe {
41            pin_init_from_closure(|slot| match init.__pinned_init(slot) {
42                Ok(()) => Ok(()),
43                Err(i) => match i {},
44            })
45        };
46        Self::try_pin_init(init)
47    }
48
49    fn try_init<E>(init: impl Init<T, E>) -> Result<Self, E>
51    where
52        E: From<AllocError>;
53
54    fn init(init: impl Init<T>) -> Result<Self, AllocError> {
56        let init = unsafe {
58            init_from_closure(|slot| match init.__init(slot) {
59                Ok(()) => Ok(()),
60                Err(i) => match i {},
61            })
62        };
63        Self::try_init(init)
64    }
65}
66
67#[cfg(feature = "alloc")]
68macro_rules! try_new_uninit {
69    ($type:ident) => {
70        $type::try_new_uninit()?
71    };
72}
73#[cfg(all(feature = "std", not(feature = "alloc")))]
74macro_rules! try_new_uninit {
75    ($type:ident) => {
76        $type::new_uninit()
77    };
78}
79
80impl<T> InPlaceInit<T> for Box<T> {
81    #[inline]
82    fn try_pin_init<E>(init: impl PinInit<T, E>) -> Result<Pin<Self>, E>
83    where
84        E: From<AllocError>,
85    {
86        try_new_uninit!(Box).write_pin_init(init)
87    }
88
89    #[inline]
90    fn try_init<E>(init: impl Init<T, E>) -> Result<Self, E>
91    where
92        E: From<AllocError>,
93    {
94        try_new_uninit!(Box).write_init(init)
95    }
96}
97
98impl<T> InPlaceInit<T> for Arc<T> {
99    #[inline]
100    fn try_pin_init<E>(init: impl PinInit<T, E>) -> Result<Pin<Self>, E>
101    where
102        E: From<AllocError>,
103    {
104        let mut this = try_new_uninit!(Arc);
105        let Some(slot) = Arc::get_mut(&mut this) else {
106            unsafe { core::hint::unreachable_unchecked() }
108        };
109        let slot = slot.as_mut_ptr();
110        unsafe { init.__pinned_init(slot)? };
113        Ok(unsafe { Pin::new_unchecked(this.assume_init()) })
115    }
116
117    #[inline]
118    fn try_init<E>(init: impl Init<T, E>) -> Result<Self, E>
119    where
120        E: From<AllocError>,
121    {
122        let mut this = try_new_uninit!(Arc);
123        let Some(slot) = Arc::get_mut(&mut this) else {
124            unsafe { core::hint::unreachable_unchecked() }
126        };
127        let slot = slot.as_mut_ptr();
128        unsafe { init.__init(slot)? };
131        Ok(unsafe { this.assume_init() })
133    }
134}
135
136impl<T> InPlaceWrite<T> for Box<MaybeUninit<T>> {
137    type Initialized = Box<T>;
138
139    fn write_init<E>(mut self, init: impl Init<T, E>) -> Result<Self::Initialized, E> {
140        let slot = self.as_mut_ptr();
141        unsafe { init.__init(slot)? };
144        Ok(unsafe { self.assume_init() })
146    }
147
148    fn write_pin_init<E>(mut self, init: impl PinInit<T, E>) -> Result<Pin<Self::Initialized>, E> {
149        let slot = self.as_mut_ptr();
150        unsafe { init.__pinned_init(slot)? };
153        Ok(unsafe { self.assume_init() }.into())
155    }
156}