4
0

entry.rs 6.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205
  1. use alloc::boxed::Box;
  2. use alloc::sync::Arc;
  3. use core::marker::PhantomData;
  4. use core::mem::ManuallyDrop;
  5. use super::*;
  6. /// A trait that should be implemented for the types users wish to store in an `XArray`.
  7. /// Items stored in an XArray are required to be 8 bytes in size, Currently it can be various pointer types.
  8. ///
  9. /// # Safety
  10. /// Users must ensure that the produced `usize` of `into_raw()` meets the requirements for an item entry in the XArray. Specifically,
  11. /// if the original type is a pointer, the last two bits should be 00; if the original
  12. /// type is a value like usize, the last bit should be 1 (TODO).
  13. pub unsafe trait ItemEntry {
  14. /// Converts the original type into a `usize`, consuming the ownership of the original type.
  15. ///
  16. /// This `usize` should be directly stored in an XArray's XEntry.
  17. fn into_raw(self) -> usize;
  18. /// Recovers the original type from a usize, reclaiming ownership.
  19. ///
  20. /// # Safety
  21. /// The raw value passed must have been produced by the corresponding `into_raw` method in this trait
  22. /// from the same type.
  23. unsafe fn from_raw(raw: usize) -> Self;
  24. }
  25. unsafe impl<T> ItemEntry for Arc<T> {
  26. fn into_raw(self) -> usize {
  27. let raw_ptr = unsafe { core::intrinsics::transmute::<Arc<T>, *const u8>(self) };
  28. debug_assert!(raw_ptr.is_aligned_to(4));
  29. raw_ptr as usize
  30. }
  31. unsafe fn from_raw(raw: usize) -> Self {
  32. let arc = core::intrinsics::transmute::<usize, Arc<T>>(raw);
  33. arc
  34. }
  35. }
  36. unsafe impl<T> ItemEntry for Box<T> {
  37. fn into_raw(self) -> usize {
  38. let raw_ptr = Box::into_raw(self) as *const u8;
  39. debug_assert!(raw_ptr.is_aligned_to(4));
  40. raw_ptr as usize
  41. }
  42. unsafe fn from_raw(raw: usize) -> Self {
  43. Box::from_raw(raw as *mut _)
  44. }
  45. }
  46. /// The type stored in the head of `XArray` and the slots of `XNode`s, which is the basic unit of storage within an XArray.
  47. /// There are the following types of `XEntry`:
  48. /// - Internal entries: These are invisible to users and have the last two bits set to 10. Currently `XArray` only have node
  49. /// entries as internal entries, which are entries that point to XNodes.
  50. /// - Item entries: Items stored by the user. Currently stored items can only be pointers and the last two bits
  51. /// of these item entries are 00.
  52. ///
  53. /// `XEntry` have the ownership. Once it generated from an item or a XNode, the ownership of the item or the XNode
  54. /// will be transferred to the `XEntry`. If the stored item in the XArray implemented Clone trait, then the XEntry
  55. /// in the XArray can also implement Clone trait.
  56. #[derive(Eq, Debug)]
  57. pub(super) struct XEntry<I, L>
  58. where
  59. I: ItemEntry,
  60. L: XLock,
  61. {
  62. raw: usize,
  63. _marker: core::marker::PhantomData<(I, L)>,
  64. }
  65. impl<I: ItemEntry, L: XLock> Drop for XEntry<I, L> {
  66. fn drop(&mut self) {
  67. if self.is_item() {
  68. unsafe {
  69. I::from_raw(self.raw);
  70. }
  71. }
  72. if self.is_node() {
  73. unsafe {
  74. Arc::from_raw((self.raw - 2) as *const XNode<I, L>);
  75. }
  76. }
  77. }
  78. }
  79. impl<I: ItemEntry + Clone, L: XLock> Clone for XEntry<I, L> {
  80. fn clone(&self) -> Self {
  81. if self.is_item() {
  82. let cloned_entry = unsafe {
  83. let item_entry = ManuallyDrop::new(I::from_raw(self.raw));
  84. XEntry::from_item((*item_entry).clone())
  85. };
  86. cloned_entry
  87. } else {
  88. if self.is_node() {
  89. unsafe {
  90. Arc::increment_strong_count((self.raw - 2) as *const XNode<I, L>);
  91. }
  92. }
  93. Self {
  94. raw: self.raw,
  95. _marker: core::marker::PhantomData,
  96. }
  97. }
  98. }
  99. }
  100. impl<I: ItemEntry, L: XLock> PartialEq for XEntry<I, L> {
  101. fn eq(&self, o: &Self) -> bool {
  102. self.raw == o.raw
  103. }
  104. }
  105. impl<I: ItemEntry, L: XLock> XEntry<I, L> {
  106. pub fn raw(&self) -> usize {
  107. self.raw
  108. }
  109. pub const EMPTY: Self = unsafe { Self::new(0) };
  110. pub const unsafe fn new(raw: usize) -> Self {
  111. Self {
  112. raw,
  113. _marker: PhantomData,
  114. }
  115. }
  116. pub fn is_null(&self) -> bool {
  117. self.raw == 0
  118. }
  119. pub fn is_internal(&self) -> bool {
  120. self.raw & 3 == 2
  121. }
  122. pub fn is_item(&self) -> bool {
  123. !self.is_null() && !self.is_internal()
  124. }
  125. pub fn is_node(&self) -> bool {
  126. self.is_internal() && self.raw > (SLOT_SIZE << 2)
  127. }
  128. pub fn from_item(item: I) -> Self {
  129. let raw = I::into_raw(item);
  130. unsafe { Self::new(raw as usize) }
  131. }
  132. pub fn into_item(self) -> Option<I> {
  133. if self.is_item() {
  134. let item = unsafe { I::from_raw(self.raw) };
  135. core::mem::forget(self);
  136. Some(item)
  137. } else {
  138. None
  139. }
  140. }
  141. pub fn from_node<Operation>(node: XNode<I, L, Operation>) -> Self {
  142. let node_ptr = {
  143. let arc_node = Arc::new(node);
  144. Arc::into_raw(arc_node)
  145. };
  146. unsafe { Self::new(node_ptr as usize | 2) }
  147. }
  148. pub fn as_node(&self) -> Option<&XNode<I, L>> {
  149. if self.is_node() {
  150. unsafe {
  151. let node_ref = &*((self.raw - 2) as *const XNode<I, L>);
  152. Some(node_ref)
  153. }
  154. } else {
  155. None
  156. }
  157. }
  158. pub fn as_node_mut<'a>(&self) -> Option<&'a XNode<I, L, ReadWrite>> {
  159. if self.is_node() {
  160. unsafe {
  161. let node_ref = &*((self.raw - 2) as *const XNode<I, L, ReadWrite>);
  162. Some(node_ref)
  163. }
  164. } else {
  165. None
  166. }
  167. }
  168. pub fn node_strong_count(&self) -> Option<usize> {
  169. if self.is_node() {
  170. let raw_ptr = (self.raw - 2) as *const u8;
  171. unsafe {
  172. let arc = Arc::from_raw(raw_ptr);
  173. let strong_count = Arc::strong_count(&arc);
  174. core::mem::forget(arc);
  175. Some(strong_count)
  176. }
  177. } else {
  178. None
  179. }
  180. }
  181. }