int_like.rs 5.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151
  1. //! These code are bring from redox-os, and I think it's a good idea to use it in our project.
  2. //!
  3. //! Helpers used to define types that are backed by integers (typically `usize`),
  4. //! without compromising safety.
  5. //!
  6. //! # Example
  7. //!
  8. //! ```
  9. //! /// Define an opaque type `Pid` backed by a `usize`.
  10. //! int_like!(Pid, usize);
  11. //!
  12. //! const ZERO: Pid = Pid::from(0);
  13. //! ```
  14. //!
  15. //! # Example
  16. //!
  17. //! ```
  18. //! /// Define opaque types `Pid` and `AtomicPid`, backed respectively by a `usize`
  19. //! /// and a `AtomicUsize`.
  20. //!
  21. //! int_like!(Pid, AtomicPid, usize, AtomicUsize);
  22. //!
  23. //! const ZERO: Pid = Pid::from(0);
  24. //! let ATOMIC_PID: AtomicPid = AtomicPid::default();
  25. //! ```
  26. #[macro_export]
  27. macro_rules! int_like {
  28. ($new_type_name:ident, $backing_type: ident) => {
  29. #[derive(PartialEq, Eq, PartialOrd, Ord, Debug, Clone, Copy, Hash)]
  30. pub struct $new_type_name($backing_type);
  31. impl $new_type_name {
  32. #[allow(dead_code)]
  33. pub const fn into(self) -> $backing_type {
  34. self.0
  35. }
  36. #[allow(dead_code)]
  37. pub const fn from(x: $backing_type) -> Self {
  38. $new_type_name(x)
  39. }
  40. #[allow(dead_code)]
  41. pub const fn new(x: $backing_type) -> Self {
  42. Self::from(x)
  43. }
  44. #[allow(dead_code)]
  45. pub const fn data(&self) -> $backing_type {
  46. self.0
  47. }
  48. }
  49. };
  50. ($new_type_name:ident, $new_atomic_type_name: ident, $backing_type:ident, $backing_atomic_type:ident) => {
  51. int_like!($new_type_name, $backing_type);
  52. /// A mutable holder for T that can safely be shared among threads.
  53. /// Runtime equivalent to using `AtomicUsize`, just type-safer.
  54. #[derive(Debug)]
  55. pub struct $new_atomic_type_name {
  56. container: $backing_atomic_type,
  57. }
  58. impl $new_atomic_type_name {
  59. #[allow(dead_code)]
  60. pub const fn new(x: $new_type_name) -> Self {
  61. $new_atomic_type_name {
  62. container: $backing_atomic_type::new(x.into()),
  63. }
  64. }
  65. #[allow(dead_code)]
  66. pub const fn default() -> Self {
  67. Self::new($new_type_name::from(0))
  68. }
  69. #[allow(dead_code)]
  70. pub fn load(&self, order: ::core::sync::atomic::Ordering) -> $new_type_name {
  71. $new_type_name::from(self.container.load(order))
  72. }
  73. #[allow(dead_code)]
  74. pub fn store(&self, val: $new_type_name, order: ::core::sync::atomic::Ordering) {
  75. self.container.store(val.into(), order)
  76. }
  77. #[allow(dead_code)]
  78. pub fn swap(
  79. &self,
  80. val: $new_type_name,
  81. order: ::core::sync::atomic::Ordering,
  82. ) -> $new_type_name {
  83. $new_type_name::from(self.container.swap(val.into(), order))
  84. }
  85. #[allow(dead_code)]
  86. pub fn compare_exchange(
  87. &self,
  88. current: $new_type_name,
  89. new: $new_type_name,
  90. success: ::core::sync::atomic::Ordering,
  91. failure: ::core::sync::atomic::Ordering,
  92. ) -> ::core::result::Result<$new_type_name, $new_type_name> {
  93. match self
  94. .container
  95. .compare_exchange(current.into(), new.into(), success, failure)
  96. {
  97. Ok(result) => Ok($new_type_name::from(result)),
  98. Err(result) => Err($new_type_name::from(result)),
  99. }
  100. }
  101. #[allow(dead_code)]
  102. pub fn compare_exchange_weak(
  103. &self,
  104. current: $new_type_name,
  105. new: $new_type_name,
  106. success: ::core::sync::atomic::Ordering,
  107. failure: ::core::sync::atomic::Ordering,
  108. ) -> ::core::result::Result<$new_type_name, $new_type_name> {
  109. match self.container.compare_exchange_weak(
  110. current.into(),
  111. new.into(),
  112. success,
  113. failure,
  114. ) {
  115. Ok(result) => Ok($new_type_name::from(result)),
  116. Err(result) => Err($new_type_name::from(result)),
  117. }
  118. }
  119. #[allow(dead_code)]
  120. pub fn fetch_add(
  121. &self,
  122. val: $new_type_name,
  123. order: ::core::sync::atomic::Ordering,
  124. ) -> $new_type_name {
  125. $new_type_name::from(self.container.fetch_add(val.into(), order))
  126. }
  127. }
  128. };
  129. }
  130. #[test]
  131. fn test() {
  132. use ::core::sync::atomic::AtomicUsize;
  133. use core::mem::size_of;
  134. // Generate type `usize_like`.
  135. int_like!(UsizeLike, usize);
  136. assert_eq!(size_of::<UsizeLike>(), size_of::<usize>());
  137. // Generate types `usize_like` and `AtomicUsize`.
  138. int_like!(UsizeLike2, AtomicUsizeLike, usize, AtomicUsize);
  139. assert_eq!(size_of::<UsizeLike2>(), size_of::<usize>());
  140. assert_eq!(size_of::<AtomicUsizeLike>(), size_of::<AtomicUsize>());
  141. }