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}