//! These code are bring from redox-os, and I think it's a good idea to use it in our project. //! //! Helpers used to define types that are backed by integers (typically `usize`), //! without compromising safety. //! //! # Example //! //! ``` //! /// Define an opaque type `Pid` backed by a `usize`. //! int_like!(Pid, usize); //! //! const ZERO: Pid = Pid::from(0); //! ``` //! //! # Example //! //! ``` //! /// Define opaque types `Pid` and `AtomicPid`, backed respectively by a `usize` //! /// and a `AtomicUsize`. //! //! int_like!(Pid, AtomicPid, usize, AtomicUsize); //! //! const ZERO: Pid = Pid::from(0); //! let ATOMIC_PID: AtomicPid = AtomicPid::default(); //! ``` #[macro_export] macro_rules! int_like { ($new_type_name:ident, $backing_type: ident) => { #[derive(PartialEq, Eq, PartialOrd, Ord, Debug, Clone, Copy)] pub struct $new_type_name($backing_type); impl $new_type_name { #[allow(dead_code)] pub const fn into(self) -> $backing_type { self.0 } #[allow(dead_code)] pub const fn from(x: $backing_type) -> Self { $new_type_name(x) } #[allow(dead_code)] pub const fn new(x: $backing_type) -> Self { Self::from(x) } #[allow(dead_code)] pub const fn data(&self) -> $backing_type { self.0 } } }; ($new_type_name:ident, $new_atomic_type_name: ident, $backing_type:ident, $backing_atomic_type:ident) => { int_like!($new_type_name, $backing_type); /// A mutable holder for T that can safely be shared among threads. /// Runtime equivalent to using `AtomicUsize`, just type-safer. #[derive(Debug)] pub struct $new_atomic_type_name { container: $backing_atomic_type, } impl $new_atomic_type_name { #[allow(dead_code)] pub const fn new(x: $new_type_name) -> Self { $new_atomic_type_name { container: $backing_atomic_type::new(x.into()), } } #[allow(dead_code)] pub const fn default() -> Self { Self::new($new_type_name::from(0)) } #[allow(dead_code)] pub fn load(&self, order: ::core::sync::atomic::Ordering) -> $new_type_name { $new_type_name::from(self.container.load(order)) } #[allow(dead_code)] pub fn store(&self, val: $new_type_name, order: ::core::sync::atomic::Ordering) { self.container.store(val.into(), order) } #[allow(dead_code)] pub fn swap( &self, val: $new_type_name, order: ::core::sync::atomic::Ordering, ) -> $new_type_name { $new_type_name::from(self.container.swap(val.into(), order)) } #[allow(dead_code)] pub fn compare_exchange( &self, current: $new_type_name, new: $new_type_name, success: ::core::sync::atomic::Ordering, failure: ::core::sync::atomic::Ordering, ) -> ::core::result::Result<$new_type_name, $new_type_name> { match self .container .compare_exchange(current.into(), new.into(), success, failure) { Ok(result) => Ok($new_type_name::from(result)), Err(result) => Err($new_type_name::from(result)), } } #[allow(dead_code)] pub fn compare_exchange_weak( &self, current: $new_type_name, new: $new_type_name, success: ::core::sync::atomic::Ordering, failure: ::core::sync::atomic::Ordering, ) -> ::core::result::Result<$new_type_name, $new_type_name> { match self.container.compare_exchange_weak( current.into(), new.into(), success, failure, ) { Ok(result) => Ok($new_type_name::from(result)), Err(result) => Err($new_type_name::from(result)), } } #[allow(dead_code)] pub fn fetch_add( &self, val: $new_type_name, order: ::core::sync::atomic::Ordering, ) -> $new_type_name { $new_type_name::from(self.container.fetch_add(val.into(), order)) } } }; } #[test] fn test() { use ::core::sync::atomic::AtomicUsize; use core::mem::size_of; // Generate type `usize_like`. int_like!(UsizeLike, usize); assert_eq!(size_of::(), size_of::()); // Generate types `usize_like` and `AtomicUsize`. int_like!(UsizeLike2, AtomicUsizeLike, usize, AtomicUsize); assert_eq!(size_of::(), size_of::()); assert_eq!(size_of::(), size_of::()); }