openthread/ot/types/
castable.rs

1// Copyright 2021 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
5/// Trait used to indicate that the implementing type can be efficiently
6/// converted into a reference to the original OpenThread type identified by
7/// `Self::OtType`.
8///
9/// Types which implement this trait may be opaque, but, unlike [`Boxable`], these
10/// types are not necessarily opaque and not necessarily ownable. If a type is
11/// not opaque then it will also implement the related trait [`Transparent`],
12/// allowing it to be used by value.
13///
14/// # Safety
15///
16/// This trait is unsafe because it is making an assertion about the
17/// data's representation that must be verified by code review.
18///
19/// In order to safely implement this trait you must verify that the implementing
20/// type is guaranteed to always be fundamentally identical in its in-memory
21/// representation to `<Self as OtCastable>::OtType`.
22///
23/// For example, this would be the case if `Self` is `#[repr(transparent)]` and
24/// has only a single member of `<Self as OtCastable>::OtType`.
25pub unsafe trait OtCastable: Sized {
26    /// Original OpenThread Type.
27    type OtType: Sized;
28
29    /// Returns a reference to the original OpenThread type [`Self::OtType`].
30    fn as_ot_ref(&self) -> &Self::OtType {
31        unsafe { &*self.as_ot_ptr() }
32    }
33
34    /// Returns a mutable reference to the original OpenThread type [`Self::OtType`].
35    fn as_ot_mut(&mut self) -> &mut Self::OtType {
36        unsafe { &mut *self.as_ot_mut_ptr() }
37    }
38
39    /// Returns a pointer to the underlying [`Self::OtType`] instance.
40    fn as_ot_ptr(&self) -> *const Self::OtType;
41
42    /// Returns a mutable pointer to the underlying [`Self::OtType`] instance.
43    fn as_ot_mut_ptr(&mut self) -> *mut Self::OtType;
44
45    /// Creates a reference from a pointer to an [`Self::OtType`].
46    ///
47    /// ## Safety ##
48    ///
49    /// This method is unsafe because unchecked conversion of pointers
50    /// to references is generally unsafe. The following assumptions
51    /// need to be verified to avoid undefined behavior:
52    ///
53    /// 1. `ptr` MUST point to a valid instance of [`Self::OtType`] that will
54    ///    remain valid over the given lifetime.
55    unsafe fn ref_from_ot_ptr<'a>(ptr: *const Self::OtType) -> Option<&'a Self>;
56
57    /// Creates a mut reference from a mut pointer to an [`Self::OtType`].
58    ///
59    /// ## Safety ##
60    ///
61    /// This method is unsafe because unchecked conversion of pointers
62    /// to references is generally unsafe. The following assumptions
63    /// need to be verified to avoid undefined behavior:
64    ///
65    /// 1. `ptr` MUST point to a valid instance of [`Self::OtType`] that will
66    ///    remain valid over the given lifetime.
67    unsafe fn mut_from_ot_mut_ptr<'a>(ptr: *mut Self::OtType) -> Option<&'a mut Self>;
68
69    /// Casts a reference to the original OpenThread type to a reference to `Self`.
70    fn ref_from_ot_ref(x: &Self::OtType) -> &Self {
71        unsafe { Self::ref_from_ot_ptr(x as *const Self::OtType) }.unwrap()
72    }
73}
74
75/// Trait used to indicate that the implementing type can be used by value
76/// and converted to/from the associated OpenThread type by value.
77///
78/// Unlike types that implement the trait [`Boxable`], types implementing
79/// this trait may be passed and used by value.
80pub trait Transparent: OtCastable + Clone {
81    /// Creates a new instance from an instance of [`Self::OtType`].
82    fn from_ot(x: Self::OtType) -> Self;
83
84    /// Converts this type into an instance of [`Self::OtType`].
85    fn into_ot(self) -> Self::OtType;
86}
87
88#[doc(hidden)]
89#[macro_export]
90macro_rules! impl_ot_castable {
91    (opaque from_only $wrapper:ty,$inner:ty) => {
92        impl<'a> From<&'a $inner> for &'a $wrapper {
93            fn from(x: &'a $inner) -> Self {
94                <$wrapper>::ref_from_ot_ref(x)
95            }
96        }
97
98        impl<'a> From<&'a $wrapper> for &'a $inner {
99            fn from(x: &'a $wrapper) -> Self {
100                x.as_ot_ref()
101            }
102        }
103    };
104
105    (from_only $wrapper:ty,$inner:ty) => {
106        impl_ot_castable!(opaque from_only $wrapper, $inner);
107
108        impl From<$inner> for $wrapper {
109            fn from(x: $inner) -> Self {
110                Self(x)
111            }
112        }
113
114        impl From<$wrapper> for $inner {
115            fn from(x: $wrapper) -> Self {
116                x.0
117            }
118        }
119    };
120
121    ($wrapper:ty,$inner:ty) => {
122        impl_ot_castable!(lifetime $wrapper, $inner);
123
124        impl_ot_castable!(from_only $wrapper, $inner);
125    };
126
127    (opaque $wrapper:ty,$inner:ty) => {
128        impl_ot_castable!(opaque lifetime $wrapper, $inner);
129
130        impl_ot_castable!(opaque from_only $wrapper, $inner);
131    };
132
133    (lifetime $wrapper:ty,$inner:ty) => {
134        impl_ot_castable!(lifetime $wrapper, $inner,);
135    };
136
137    (opaque lifetime $wrapper:ty,$inner:ty) => {
138        impl_ot_castable!(opaque lifetime $wrapper, $inner,);
139    };
140
141    (opaque lifetime $wrapper:ty,$inner:ty, $($other:expr),*) => {
142        // Note that we don't implement the PointerSafe trait on
143        // types with lifetimes.
144        unsafe impl $crate::ot::OtCastable for $wrapper {
145            type OtType = $inner;
146
147            fn as_ot_ptr(&self) -> *const Self::OtType {
148                &self.0 as *const Self::OtType
149            }
150
151            fn as_ot_mut_ptr(&mut self) -> *mut Self::OtType {
152                &mut self.0 as *mut Self::OtType
153            }
154
155            unsafe fn ref_from_ot_ptr<'a>(ptr: *const Self::OtType) -> Option<&'a Self> {
156                static_assertions::assert_eq_size!($wrapper, $inner);
157                static_assertions::assert_eq_align!($wrapper, $inner);
158                if ptr.is_null() {
159                    None
160                } else {
161                    Some(&*(ptr as *const Self))
162                }
163            }
164
165            unsafe fn mut_from_ot_mut_ptr<'a>(ptr: *mut Self::OtType) -> Option<&'a mut Self> {
166                static_assertions::assert_eq_size!($wrapper, $inner);
167                static_assertions::assert_eq_align!($wrapper, $inner);
168                if ptr.is_null() {
169                    None
170                } else {
171                    Some(&mut *(ptr as *mut Self))
172                }
173            }
174        }
175    };
176
177    (lifetime $wrapper:ty,$inner:ty, $($other:expr),*) => {
178        impl_ot_castable!(opaque lifetime $wrapper, $inner, $($other),*);
179        // Note that we don't implement the PointerSafe trait on
180        // types with lifetimes.
181        impl $crate::ot::Transparent for $wrapper {
182            fn from_ot(x: Self::OtType) -> Self {
183                Self(x,$($other),*)
184            }
185
186            fn into_ot(self) -> Self::OtType {
187                self.0
188            }
189        }
190    };
191}