| /* |
| * Copyright (c) 2024 Google Inc. All rights reserved |
| * |
| * Permission is hereby granted, free of charge, to any person obtaining |
| * a copy of this software and associated documentation files |
| * (the "Software"), to deal in the Software without restriction, |
| * including without limitation the rights to use, copy, modify, merge, |
| * publish, distribute, sublicense, and/or sell copies of the Software, |
| * and to permit persons to whom the Software is furnished to do so, |
| * subject to the following conditions: |
| * |
| * The above copyright notice and this permission notice shall be |
| * included in all copies or substantial portions of the Software. |
| * |
| * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, |
| * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF |
| * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. |
| * IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY |
| * CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, |
| * TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE |
| * SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. |
| */ |
| |
| /// Helper macro for container_of and container_of_mut. Exported so it can be |
| /// used by those macros and not meant to be used directly. |
| #[macro_export] |
| macro_rules! container_of_const_or_mut { |
| ($ptr:ident, $T:ty, $m:ident, $const_or_mut:ident) => {{ |
| // SAFETY: The caller must ensure that $ptr is a pointer to the $m |
| // field in an object of type $T. This means that $ptr came from |
| // addr_of!((*original_ptr).$m) so subtracting the offset of $m from |
| // $ptr will restore the original pointer. |
| let original_ptr = (($ptr).byte_sub(core::mem::offset_of!($T, $m)) as *$const_or_mut $T); |
| |
| // Check that type of $ptr matches type of $T.$m. This detects a |
| // subclass of bugs at compile time where the wrong field or pointer |
| // is passed and two types does not match. |
| // |
| // SAFETY: This should not generate any code. |
| let _always_true = core::ptr::addr_of!((*original_ptr).$m) == $ptr; |
| |
| original_ptr |
| }}; |
| } |
| |
| /// Get the pointer to a struct from a pointer to an embedded field. |
| /// Matches the C containerof define in include/shared/lk/macros.h. |
| /// Const version. |
| #[macro_export] |
| macro_rules! container_of { |
| ($ptr:ident, $T:ty, $m:ident) => { |
| $crate::container_of_const_or_mut!($ptr, $T, $m, const) |
| }; |
| } |
| |
| /// Get the pointer to a struct from a pointer to an embedded field. |
| /// Matches the C containerof define in include/shared/lk/macros.h. |
| /// Mutable version. |
| /// |
| /// To convert a pointer received by C code to a reference to a wrapping |
| /// rust struct a helper function could be used like so: |
| /// struct A {} |
| /// struct B { |
| /// a: A, |
| /// } |
| /// /// # SAFETY |
| /// /// |
| /// /// ptr_a must point to the a field in a B struct |
| /// unsafe fn ptr_a_to_ref_b<'a>(ptr_a: *mut A) -> &'a mut B { |
| /// unsafe { &mut *container_of_mut!(ptr_a, B, a) } |
| /// } |
| #[macro_export] |
| macro_rules! container_of_mut { |
| ($ptr:ident, $T:ty, $m:ident) => { |
| $crate::container_of_const_or_mut!($ptr, $T, $m, mut) |
| }; |
| } |