fidl_next_codec/
slot.rs

1// Copyright 2024 The Fuchsia Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
5use core::marker::PhantomData;
6use core::mem::MaybeUninit;
7use core::ops::{Deref, DerefMut};
8use core::ptr::slice_from_raw_parts_mut;
9use core::slice::from_raw_parts;
10
11use munge::{Destructure, Move, Restructure};
12use zerocopy::{FromBytes, IntoBytes};
13
14/// An initialized but potentially invalid value.
15///
16/// The bytes of a `Slot` are always valid to read, but may not represent a
17/// valid value of its type. For example, a `Slot<'_, bool>` may not be set to
18/// 0 or 1.
19#[repr(transparent)]
20pub struct Slot<'de, T: ?Sized> {
21    ptr: *mut T,
22    _phantom: PhantomData<&'de mut [u8]>,
23}
24
25unsafe impl<T: Send> Send for Slot<'_, T> {}
26unsafe impl<T: Sync> Sync for Slot<'_, T> {}
27
28impl<'de, T: ?Sized> Slot<'de, T> {
29    /// Returns a new `Slot` backed by the given `MaybeUninit`.
30    pub fn new(backing: &'de mut MaybeUninit<T>) -> Self
31    where
32        T: Sized,
33    {
34        unsafe {
35            backing.as_mut_ptr().write_bytes(0, 1);
36        }
37        unsafe { Self::new_unchecked(backing.as_mut_ptr()) }
38    }
39
40    /// Creates a new slot from the given pointer.
41    ///
42    /// # Safety
43    ///
44    /// `ptr` must point to enough initialized bytes with the correct alignment
45    /// to represent a `T`.
46    pub unsafe fn new_unchecked(ptr: *mut T) -> Self {
47        Self { ptr, _phantom: PhantomData }
48    }
49
50    /// Mutably reborrows the slot.
51    pub fn as_mut(&mut self) -> Slot<'_, T> {
52        Self { ptr: self.ptr, _phantom: PhantomData }
53    }
54
55    /// Returns a mutable pointer to the underlying potentially-invalid value.
56    pub fn as_mut_ptr(&mut self) -> *mut T {
57        self.ptr
58    }
59
60    /// Returns a pointer to the underlying potentially-invalid value.
61    pub fn as_ptr(&self) -> *const T {
62        self.ptr
63    }
64
65    /// Returns a reference to the contained value.
66    ///
67    /// # Safety
68    ///
69    /// The slot must contain a valid `T`.
70    pub unsafe fn deref_unchecked(&self) -> &T {
71        unsafe { &*self.as_ptr() }
72    }
73
74    /// Returns a mutable reference to the contained value.
75    ///
76    /// # Safety
77    ///
78    /// The slot must contain a valid `T`.
79    pub unsafe fn deref_mut_unchecked(&mut self) -> &mut T {
80        unsafe { &mut *self.as_mut_ptr() }
81    }
82
83    /// Writes the given value into the slot.
84    pub fn write(&mut self, value: T)
85    where
86        T: IntoBytes + Sized,
87    {
88        unsafe {
89            self.as_mut_ptr().write(value);
90        }
91    }
92}
93
94impl<'de, T: Sized> Slot<'de, T> {
95    /// Creates a new slot from the given backing storage.
96    ///
97    /// # Safety
98    ///
99    /// `backing` must actually be initialized with valid `T`.
100    pub unsafe fn new_unchecked_from_maybe_uninit(backing: &mut MaybeUninit<T>) -> Self {
101        Self { ptr: backing.as_mut_ptr(), _phantom: PhantomData }
102    }
103}
104
105impl<T> Slot<'_, T> {
106    /// Returns a slice of the underlying bytes.
107    pub fn as_bytes(&self) -> &[u8] {
108        unsafe { from_raw_parts(self.ptr.cast::<u8>(), size_of::<T>()) }
109    }
110}
111
112impl<T, const N: usize> Slot<'_, [T; N]> {
113    /// Returns a slot of the element at the given index.
114    pub fn index(&mut self, index: usize) -> Slot<'_, T> {
115        assert!(index < N, "attempted to index out-of-bounds");
116
117        Slot { ptr: unsafe { self.as_mut_ptr().cast::<T>().add(index) }, _phantom: PhantomData }
118    }
119}
120
121impl<T> Slot<'_, [T]> {
122    /// Creates a new slice slot from the given pointer.
123    ///
124    /// # Safety
125    ///
126    /// `ptr` must point to enough initialized bytes with the correct alignment
127    /// to represent a slice of `len` `T`s.
128    pub unsafe fn new_slice_unchecked(ptr: *mut T, len: usize) -> Self {
129        Self { ptr: slice_from_raw_parts_mut(ptr, len), _phantom: PhantomData }
130    }
131
132    /// Returns a slot of the element at the given index.
133    pub fn index(&mut self, index: usize) -> Slot<'_, T> {
134        assert!(index < self.ptr.len(), "attempted to index out-of-bounds");
135
136        Slot { ptr: unsafe { self.as_mut_ptr().cast::<T>().add(index) }, _phantom: PhantomData }
137    }
138}
139
140impl<T: FromBytes> Deref for Slot<'_, T> {
141    type Target = T;
142
143    fn deref(&self) -> &Self::Target {
144        unsafe { &*self.as_ptr() }
145    }
146}
147
148impl<T: FromBytes> DerefMut for Slot<'_, T> {
149    fn deref_mut(&mut self) -> &mut Self::Target {
150        unsafe { &mut *self.as_mut_ptr() }
151    }
152}
153
154impl<'de, T> Iterator for Slot<'de, [T]> {
155    type Item = Slot<'de, T>;
156
157    fn next(&mut self) -> Option<Self::Item> {
158        if self.ptr.len() == 0 {
159            return None;
160        }
161
162        let result = Slot { ptr: self.ptr.cast::<T>(), _phantom: PhantomData };
163
164        self.ptr =
165            slice_from_raw_parts_mut(unsafe { self.ptr.cast::<T>().add(1) }, self.ptr.len() - 1);
166
167        Some(result)
168    }
169}
170
171unsafe impl<T> Destructure for Slot<'_, T> {
172    type Underlying = T;
173    type Destructuring = Move;
174
175    fn underlying(&mut self) -> *mut Self::Underlying {
176        self.as_mut_ptr()
177    }
178}
179
180unsafe impl<'de, T, U: 'de> Restructure<U> for Slot<'de, T> {
181    type Restructured = Slot<'de, U>;
182
183    unsafe fn restructure(&self, ptr: *mut U) -> Self::Restructured {
184        Slot { ptr, _phantom: PhantomData }
185    }
186}