wiwi/chain/
array.rs

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
use crate::prelude::*;
use super::{ Chain as _, ChainInner as _ };

super::decl_chain! {
	generics_decl: [T, const N: usize]
	generics_decl_struct_def: [T, const N: usize]
	generics: [T, N]
	chain: ArrayChain
	inner: [T; N]
}

impl<T, const N: usize> ArrayChain<T, N> {
	#[inline]
	pub fn new_uninit() -> ArrayChain<MaybeUninit<T>, N> {
		// SAFETY: `MaybeUninit` has no initialisation requirement, so
		// `[MaybeUninit<T>; N]` is always valid
		unsafe { ArrayChain::from_inner(MaybeUninit::uninit().assume_init()) }
	}

	#[inline]
	pub fn new_zeroed() -> ArrayChain<MaybeUninit<T>, N> {
		// SAFETY: `MaybeUninit` has no initialisation requirement, so
		// `[MaybeUninit<T>; N]` is always valid
		unsafe { ArrayChain::from_inner(MaybeUninit::zeroed().assume_init()) }
	}
}

impl<T, const N: usize> ArrayChain<MaybeUninit<T>, N> {
	/// Assumes all slots inside the array are initialised according to `T`'s
	/// requirements, and converts into an array of T
	///
	/// Note: this implementation is currently subpar, as it does fully copy `self`
	/// into a new container using `ptr::read`. We have to do this because, at the
	/// time of writing:
	///
	/// - `transmute` is a bit too dumb, and is not able to prove `[T; N]` and
	///   `[MaybeUninit<T>; N]` are guaranteed to be equal sized, even though
	///   we can see and prove it
	/// - `transmute_unchecked` is like `transmute` but without that compile time
	///   size check, but it is unstable, and according to a code comment will
	///   almost certainly never be stabilised (reasoning is that it's too unsafe,
	///   too much power to give users :p, and to hopefully find other methods for
	///   doing things without it so that it isn't needed)
	/// - `MaybeUninit::array_assume_init` is unstable (it internally makes use of
	///   `transmute_unchecked`)
	///
	/// We don't know of any other option than to perform a ptr cast,
	/// then read from it.
	///
	/// # Safety
	///
	/// All slots in `self` must be fully initialised with valid values of `T`.
	#[inline]
	pub unsafe fn assume_init(self) -> ArrayChain<T, N> {
		#[expect(clippy::as_conversions, reason = "ptr cast")]
		let ptr = self.as_inner() as *const [MaybeUninit<T>; N] as *const [T; N];

		// SAFETY: `ptr` is obtained from `self`, performed valid cast from array
		// of MaybeUninit to array of T, and caller promises that all slots in the
		// array are properly initialised values of `T`. Also see doc comment on
		// this function for why this is a ptr read rather than a transmute
		unsafe { ptr.read().into_chain() }
	}
}