From 072a239505f20e3f41c69fe579ca1b7c96731b7a Mon Sep 17 00:00:00 2001 From: Jacob Hughes Date: Thu, 10 Mar 2022 04:55:24 -0500 Subject: [PATCH 1/4] BTreeMap: Add alloc param --- library/alloc/src/collections/btree/append.rs | 20 +- library/alloc/src/collections/btree/fix.rs | 54 +-- library/alloc/src/collections/btree/map.rs | 386 ++++++++++++------ .../alloc/src/collections/btree/map/entry.rs | 68 +-- .../alloc/src/collections/btree/map/tests.rs | 6 +- .../alloc/src/collections/btree/navigate.rs | 57 ++- library/alloc/src/collections/btree/node.rs | 99 +++-- .../alloc/src/collections/btree/node/tests.rs | 13 +- library/alloc/src/collections/btree/remove.rs | 22 +- library/alloc/src/collections/btree/set.rs | 266 ++++++++---- library/alloc/src/collections/btree/split.rs | 15 +- 11 files changed, 669 insertions(+), 337 deletions(-) diff --git a/library/alloc/src/collections/btree/append.rs b/library/alloc/src/collections/btree/append.rs index a30a21db5347f..5434ffbc3cb61 100644 --- a/library/alloc/src/collections/btree/append.rs +++ b/library/alloc/src/collections/btree/append.rs @@ -1,5 +1,6 @@ use super::merge_iter::MergeIterInner; use super::node::{self, Root}; +use core::alloc::Allocator; use core::iter::FusedIterator; impl Root { @@ -14,8 +15,13 @@ impl Root { /// a `BTreeMap`, both iterators should produce keys in strictly ascending /// order, each greater than all keys in the tree, including any keys /// already in the tree upon entry. - pub fn append_from_sorted_iters(&mut self, left: I, right: I, length: &mut usize) - where + pub fn append_from_sorted_iters( + &mut self, + left: I, + right: I, + length: &mut usize, + alloc: &A, + ) where K: Ord, I: Iterator + FusedIterator, { @@ -23,13 +29,13 @@ impl Root { let iter = MergeIter(MergeIterInner::new(left, right)); // Meanwhile, we build a tree from the sorted sequence in linear time. - self.bulk_push(iter, length) + self.bulk_push(iter, length, alloc) } /// Pushes all key-value pairs to the end of the tree, incrementing a /// `length` variable along the way. The latter makes it easier for the /// caller to avoid a leak when the iterator panicks. - pub fn bulk_push(&mut self, iter: I, length: &mut usize) + pub fn bulk_push(&mut self, iter: I, length: &mut usize, alloc: &A) where I: Iterator, { @@ -58,7 +64,7 @@ impl Root { } Err(_) => { // We are at the top, create a new root node and push there. - open_node = self.push_internal_level(); + open_node = self.push_internal_level(alloc); break; } } @@ -66,9 +72,9 @@ impl Root { // Push key-value pair and new right subtree. let tree_height = open_node.height() - 1; - let mut right_tree = Root::new(); + let mut right_tree = Root::new(alloc); for _ in 0..tree_height { - right_tree.push_internal_level(); + right_tree.push_internal_level(alloc); } open_node.push(key, value, right_tree); diff --git a/library/alloc/src/collections/btree/fix.rs b/library/alloc/src/collections/btree/fix.rs index c4861817dd05d..a9b512252ceee 100644 --- a/library/alloc/src/collections/btree/fix.rs +++ b/library/alloc/src/collections/btree/fix.rs @@ -1,13 +1,15 @@ use super::map::MIN_LEN; use super::node::{marker, ForceResult::*, Handle, LeftOrRight::*, NodeRef, Root}; +use core::alloc::Allocator; impl<'a, K: 'a, V: 'a> NodeRef, K, V, marker::LeafOrInternal> { /// Stocks up a possibly underfull node by merging with or stealing from a /// sibling. If successful but at the cost of shrinking the parent node, /// returns that shrunk parent node. Returns an `Err` if the node is /// an empty root. - fn fix_node_through_parent( + fn fix_node_through_parent( self, + alloc: &A, ) -> Result, K, V, marker::Internal>>, Self> { let len = self.len(); if len >= MIN_LEN { @@ -16,7 +18,7 @@ impl<'a, K: 'a, V: 'a> NodeRef, K, V, marker::LeafOrInternal> { match self.choose_parent_kv() { Ok(Left(mut left_parent_kv)) => { if left_parent_kv.can_merge() { - let parent = left_parent_kv.merge_tracking_parent(); + let parent = left_parent_kv.merge_tracking_parent(alloc); Ok(Some(parent)) } else { left_parent_kv.bulk_steal_left(MIN_LEN - len); @@ -25,7 +27,7 @@ impl<'a, K: 'a, V: 'a> NodeRef, K, V, marker::LeafOrInternal> { } Ok(Right(mut right_parent_kv)) => { if right_parent_kv.can_merge() { - let parent = right_parent_kv.merge_tracking_parent(); + let parent = right_parent_kv.merge_tracking_parent(alloc); Ok(Some(parent)) } else { right_parent_kv.bulk_steal_right(MIN_LEN - len); @@ -52,9 +54,9 @@ impl<'a, K: 'a, V: 'a> NodeRef, K, V, marker::LeafOrInternal> { /// /// This method does not expect ancestors to already be underfull upon entry /// and panics if it encounters an empty ancestor. - pub fn fix_node_and_affected_ancestors(mut self) -> bool { + pub fn fix_node_and_affected_ancestors(mut self, alloc: &A) -> bool { loop { - match self.fix_node_through_parent() { + match self.fix_node_through_parent(alloc) { Ok(Some(parent)) => self = parent.forget_type(), Ok(None) => return true, Err(_) => return false, @@ -65,29 +67,29 @@ impl<'a, K: 'a, V: 'a> NodeRef, K, V, marker::LeafOrInternal> { impl Root { /// Removes empty levels on the top, but keeps an empty leaf if the entire tree is empty. - pub fn fix_top(&mut self) { + pub fn fix_top(&mut self, alloc: &A) { while self.height() > 0 && self.len() == 0 { - self.pop_internal_level(); + self.pop_internal_level(alloc); } } /// Stocks up or merge away any underfull nodes on the right border of the /// tree. The other nodes, those that are not the root nor a rightmost edge, /// must already have at least MIN_LEN elements. - pub fn fix_right_border(&mut self) { - self.fix_top(); + pub fn fix_right_border(&mut self, alloc: &A) { + self.fix_top(alloc); if self.len() > 0 { - self.borrow_mut().last_kv().fix_right_border_of_right_edge(); - self.fix_top(); + self.borrow_mut().last_kv().fix_right_border_of_right_edge(alloc); + self.fix_top(alloc); } } /// The symmetric clone of `fix_right_border`. - pub fn fix_left_border(&mut self) { - self.fix_top(); + pub fn fix_left_border(&mut self, alloc: &A) { + self.fix_top(alloc); if self.len() > 0 { - self.borrow_mut().first_kv().fix_left_border_of_left_edge(); - self.fix_top(); + self.borrow_mut().first_kv().fix_left_border_of_left_edge(alloc); + self.fix_top(alloc); } } @@ -113,16 +115,16 @@ impl Root { } impl<'a, K: 'a, V: 'a> Handle, K, V, marker::LeafOrInternal>, marker::KV> { - fn fix_left_border_of_left_edge(mut self) { + fn fix_left_border_of_left_edge(mut self, alloc: &A) { while let Internal(internal_kv) = self.force() { - self = internal_kv.fix_left_child().first_kv(); + self = internal_kv.fix_left_child(alloc).first_kv(); debug_assert!(self.reborrow().into_node().len() > MIN_LEN); } } - fn fix_right_border_of_right_edge(mut self) { + fn fix_right_border_of_right_edge(mut self, alloc: &A) { while let Internal(internal_kv) = self.force() { - self = internal_kv.fix_right_child().last_kv(); + self = internal_kv.fix_right_child(alloc).last_kv(); debug_assert!(self.reborrow().into_node().len() > MIN_LEN); } } @@ -133,12 +135,15 @@ impl<'a, K: 'a, V: 'a> Handle, K, V, marker::Internal>, /// provisions an extra element to allow merging its children in turn /// without becoming underfull. /// Returns the left child. - fn fix_left_child(self) -> NodeRef, K, V, marker::LeafOrInternal> { + fn fix_left_child( + self, + alloc: &A, + ) -> NodeRef, K, V, marker::LeafOrInternal> { let mut internal_kv = self.consider_for_balancing(); let left_len = internal_kv.left_child_len(); debug_assert!(internal_kv.right_child_len() >= MIN_LEN); if internal_kv.can_merge() { - internal_kv.merge_tracking_child() + internal_kv.merge_tracking_child(alloc) } else { // `MIN_LEN + 1` to avoid readjust if merge happens on the next level. let count = (MIN_LEN + 1).saturating_sub(left_len); @@ -153,12 +158,15 @@ impl<'a, K: 'a, V: 'a> Handle, K, V, marker::Internal>, /// provisions an extra element to allow merging its children in turn /// without becoming underfull. /// Returns wherever the right child ended up. - fn fix_right_child(self) -> NodeRef, K, V, marker::LeafOrInternal> { + fn fix_right_child( + self, + alloc: &A, + ) -> NodeRef, K, V, marker::LeafOrInternal> { let mut internal_kv = self.consider_for_balancing(); let right_len = internal_kv.right_child_len(); debug_assert!(internal_kv.left_child_len() >= MIN_LEN); if internal_kv.can_merge() { - internal_kv.merge_tracking_child() + internal_kv.merge_tracking_child(alloc) } else { // `MIN_LEN + 1` to avoid readjust if merge happens on the next level. let count = (MIN_LEN + 1).saturating_sub(right_len); diff --git a/library/alloc/src/collections/btree/map.rs b/library/alloc/src/collections/btree/map.rs index c178d3e3b03c9..10a2f37d03743 100644 --- a/library/alloc/src/collections/btree/map.rs +++ b/library/alloc/src/collections/btree/map.rs @@ -9,6 +9,8 @@ use core::mem::{self, ManuallyDrop}; use core::ops::{Index, RangeBounds}; use core::ptr; +use crate::alloc::{Allocator, Global}; + use super::borrow::DormantMutRef; use super::dedup_sorted_iter::DedupSortedIter; use super::navigate::{LazyLeafRange, LeafRange}; @@ -163,31 +165,41 @@ pub(super) const MIN_LEN: usize = node::MIN_LEN_AFTER_SPLIT; #[stable(feature = "rust1", since = "1.0.0")] #[cfg_attr(not(test), rustc_diagnostic_item = "BTreeMap")] #[rustc_insignificant_dtor] -pub struct BTreeMap { +pub struct BTreeMap< + K, + V, + #[unstable(feature = "allocator_api", issue = "32838")] A: Allocator = Global, +> { root: Option>, length: usize, + pub(super) alloc: ManuallyDrop, } #[stable(feature = "btree_drop", since = "1.7.0")] -unsafe impl<#[may_dangle] K, #[may_dangle] V> Drop for BTreeMap { +unsafe impl<#[may_dangle] K, #[may_dangle] V, A: Allocator> Drop for BTreeMap { fn drop(&mut self) { drop(unsafe { ptr::read(self) }.into_iter()) } } #[stable(feature = "rust1", since = "1.0.0")] -impl Clone for BTreeMap { - fn clone(&self) -> BTreeMap { - fn clone_subtree<'a, K: Clone, V: Clone>( +impl Clone for BTreeMap { + fn clone(&self) -> BTreeMap { + fn clone_subtree<'a, K: Clone, V: Clone, A: Clone + Allocator>( node: NodeRef, K, V, marker::LeafOrInternal>, - ) -> BTreeMap + alloc: &A, + ) -> BTreeMap where K: 'a, V: 'a, { match node.force() { Leaf(leaf) => { - let mut out_tree = BTreeMap { root: Some(Root::new()), length: 0 }; + let mut out_tree = BTreeMap { + root: Some(Root::new(alloc)), + length: 0, + alloc: ManuallyDrop::new((*alloc).clone()), + }; { let root = out_tree.root.as_mut().unwrap(); // unwrap succeeds because we just wrapped @@ -209,11 +221,11 @@ impl Clone for BTreeMap { out_tree } Internal(internal) => { - let mut out_tree = clone_subtree(internal.first_edge().descend()); + let mut out_tree = clone_subtree(internal.first_edge().descend(), alloc); { let out_root = out_tree.root.as_mut().unwrap(); - let mut out_node = out_root.push_internal_level(); + let mut out_node = out_root.push_internal_level(alloc); let mut in_edge = internal.first_edge(); while let Ok(kv) = in_edge.right_kv() { let (k, v) = kv.into_kv(); @@ -221,7 +233,7 @@ impl Clone for BTreeMap { let k = (*k).clone(); let v = (*v).clone(); - let subtree = clone_subtree(in_edge.descend()); + let subtree = clone_subtree(in_edge.descend(), alloc); // We can't destructure subtree directly // because BTreeMap implements Drop @@ -232,7 +244,7 @@ impl Clone for BTreeMap { (root, length) }; - out_node.push(k, v, subroot.unwrap_or_else(Root::new)); + out_node.push(k, v, subroot.unwrap_or_else(|| Root::new(alloc))); out_tree.length += 1 + sublength; } } @@ -243,14 +255,14 @@ impl Clone for BTreeMap { } if self.is_empty() { - BTreeMap::new() + BTreeMap::new_in(ManuallyDrop::into_inner(self.alloc.clone())) } else { - clone_subtree(self.root.as_ref().unwrap().reborrow()) // unwrap succeeds because not empty + clone_subtree(self.root.as_ref().unwrap().reborrow(), &*self.alloc) // unwrap succeeds because not empty } } } -impl super::Recover for BTreeMap +impl super::Recover for BTreeMap where K: Borrow + Ord, Q: Ord, @@ -269,21 +281,29 @@ where let (map, dormant_map) = DormantMutRef::new(self); let root_node = map.root.as_mut()?.borrow_mut(); match root_node.search_tree(key) { - Found(handle) => { - Some(OccupiedEntry { handle, dormant_map, _marker: PhantomData }.remove_kv().0) - } + Found(handle) => Some( + OccupiedEntry { handle, dormant_map, alloc: &*map.alloc, _marker: PhantomData } + .remove_kv() + .0, + ), GoDown(_) => None, } } fn replace(&mut self, key: K) -> Option { let (map, dormant_map) = DormantMutRef::new(self); - let root_node = map.root.get_or_insert_with(Root::new).borrow_mut(); + let root_node = map.root.get_or_insert_with(|| Root::new(&*map.alloc)).borrow_mut(); match root_node.search_tree::(&key) { Found(mut kv) => Some(mem::replace(kv.key_mut(), key)), GoDown(handle) => { - VacantEntry { key, handle: Some(handle), dormant_map, _marker: PhantomData } - .insert(()); + VacantEntry { + key, + handle: Some(handle), + dormant_map, + alloc: &*map.alloc, + _marker: PhantomData, + } + .insert(()); None } } @@ -343,12 +363,17 @@ impl fmt::Debug for IterMut<'_, K, V> { /// [`IntoIterator`]: core::iter::IntoIterator #[stable(feature = "rust1", since = "1.0.0")] #[rustc_insignificant_dtor] -pub struct IntoIter { +pub struct IntoIter< + K, + V, + #[unstable(feature = "allocator_api", issue = "32838")] A: Allocator = Global, +> { range: LazyLeafRange, length: usize, + alloc: A, } -impl IntoIter { +impl IntoIter { /// Returns an iterator of references over the remaining items. #[inline] pub(super) fn iter(&self) -> Iter<'_, K, V> { @@ -357,7 +382,7 @@ impl IntoIter { } #[stable(feature = "collection_debug", since = "1.17.0")] -impl fmt::Debug for IntoIter { +impl Debug for IntoIter { fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { f.debug_list().entries(self.iter()).finish() } @@ -371,7 +396,7 @@ impl fmt::Debug for IntoIter { /// [`keys`]: BTreeMap::keys #[must_use = "iterators are lazy and do nothing unless consumed"] #[stable(feature = "rust1", since = "1.0.0")] -pub struct Keys<'a, K: 'a, V: 'a> { +pub struct Keys<'a, K, V> { inner: Iter<'a, K, V>, } @@ -390,7 +415,7 @@ impl fmt::Debug for Keys<'_, K, V> { /// [`values`]: BTreeMap::values #[must_use = "iterators are lazy and do nothing unless consumed"] #[stable(feature = "rust1", since = "1.0.0")] -pub struct Values<'a, K: 'a, V: 'a> { +pub struct Values<'a, K, V> { inner: Iter<'a, K, V>, } @@ -409,7 +434,7 @@ impl fmt::Debug for Values<'_, K, V> { /// [`values_mut`]: BTreeMap::values_mut #[must_use = "iterators are lazy and do nothing unless consumed"] #[stable(feature = "map_values_mut", since = "1.10.0")] -pub struct ValuesMut<'a, K: 'a, V: 'a> { +pub struct ValuesMut<'a, K, V> { inner: IterMut<'a, K, V>, } @@ -428,12 +453,12 @@ impl fmt::Debug for ValuesMut<'_, K, V> { /// [`into_keys`]: BTreeMap::into_keys #[must_use = "iterators are lazy and do nothing unless consumed"] #[stable(feature = "map_into_keys_values", since = "1.54.0")] -pub struct IntoKeys { - inner: IntoIter, +pub struct IntoKeys { + inner: IntoIter, } #[stable(feature = "map_into_keys_values", since = "1.54.0")] -impl fmt::Debug for IntoKeys { +impl fmt::Debug for IntoKeys { fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { f.debug_list().entries(self.inner.iter().map(|(key, _)| key)).finish() } @@ -447,12 +472,16 @@ impl fmt::Debug for IntoKeys { /// [`into_values`]: BTreeMap::into_values #[must_use = "iterators are lazy and do nothing unless consumed"] #[stable(feature = "map_into_keys_values", since = "1.54.0")] -pub struct IntoValues { - inner: IntoIter, +pub struct IntoValues< + K, + V, + #[unstable(feature = "allocator_api", issue = "32838")] A: Allocator = Global, +> { + inner: IntoIter, } #[stable(feature = "map_into_keys_values", since = "1.54.0")] -impl fmt::Debug for IntoValues { +impl fmt::Debug for IntoValues { fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { f.debug_list().entries(self.inner.iter().map(|(_, val)| val)).finish() } @@ -521,9 +550,11 @@ impl BTreeMap { #[rustc_const_unstable(feature = "const_btree_new", issue = "71835")] #[must_use] pub const fn new() -> BTreeMap { - BTreeMap { root: None, length: 0 } + BTreeMap { root: None, length: 0, alloc: ManuallyDrop::new(Global) } } +} +impl BTreeMap { /// Clears the map, removing all elements. /// /// # Examples @@ -540,9 +571,37 @@ impl BTreeMap { /// ``` #[stable(feature = "rust1", since = "1.0.0")] pub fn clear(&mut self) { - *self = BTreeMap::new(); + let alloc = unsafe { + // drop all elements and retrieve allocator + ptr::read(self).into_iter().into_alloc() + }; + *self = BTreeMap::new_in(alloc); } + /// Makes a new empty BTreeMap with a reasonable choice for B. + /// + /// # Examples + /// + /// Basic usage: + /// + /// ``` + /// # #![feature(allocator_api)] + /// # #![feature(btreemap_alloc)] + /// use std::collections::BTreeMap; + /// use std::alloc::Global; + /// + /// let mut map = BTreeMap::new_in(Global); + /// + /// // entries can now be inserted into the empty map + /// map.insert(1, "a"); + /// ``` + #[unstable(feature = "btreemap_alloc", issue = "32838")] + pub fn new_in(alloc: A) -> BTreeMap { + BTreeMap { root: None, length: 0, alloc: ManuallyDrop::new(alloc) } + } +} + +impl BTreeMap { /// Returns a reference to the value corresponding to the key. /// /// The key may be any borrowed form of the map's key type, but the ordering @@ -648,14 +707,19 @@ impl BTreeMap { /// assert_eq!(*map.get(&2).unwrap(), "b"); /// ``` #[unstable(feature = "map_first_last", issue = "62924")] - pub fn first_entry(&mut self) -> Option> + pub fn first_entry(&mut self) -> Option> where K: Ord, { let (map, dormant_map) = DormantMutRef::new(self); let root_node = map.root.as_mut()?.borrow_mut(); let kv = root_node.first_leaf_edge().right_kv().ok()?; - Some(OccupiedEntry { handle: kv.forget_node_type(), dormant_map, _marker: PhantomData }) + Some(OccupiedEntry { + handle: kv.forget_node_type(), + dormant_map, + alloc: &*map.alloc, + _marker: PhantomData, + }) } /// Removes and returns the first element in the map. @@ -731,14 +795,19 @@ impl BTreeMap { /// assert_eq!(*map.get(&2).unwrap(), "last"); /// ``` #[unstable(feature = "map_first_last", issue = "62924")] - pub fn last_entry(&mut self) -> Option> + pub fn last_entry(&mut self) -> Option> where K: Ord, { let (map, dormant_map) = DormantMutRef::new(self); let root_node = map.root.as_mut()?.borrow_mut(); let kv = root_node.last_leaf_edge().left_kv().ok()?; - Some(OccupiedEntry { handle: kv.forget_node_type(), dormant_map, _marker: PhantomData }) + Some(OccupiedEntry { + handle: kv.forget_node_type(), + dormant_map, + alloc: &*map.alloc, + _marker: PhantomData, + }) } /// Removes and returns the last element in the map. @@ -891,7 +960,7 @@ impl BTreeMap { /// assert_eq!(err.value, "b"); /// ``` #[unstable(feature = "map_try_insert", issue = "82766")] - pub fn try_insert(&mut self, key: K, value: V) -> Result<&mut V, OccupiedError<'_, K, V>> + pub fn try_insert(&mut self, key: K, value: V) -> Result<&mut V, OccupiedError<'_, K, V, A>> where K: Ord, { @@ -955,9 +1024,10 @@ impl BTreeMap { let (map, dormant_map) = DormantMutRef::new(self); let root_node = map.root.as_mut()?.borrow_mut(); match root_node.search_tree(key) { - Found(handle) => { - Some(OccupiedEntry { handle, dormant_map, _marker: PhantomData }.remove_entry()) - } + Found(handle) => Some( + OccupiedEntry { handle, dormant_map, alloc: &*map.alloc, _marker: PhantomData } + .remove_entry(), + ), GoDown(_) => None, } } @@ -1019,6 +1089,7 @@ impl BTreeMap { pub fn append(&mut self, other: &mut Self) where K: Ord, + A: Clone, { // Do we have to append anything at all? if other.is_empty() { @@ -1031,10 +1102,14 @@ impl BTreeMap { return; } - let self_iter = mem::take(self).into_iter(); - let other_iter = mem::take(other).into_iter(); - let root = self.root.get_or_insert_with(Root::new); - root.append_from_sorted_iters(self_iter, other_iter, &mut self.length) + let self_iter = + mem::replace(self, Self::new_in(ManuallyDrop::into_inner(self.alloc.clone()))) + .into_iter(); + let other_iter = + mem::replace(other, Self::new_in(ManuallyDrop::into_inner(self.alloc.clone()))) + .into_iter(); + let root = self.root.get_or_insert_with(|| Root::new(&*self.alloc)); + root.append_from_sorted_iters(self_iter, other_iter, &mut self.length, &*self.alloc) } /// Constructs a double-ended iterator over a sub-range of elements in the map. @@ -1141,21 +1216,31 @@ impl BTreeMap { /// assert_eq!(count["a"], 3); /// ``` #[stable(feature = "rust1", since = "1.0.0")] - pub fn entry(&mut self, key: K) -> Entry<'_, K, V> + pub fn entry(&mut self, key: K) -> Entry<'_, K, V, A> where K: Ord, { let (map, dormant_map) = DormantMutRef::new(self); match map.root { - None => Vacant(VacantEntry { key, handle: None, dormant_map, _marker: PhantomData }), + None => Vacant(VacantEntry { + key, + handle: None, + dormant_map, + alloc: &*map.alloc, + _marker: PhantomData, + }), Some(ref mut root) => match root.borrow_mut().search_tree(&key) { - Found(handle) => { - Occupied(OccupiedEntry { handle, dormant_map, _marker: PhantomData }) - } + Found(handle) => Occupied(OccupiedEntry { + handle, + dormant_map, + alloc: &*map.alloc, + _marker: PhantomData, + }), GoDown(handle) => Vacant(VacantEntry { key, handle: Some(handle), dormant_map, + alloc: &*map.alloc, _marker: PhantomData, }), }, @@ -1195,20 +1280,25 @@ impl BTreeMap { pub fn split_off(&mut self, key: &Q) -> Self where K: Borrow + Ord, + A: Clone, { if self.is_empty() { - return Self::new(); + return Self::new_in(ManuallyDrop::into_inner(self.alloc.clone())); } let total_num = self.len(); let left_root = self.root.as_mut().unwrap(); // unwrap succeeds because not empty - let right_root = left_root.split_off(key); + let right_root = left_root.split_off(key, &*self.alloc); let (new_left_len, right_len) = Root::calc_split_length(total_num, &left_root, &right_root); self.length = new_left_len; - BTreeMap { root: Some(right_root), length: right_len } + BTreeMap { + root: Some(right_root), + length: right_len, + alloc: ManuallyDrop::new((*self.alloc).clone()), + } } /// Creates an iterator that visits all elements (key-value pairs) in @@ -1244,28 +1334,39 @@ impl BTreeMap { /// assert_eq!(odds.keys().copied().collect::>(), [1, 3, 5, 7]); /// ``` #[unstable(feature = "btree_drain_filter", issue = "70530")] - pub fn drain_filter(&mut self, pred: F) -> DrainFilter<'_, K, V, F> + pub fn drain_filter(&mut self, pred: F) -> DrainFilter<'_, K, V, F, &A> where K: Ord, F: FnMut(&K, &mut V) -> bool, { - DrainFilter { pred, inner: self.drain_filter_inner() } + let (inner, alloc) = self.drain_filter_inner(); + DrainFilter { pred, inner, alloc } } - pub(super) fn drain_filter_inner(&mut self) -> DrainFilterInner<'_, K, V> + pub(super) fn drain_filter_inner(&mut self) -> (DrainFilterInner<'_, K, V>, &A) where K: Ord, { if let Some(root) = self.root.as_mut() { let (root, dormant_root) = DormantMutRef::new(root); let front = root.borrow_mut().first_leaf_edge(); - DrainFilterInner { - length: &mut self.length, - dormant_root: Some(dormant_root), - cur_leaf_edge: Some(front), - } + ( + DrainFilterInner { + length: &mut self.length, + dormant_root: Some(dormant_root), + cur_leaf_edge: Some(front), + }, + &*self.alloc, + ) } else { - DrainFilterInner { length: &mut self.length, dormant_root: None, cur_leaf_edge: None } + ( + DrainFilterInner { + length: &mut self.length, + dormant_root: None, + cur_leaf_edge: None, + }, + &*self.alloc, + ) } } @@ -1287,7 +1388,7 @@ impl BTreeMap { /// ``` #[inline] #[stable(feature = "map_into_keys_values", since = "1.54.0")] - pub fn into_keys(self) -> IntoKeys { + pub fn into_keys(self) -> IntoKeys { IntoKeys { inner: self.into_iter() } } @@ -1309,25 +1410,25 @@ impl BTreeMap { /// ``` #[inline] #[stable(feature = "map_into_keys_values", since = "1.54.0")] - pub fn into_values(self) -> IntoValues { + pub fn into_values(self) -> IntoValues { IntoValues { inner: self.into_iter() } } /// Makes a `BTreeMap` from a sorted iterator. - pub(crate) fn bulk_build_from_sorted_iter(iter: I) -> Self + pub(crate) fn bulk_build_from_sorted_iter(iter: I, alloc: A) -> Self where K: Ord, I: IntoIterator, { - let mut root = Root::new(); + let mut root = Root::new(&alloc); let mut length = 0; - root.bulk_push(DedupSortedIter::new(iter.into_iter()), &mut length); - BTreeMap { root: Some(root), length } + root.bulk_push(DedupSortedIter::new(iter.into_iter()), &mut length, &alloc); + BTreeMap { root: Some(root), length, alloc: ManuallyDrop::new(alloc) } } } #[stable(feature = "rust1", since = "1.0.0")] -impl<'a, K, V> IntoIterator for &'a BTreeMap { +impl<'a, K, V, A: Allocator> IntoIterator for &'a BTreeMap { type Item = (&'a K, &'a V); type IntoIter = Iter<'a, K, V>; @@ -1396,7 +1497,7 @@ impl Clone for Iter<'_, K, V> { } #[stable(feature = "rust1", since = "1.0.0")] -impl<'a, K, V> IntoIterator for &'a mut BTreeMap { +impl<'a, K, V, A: Allocator> IntoIterator for &'a mut BTreeMap { type Item = (&'a K, &'a mut V); type IntoIter = IterMut<'a, K, V>; @@ -1406,7 +1507,7 @@ impl<'a, K, V> IntoIterator for &'a mut BTreeMap { } #[stable(feature = "rust1", since = "1.0.0")] -impl<'a, K: 'a, V: 'a> Iterator for IterMut<'a, K, V> { +impl<'a, K, V> Iterator for IterMut<'a, K, V> { type Item = (&'a K, &'a mut V); fn next(&mut self) -> Option<(&'a K, &'a mut V)> { @@ -1436,7 +1537,7 @@ impl<'a, K: 'a, V: 'a> Iterator for IterMut<'a, K, V> { } #[stable(feature = "rust1", since = "1.0.0")] -impl<'a, K: 'a, V: 'a> DoubleEndedIterator for IterMut<'a, K, V> { +impl<'a, K, V> DoubleEndedIterator for IterMut<'a, K, V> { fn next_back(&mut self) -> Option<(&'a K, &'a mut V)> { if self.length == 0 { None @@ -1466,28 +1567,41 @@ impl<'a, K, V> IterMut<'a, K, V> { } #[stable(feature = "rust1", since = "1.0.0")] -impl IntoIterator for BTreeMap { +impl IntoIterator for BTreeMap { type Item = (K, V); - type IntoIter = IntoIter; + type IntoIter = IntoIter; - fn into_iter(self) -> IntoIter { + fn into_iter(self) -> IntoIter { let mut me = ManuallyDrop::new(self); if let Some(root) = me.root.take() { let full_range = root.into_dying().full_range(); - IntoIter { range: full_range, length: me.length } + IntoIter { + range: full_range, + length: me.length, + alloc: unsafe { ManuallyDrop::take(&mut me.alloc) }, + } } else { - IntoIter { range: LazyLeafRange::none(), length: 0 } + IntoIter { + range: LazyLeafRange::none(), + length: 0, + alloc: unsafe { ManuallyDrop::take(&mut me.alloc) }, + } } } } #[stable(feature = "btree_drop", since = "1.7.0")] -impl Drop for IntoIter { +impl Drop for IntoIter { fn drop(&mut self) { - struct DropGuard<'a, K, V>(&'a mut IntoIter); + self.dealloc() + } +} +impl IntoIter { + fn dealloc(&mut self) { + struct DropGuard<'a, K, V, A: Allocator>(&'a mut IntoIter); - impl<'a, K, V> Drop for DropGuard<'a, K, V> { + impl<'a, K, V, A: Allocator> Drop for DropGuard<'a, K, V, A> { fn drop(&mut self) { // Continue the same loop we perform below. This only runs when unwinding, so we // don't have to care about panics this time (they'll abort). @@ -1507,18 +1621,18 @@ impl Drop for IntoIter { } } -impl IntoIter { +impl IntoIter { /// Core of a `next` method returning a dying KV handle, /// invalidated by further calls to this function and some others. fn dying_next( &mut self, ) -> Option, marker::KV>> { if self.length == 0 { - self.range.deallocating_end(); + self.range.deallocating_end(&self.alloc); None } else { self.length -= 1; - Some(unsafe { self.range.deallocating_next_unchecked() }) + Some(unsafe { self.range.deallocating_next_unchecked(&self.alloc) }) } } @@ -1528,17 +1642,22 @@ impl IntoIter { &mut self, ) -> Option, marker::KV>> { if self.length == 0 { - self.range.deallocating_end(); + self.range.deallocating_end(&self.alloc); None } else { self.length -= 1; - Some(unsafe { self.range.deallocating_next_back_unchecked() }) + Some(unsafe { self.range.deallocating_next_back_unchecked(&self.alloc) }) } } + fn into_alloc(mut self) -> A { + self.dealloc(); // Deallocate, then don't drop as drop will also call dealloc + let iter = ManuallyDrop::new(self); + unsafe { ptr::read(&iter.alloc) } + } } #[stable(feature = "rust1", since = "1.0.0")] -impl Iterator for IntoIter { +impl Iterator for IntoIter { type Item = (K, V); fn next(&mut self) -> Option<(K, V)> { @@ -1552,7 +1671,7 @@ impl Iterator for IntoIter { } #[stable(feature = "rust1", since = "1.0.0")] -impl DoubleEndedIterator for IntoIter { +impl DoubleEndedIterator for IntoIter { fn next_back(&mut self) -> Option<(K, V)> { // SAFETY: we consume the dying handle immediately. self.dying_next_back().map(unsafe { |kv| kv.into_key_val() }) @@ -1560,14 +1679,14 @@ impl DoubleEndedIterator for IntoIter { } #[stable(feature = "rust1", since = "1.0.0")] -impl ExactSizeIterator for IntoIter { +impl ExactSizeIterator for IntoIter { fn len(&self) -> usize { self.length } } #[stable(feature = "fused", since = "1.26.0")] -impl FusedIterator for IntoIter {} +impl FusedIterator for IntoIter {} #[stable(feature = "rust1", since = "1.0.0")] impl<'a, K, V> Iterator for Keys<'a, K, V> { @@ -1661,18 +1780,22 @@ impl Clone for Values<'_, K, V> { /// An iterator produced by calling `drain_filter` on BTreeMap. #[unstable(feature = "btree_drain_filter", issue = "70530")] -pub struct DrainFilter<'a, K, V, F> -where - K: 'a, - V: 'a, +pub struct DrainFilter< + 'a, + K, + V, + F, + #[unstable(feature = "allocator_api", issue = "32838")] A: Allocator = Global, +> where F: 'a + FnMut(&K, &mut V) -> bool, { pred: F, inner: DrainFilterInner<'a, K, V>, + alloc: A, } /// Most of the implementation of DrainFilter are generic over the type /// of the predicate, thus also serving for BTreeSet::DrainFilter. -pub(super) struct DrainFilterInner<'a, K: 'a, V: 'a> { +pub(super) struct DrainFilterInner<'a, K, V> { /// Reference to the length field in the borrowed map, updated live. length: &'a mut usize, /// Buried reference to the root field in the borrowed map. @@ -1685,7 +1808,7 @@ pub(super) struct DrainFilterInner<'a, K: 'a, V: 'a> { } #[unstable(feature = "btree_drain_filter", issue = "70530")] -impl Drop for DrainFilter<'_, K, V, F> +impl Drop for DrainFilter<'_, K, V, F, A> where F: FnMut(&K, &mut V) -> bool, { @@ -1707,14 +1830,14 @@ where } #[unstable(feature = "btree_drain_filter", issue = "70530")] -impl Iterator for DrainFilter<'_, K, V, F> +impl Iterator for DrainFilter<'_, K, V, F, A> where F: FnMut(&K, &mut V) -> bool, { type Item = (K, V); fn next(&mut self) -> Option<(K, V)> { - self.inner.next(&mut self.pred) + self.inner.next(&mut self.pred, &self.alloc) } fn size_hint(&self) -> (usize, Option) { @@ -1722,7 +1845,7 @@ where } } -impl<'a, K: 'a, V: 'a> DrainFilterInner<'a, K, V> { +impl<'a, K, V> DrainFilterInner<'a, K, V> { /// Allow Debug implementations to predict the next element. pub(super) fn peek(&self) -> Option<(&K, &V)> { let edge = self.cur_leaf_edge.as_ref()?; @@ -1730,7 +1853,7 @@ impl<'a, K: 'a, V: 'a> DrainFilterInner<'a, K, V> { } /// Implementation of a typical `DrainFilter::next` method, given the predicate. - pub(super) fn next(&mut self, pred: &mut F) -> Option<(K, V)> + pub(super) fn next(&mut self, pred: &mut F, alloc: &A) -> Option<(K, V)> where F: FnMut(&K, &mut V) -> bool, { @@ -1738,13 +1861,16 @@ impl<'a, K: 'a, V: 'a> DrainFilterInner<'a, K, V> { let (k, v) = kv.kv_mut(); if pred(k, v) { *self.length -= 1; - let (kv, pos) = kv.remove_kv_tracking(|| { - // SAFETY: we will touch the root in a way that will not - // invalidate the position returned. - let root = unsafe { self.dormant_root.take().unwrap().awaken() }; - root.pop_internal_level(); - self.dormant_root = Some(DormantMutRef::new(root).1); - }); + let (kv, pos) = kv.remove_kv_tracking( + || { + // SAFETY: we will touch the root in a way that will not + // invalidate the position returned. + let root = unsafe { self.dormant_root.take().unwrap().awaken() }; + root.pop_internal_level(alloc); + self.dormant_root = Some(DormantMutRef::new(root).1); + }, + alloc, + ); self.cur_leaf_edge = Some(pos); return Some(kv); } @@ -1822,7 +1948,7 @@ impl ExactSizeIterator for ValuesMut<'_, K, V> { impl FusedIterator for ValuesMut<'_, K, V> {} #[stable(feature = "map_into_keys_values", since = "1.54.0")] -impl Iterator for IntoKeys { +impl Iterator for IntoKeys { type Item = K; fn next(&mut self) -> Option { @@ -1847,24 +1973,24 @@ impl Iterator for IntoKeys { } #[stable(feature = "map_into_keys_values", since = "1.54.0")] -impl DoubleEndedIterator for IntoKeys { +impl DoubleEndedIterator for IntoKeys { fn next_back(&mut self) -> Option { self.inner.next_back().map(|(k, _)| k) } } #[stable(feature = "map_into_keys_values", since = "1.54.0")] -impl ExactSizeIterator for IntoKeys { +impl ExactSizeIterator for IntoKeys { fn len(&self) -> usize { self.inner.len() } } #[stable(feature = "map_into_keys_values", since = "1.54.0")] -impl FusedIterator for IntoKeys {} +impl FusedIterator for IntoKeys {} #[stable(feature = "map_into_keys_values", since = "1.54.0")] -impl Iterator for IntoValues { +impl Iterator for IntoValues { type Item = V; fn next(&mut self) -> Option { @@ -1881,21 +2007,21 @@ impl Iterator for IntoValues { } #[stable(feature = "map_into_keys_values", since = "1.54.0")] -impl DoubleEndedIterator for IntoValues { +impl DoubleEndedIterator for IntoValues { fn next_back(&mut self) -> Option { self.inner.next_back().map(|(_, v)| v) } } #[stable(feature = "map_into_keys_values", since = "1.54.0")] -impl ExactSizeIterator for IntoValues { +impl ExactSizeIterator for IntoValues { fn len(&self) -> usize { self.inner.len() } } #[stable(feature = "map_into_keys_values", since = "1.54.0")] -impl FusedIterator for IntoValues {} +impl FusedIterator for IntoValues {} #[stable(feature = "btree_range", since = "1.17.0")] impl<'a, K, V> DoubleEndedIterator for Range<'a, K, V> { @@ -1956,12 +2082,12 @@ impl FromIterator<(K, V)> for BTreeMap { // use stable sort to preserve the insertion order. inputs.sort_by(|a, b| a.0.cmp(&b.0)); - BTreeMap::bulk_build_from_sorted_iter(inputs) + BTreeMap::bulk_build_from_sorted_iter(inputs, Global) } } #[stable(feature = "rust1", since = "1.0.0")] -impl Extend<(K, V)> for BTreeMap { +impl Extend<(K, V)> for BTreeMap { #[inline] fn extend>(&mut self, iter: T) { iter.into_iter().for_each(move |(k, v)| { @@ -1976,7 +2102,7 @@ impl Extend<(K, V)> for BTreeMap { } #[stable(feature = "extend_ref", since = "1.2.0")] -impl<'a, K: Ord + Copy, V: Copy> Extend<(&'a K, &'a V)> for BTreeMap { +impl<'a, K: Ord + Copy, V: Copy, A: Allocator> Extend<(&'a K, &'a V)> for BTreeMap { fn extend>(&mut self, iter: I) { self.extend(iter.into_iter().map(|(&key, &value)| (key, value))); } @@ -1988,7 +2114,7 @@ impl<'a, K: Ord + Copy, V: Copy> Extend<(&'a K, &'a V)> for BTreeMap { } #[stable(feature = "rust1", since = "1.0.0")] -impl Hash for BTreeMap { +impl Hash for BTreeMap { fn hash(&self, state: &mut H) { self.len().hash(state); for elt in self { @@ -2006,40 +2132,40 @@ impl Default for BTreeMap { } #[stable(feature = "rust1", since = "1.0.0")] -impl PartialEq for BTreeMap { - fn eq(&self, other: &BTreeMap) -> bool { +impl PartialEq for BTreeMap { + fn eq(&self, other: &BTreeMap) -> bool { self.len() == other.len() && self.iter().zip(other).all(|(a, b)| a == b) } } #[stable(feature = "rust1", since = "1.0.0")] -impl Eq for BTreeMap {} +impl Eq for BTreeMap {} #[stable(feature = "rust1", since = "1.0.0")] -impl PartialOrd for BTreeMap { +impl PartialOrd for BTreeMap { #[inline] - fn partial_cmp(&self, other: &BTreeMap) -> Option { + fn partial_cmp(&self, other: &BTreeMap) -> Option { self.iter().partial_cmp(other.iter()) } } #[stable(feature = "rust1", since = "1.0.0")] -impl Ord for BTreeMap { +impl Ord for BTreeMap { #[inline] - fn cmp(&self, other: &BTreeMap) -> Ordering { + fn cmp(&self, other: &BTreeMap) -> Ordering { self.iter().cmp(other.iter()) } } #[stable(feature = "rust1", since = "1.0.0")] -impl Debug for BTreeMap { +impl Debug for BTreeMap { fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { f.debug_map().entries(self.iter()).finish() } } #[stable(feature = "rust1", since = "1.0.0")] -impl Index<&Q> for BTreeMap +impl Index<&Q> for BTreeMap where K: Borrow + Ord, Q: Ord, @@ -2075,11 +2201,11 @@ impl From<[(K, V); N]> for BTreeMap { // use stable sort to preserve the insertion order. arr.sort_by(|a, b| a.0.cmp(&b.0)); - BTreeMap::bulk_build_from_sorted_iter(arr) + BTreeMap::bulk_build_from_sorted_iter(arr, Global) } } -impl BTreeMap { +impl BTreeMap { /// Gets an iterator over the entries of the map, sorted by key. /// /// # Examples diff --git a/library/alloc/src/collections/btree/map/entry.rs b/library/alloc/src/collections/btree/map/entry.rs index cacd06b5df153..b91898a51aaa9 100644 --- a/library/alloc/src/collections/btree/map/entry.rs +++ b/library/alloc/src/collections/btree/map/entry.rs @@ -2,6 +2,8 @@ use core::fmt::{self, Debug}; use core::marker::PhantomData; use core::mem; +use crate::alloc::{Allocator, Global}; + use super::super::borrow::DormantMutRef; use super::super::node::{marker, Handle, NodeRef}; use super::BTreeMap; @@ -15,18 +17,23 @@ use Entry::*; /// [`entry`]: BTreeMap::entry #[stable(feature = "rust1", since = "1.0.0")] #[cfg_attr(not(test), rustc_diagnostic_item = "BTreeEntry")] -pub enum Entry<'a, K: 'a, V: 'a> { +pub enum Entry< + 'a, + K: 'a, + V: 'a, + #[unstable(feature = "allocator_api", issue = "32838")] A: Allocator = Global, +> { /// A vacant entry. #[stable(feature = "rust1", since = "1.0.0")] - Vacant(#[stable(feature = "rust1", since = "1.0.0")] VacantEntry<'a, K, V>), + Vacant(#[stable(feature = "rust1", since = "1.0.0")] VacantEntry<'a, K, V, A>), /// An occupied entry. #[stable(feature = "rust1", since = "1.0.0")] - Occupied(#[stable(feature = "rust1", since = "1.0.0")] OccupiedEntry<'a, K, V>), + Occupied(#[stable(feature = "rust1", since = "1.0.0")] OccupiedEntry<'a, K, V, A>), } #[stable(feature = "debug_btree_map", since = "1.12.0")] -impl Debug for Entry<'_, K, V> { +impl Debug for Entry<'_, K, V, A> { fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { match *self { Vacant(ref v) => f.debug_tuple("Entry").field(v).finish(), @@ -38,18 +45,25 @@ impl Debug for Entry<'_, K, V> { /// A view into a vacant entry in a `BTreeMap`. /// It is part of the [`Entry`] enum. #[stable(feature = "rust1", since = "1.0.0")] -pub struct VacantEntry<'a, K: 'a, V: 'a> { +pub struct VacantEntry< + 'a, + K, + V, + #[unstable(feature = "allocator_api", issue = "32838")] A: Allocator = Global, +> { pub(super) key: K, /// `None` for a (empty) map without root pub(super) handle: Option, K, V, marker::Leaf>, marker::Edge>>, - pub(super) dormant_map: DormantMutRef<'a, BTreeMap>, + pub(super) dormant_map: DormantMutRef<'a, BTreeMap>, + + pub(super) alloc: &'a A, // Be invariant in `K` and `V` pub(super) _marker: PhantomData<&'a mut (K, V)>, } #[stable(feature = "debug_btree_map", since = "1.12.0")] -impl Debug for VacantEntry<'_, K, V> { +impl Debug for VacantEntry<'_, K, V, A> { fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { f.debug_tuple("VacantEntry").field(self.key()).finish() } @@ -58,16 +72,23 @@ impl Debug for VacantEntry<'_, K, V> { /// A view into an occupied entry in a `BTreeMap`. /// It is part of the [`Entry`] enum. #[stable(feature = "rust1", since = "1.0.0")] -pub struct OccupiedEntry<'a, K: 'a, V: 'a> { +pub struct OccupiedEntry< + 'a, + K, + V, + #[unstable(feature = "allocator_api", issue = "32838")] A: Allocator = Global, +> { pub(super) handle: Handle, K, V, marker::LeafOrInternal>, marker::KV>, - pub(super) dormant_map: DormantMutRef<'a, BTreeMap>, + pub(super) dormant_map: DormantMutRef<'a, BTreeMap>, + + pub(super) alloc: &'a A, // Be invariant in `K` and `V` pub(super) _marker: PhantomData<&'a mut (K, V)>, } #[stable(feature = "debug_btree_map", since = "1.12.0")] -impl Debug for OccupiedEntry<'_, K, V> { +impl Debug for OccupiedEntry<'_, K, V, A> { fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { f.debug_struct("OccupiedEntry").field("key", self.key()).field("value", self.get()).finish() } @@ -77,15 +98,15 @@ impl Debug for OccupiedEntry<'_, K, V> { /// /// Contains the occupied entry, and the value that was not inserted. #[unstable(feature = "map_try_insert", issue = "82766")] -pub struct OccupiedError<'a, K: 'a, V: 'a> { +pub struct OccupiedError<'a, K: 'a, V: 'a, A: Allocator = Global> { /// The entry in the map that was already occupied. - pub entry: OccupiedEntry<'a, K, V>, + pub entry: OccupiedEntry<'a, K, V, A>, /// The value which was not inserted, because the entry was already occupied. pub value: V, } #[unstable(feature = "map_try_insert", issue = "82766")] -impl Debug for OccupiedError<'_, K, V> { +impl Debug for OccupiedError<'_, K, V, A> { fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { f.debug_struct("OccupiedError") .field("key", self.entry.key()) @@ -96,7 +117,7 @@ impl Debug for OccupiedError<'_, K, V> { } #[unstable(feature = "map_try_insert", issue = "82766")] -impl<'a, K: Debug + Ord, V: Debug> fmt::Display for OccupiedError<'a, K, V> { +impl<'a, K: Debug + Ord, V: Debug, A: Allocator> fmt::Display for OccupiedError<'a, K, V, A> { fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { write!( f, @@ -108,7 +129,7 @@ impl<'a, K: Debug + Ord, V: Debug> fmt::Display for OccupiedError<'a, K, V> { } } -impl<'a, K: Ord, V> Entry<'a, K, V> { +impl<'a, K: Ord, V, A: Allocator> Entry<'a, K, V, A> { /// Ensures a value is in the entry by inserting the default if empty, and returns /// a mutable reference to the value in the entry. /// @@ -236,7 +257,7 @@ impl<'a, K: Ord, V> Entry<'a, K, V> { } } -impl<'a, K: Ord, V: Default> Entry<'a, K, V> { +impl<'a, K: Ord, V: Default, A: Allocator> Entry<'a, K, V, A> { #[stable(feature = "entry_or_default", since = "1.28.0")] /// Ensures a value is in the entry by inserting the default value if empty, /// and returns a mutable reference to the value in the entry. @@ -259,7 +280,7 @@ impl<'a, K: Ord, V: Default> Entry<'a, K, V> { } } -impl<'a, K: Ord, V> VacantEntry<'a, K, V> { +impl<'a, K: Ord, V, A: Allocator> VacantEntry<'a, K, V, A> { /// Gets a reference to the key that would be used when inserting a value /// through the VacantEntry. /// @@ -317,13 +338,13 @@ impl<'a, K: Ord, V> VacantEntry<'a, K, V> { None => { // SAFETY: We have consumed self.handle and the reference returned. let map = unsafe { self.dormant_map.awaken() }; - let mut root = NodeRef::new_leaf(); + let mut root = NodeRef::new_leaf(self.alloc); let val_ptr = root.borrow_mut().push(self.key, value) as *mut V; map.root = Some(root.forget_type()); map.length = 1; val_ptr } - Some(handle) => match handle.insert_recursing(self.key, value) { + Some(handle) => match handle.insert_recursing(self.key, value, self.alloc) { (None, val_ptr) => { // SAFETY: We have consumed self.handle and the handle returned. let map = unsafe { self.dormant_map.awaken() }; @@ -335,7 +356,7 @@ impl<'a, K: Ord, V> VacantEntry<'a, K, V> { // SAFETY: We have consumed self.handle and the reference returned. let map = unsafe { self.dormant_map.awaken() }; let root = map.root.as_mut().unwrap(); - root.push_internal_level().push(ins.kv.0, ins.kv.1, ins.right); + root.push_internal_level(self.alloc).push(ins.kv.0, ins.kv.1, ins.right); map.length += 1; val_ptr } @@ -347,7 +368,7 @@ impl<'a, K: Ord, V> VacantEntry<'a, K, V> { } } -impl<'a, K: Ord, V> OccupiedEntry<'a, K, V> { +impl<'a, K: Ord, V, A: Allocator> OccupiedEntry<'a, K, V, A> { /// Gets a reference to the key in the entry. /// /// # Examples @@ -515,13 +536,14 @@ impl<'a, K: Ord, V> OccupiedEntry<'a, K, V> { // Body of `remove_entry`, probably separate because the name reflects the returned pair. pub(super) fn remove_kv(self) -> (K, V) { let mut emptied_internal_root = false; - let (old_kv, _) = self.handle.remove_kv_tracking(|| emptied_internal_root = true); + let (old_kv, _) = + self.handle.remove_kv_tracking(|| emptied_internal_root = true, self.alloc); // SAFETY: we consumed the intermediate root borrow, `self.handle`. let map = unsafe { self.dormant_map.awaken() }; map.length -= 1; if emptied_internal_root { let root = map.root.as_mut().unwrap(); - root.pop_internal_level(); + root.pop_internal_level(&*self.alloc); } old_kv } diff --git a/library/alloc/src/collections/btree/map/tests.rs b/library/alloc/src/collections/btree/map/tests.rs index cc986e93698b7..0e2d1bcce415d 100644 --- a/library/alloc/src/collections/btree/map/tests.rs +++ b/library/alloc/src/collections/btree/map/tests.rs @@ -116,7 +116,11 @@ impl BTreeMap { { let iter = mem::take(self).into_iter(); if !iter.is_empty() { - self.root.insert(Root::new()).bulk_push(iter, &mut self.length); + self.root.insert(Root::new(&*self.alloc)).bulk_push( + iter, + &mut self.length, + &*self.alloc, + ); } } } diff --git a/library/alloc/src/collections/btree/navigate.rs b/library/alloc/src/collections/btree/navigate.rs index 9d0db34500945..d44cb57618dfa 100644 --- a/library/alloc/src/collections/btree/navigate.rs +++ b/library/alloc/src/collections/btree/navigate.rs @@ -5,6 +5,7 @@ use core::ptr; use super::node::{marker, ForceResult::*, Handle, NodeRef}; +use crate::alloc::Allocator; // `front` and `back` are always both `None` or both `Some`. pub struct LeafRange { front: Option, marker::Edge>>, @@ -177,27 +178,29 @@ impl LazyLeafRange { } #[inline] - pub unsafe fn deallocating_next_unchecked( + pub unsafe fn deallocating_next_unchecked( &mut self, + alloc: &A, ) -> Handle, marker::KV> { debug_assert!(self.front.is_some()); let front = self.init_front().unwrap(); - unsafe { front.deallocating_next_unchecked() } + unsafe { front.deallocating_next_unchecked(alloc) } } #[inline] - pub unsafe fn deallocating_next_back_unchecked( + pub unsafe fn deallocating_next_back_unchecked( &mut self, + alloc: &A, ) -> Handle, marker::KV> { debug_assert!(self.back.is_some()); let back = self.init_back().unwrap(); - unsafe { back.deallocating_next_back_unchecked() } + unsafe { back.deallocating_next_back_unchecked(alloc) } } #[inline] - pub fn deallocating_end(&mut self) { + pub fn deallocating_end(&mut self, alloc: &A) { if let Some(front) = self.take_front() { - front.deallocating_end() + front.deallocating_end(alloc) } } } @@ -441,18 +444,21 @@ impl Handle, marker::Edge> { /// `deallocating_next_back`. /// - The returned KV handle is only valid to access the key and value, /// and only valid until the next call to a `deallocating_` method. - unsafe fn deallocating_next( + unsafe fn deallocating_next( self, + alloc: &A, ) -> Option<(Self, Handle, marker::KV>)> { let mut edge = self.forget_node_type(); loop { edge = match edge.right_kv() { Ok(kv) => return Some((unsafe { ptr::read(&kv) }.next_leaf_edge(), kv)), - Err(last_edge) => match unsafe { last_edge.into_node().deallocate_and_ascend() } { - Some(parent_edge) => parent_edge.forget_node_type(), - None => return None, - }, + Err(last_edge) => { + match unsafe { last_edge.into_node().deallocate_and_ascend(alloc) } { + Some(parent_edge) => parent_edge.forget_node_type(), + None => return None, + } + } } } } @@ -470,18 +476,21 @@ impl Handle, marker::Edge> { /// `deallocating_next`. /// - The returned KV handle is only valid to access the key and value, /// and only valid until the next call to a `deallocating_` method. - unsafe fn deallocating_next_back( + unsafe fn deallocating_next_back( self, + alloc: &A, ) -> Option<(Self, Handle, marker::KV>)> { let mut edge = self.forget_node_type(); loop { edge = match edge.left_kv() { Ok(kv) => return Some((unsafe { ptr::read(&kv) }.next_back_leaf_edge(), kv)), - Err(last_edge) => match unsafe { last_edge.into_node().deallocate_and_ascend() } { - Some(parent_edge) => parent_edge.forget_node_type(), - None => return None, - }, + Err(last_edge) => { + match unsafe { last_edge.into_node().deallocate_and_ascend(alloc) } { + Some(parent_edge) => parent_edge.forget_node_type(), + None => return None, + } + } } } } @@ -492,9 +501,9 @@ impl Handle, marker::Edge> { /// both sides of the tree, and have hit the same edge. As it is intended /// only to be called when all keys and values have been returned, /// no cleanup is done on any of the keys or values. - fn deallocating_end(self) { + fn deallocating_end(self, alloc: &A) { let mut edge = self.forget_node_type(); - while let Some(parent_edge) = unsafe { edge.into_node().deallocate_and_ascend() } { + while let Some(parent_edge) = unsafe { edge.into_node().deallocate_and_ascend(alloc) } { edge = parent_edge.forget_node_type(); } } @@ -569,10 +578,13 @@ impl Handle, marker::Edge> { /// /// The only safe way to proceed with the updated handle is to compare it, drop it, /// or call this method or counterpart `deallocating_next_back_unchecked` again. - unsafe fn deallocating_next_unchecked( + unsafe fn deallocating_next_unchecked( &mut self, + alloc: &A, ) -> Handle, marker::KV> { - super::mem::replace(self, |leaf_edge| unsafe { leaf_edge.deallocating_next().unwrap() }) + super::mem::replace(self, |leaf_edge| unsafe { + leaf_edge.deallocating_next(alloc).unwrap() + }) } /// Moves the leaf edge handle to the previous leaf edge and returns the key and value @@ -587,11 +599,12 @@ impl Handle, marker::Edge> { /// /// The only safe way to proceed with the updated handle is to compare it, drop it, /// or call this method or counterpart `deallocating_next_unchecked` again. - unsafe fn deallocating_next_back_unchecked( + unsafe fn deallocating_next_back_unchecked( &mut self, + alloc: &A, ) -> Handle, marker::KV> { super::mem::replace(self, |leaf_edge| unsafe { - leaf_edge.deallocating_next_back().unwrap() + leaf_edge.deallocating_next_back(alloc).unwrap() }) } } diff --git a/library/alloc/src/collections/btree/node.rs b/library/alloc/src/collections/btree/node.rs index b5f0edf6b33a7..5ae0a554aeea6 100644 --- a/library/alloc/src/collections/btree/node.rs +++ b/library/alloc/src/collections/btree/node.rs @@ -36,7 +36,7 @@ use core::mem::{self, MaybeUninit}; use core::ptr::{self, NonNull}; use core::slice::SliceIndex; -use crate::alloc::{Allocator, Global, Layout}; +use crate::alloc::{Allocator, Layout}; use crate::boxed::Box; const B: usize = 6; @@ -78,9 +78,9 @@ impl LeafNode { } /// Creates a new boxed `LeafNode`. - fn new() -> Box { + fn new(alloc: &A) -> Box { unsafe { - let mut leaf = Box::new_uninit(); + let mut leaf = Box::new_uninit_in(alloc); LeafNode::init(leaf.as_mut_ptr()); leaf.assume_init() } @@ -110,9 +110,9 @@ impl InternalNode { /// An invariant of internal nodes is that they have at least one /// initialized and valid edge. This function does not set up /// such an edge. - unsafe fn new() -> Box { + unsafe fn new(alloc: &A) -> Box { unsafe { - let mut node = Box::::new_uninit(); + let mut node = Box::::new_uninit_in(alloc); // We only need to initialize the data; the edges are MaybeUninit. LeafNode::init(ptr::addr_of_mut!((*node.as_mut_ptr()).data)); node.assume_init() @@ -213,25 +213,28 @@ unsafe impl Send for NodeRef unsafe impl Send for NodeRef {} impl NodeRef { - pub fn new_leaf() -> Self { - Self::from_new_leaf(LeafNode::new()) + pub fn new_leaf(alloc: &A) -> Self { + Self::from_new_leaf(LeafNode::new(alloc)) } - fn from_new_leaf(leaf: Box>) -> Self { + fn from_new_leaf(leaf: Box, A>) -> Self { NodeRef { height: 0, node: NonNull::from(Box::leak(leaf)), _marker: PhantomData } } } impl NodeRef { - fn new_internal(child: Root) -> Self { - let mut new_node = unsafe { InternalNode::new() }; + fn new_internal(child: Root, alloc: &A) -> Self { + let mut new_node = unsafe { InternalNode::new(alloc) }; new_node.edges[0].write(child.node); unsafe { NodeRef::from_new_internal(new_node, child.height + 1) } } /// # Safety /// `height` must not be zero. - unsafe fn from_new_internal(internal: Box>, height: usize) -> Self { + unsafe fn from_new_internal( + internal: Box, A>, + height: usize, + ) -> Self { debug_assert!(height > 0); let node = NonNull::from(Box::leak(internal)).cast(); let mut this = NodeRef { height, node, _marker: PhantomData }; @@ -387,14 +390,15 @@ impl NodeRef { /// Similar to `ascend`, gets a reference to a node's parent node, but also /// deallocates the current node in the process. This is unsafe because the /// current node will still be accessible despite being deallocated. - pub unsafe fn deallocate_and_ascend( + pub unsafe fn deallocate_and_ascend( self, + alloc: &A, ) -> Option, marker::Edge>> { let height = self.height; let node = self.node; let ret = self.ascend().ok(); unsafe { - Global.deallocate( + alloc.deallocate( node.cast(), if height > 0 { Layout::new::>() @@ -555,15 +559,18 @@ impl NodeRef { impl NodeRef { /// Returns a new owned tree, with its own root node that is initially empty. - pub fn new() -> Self { - NodeRef::new_leaf().forget_type() + pub fn new(alloc: &A) -> Self { + NodeRef::new_leaf(alloc).forget_type() } /// Adds a new internal node with a single edge pointing to the previous root node, /// make that new node the root node, and return it. This increases the height by 1 /// and is the opposite of `pop_internal_level`. - pub fn push_internal_level(&mut self) -> NodeRef, K, V, marker::Internal> { - super::mem::take_mut(self, |old_root| NodeRef::new_internal(old_root).forget_type()); + pub fn push_internal_level( + &mut self, + alloc: &A, + ) -> NodeRef, K, V, marker::Internal> { + super::mem::take_mut(self, |old_root| NodeRef::new_internal(old_root, alloc).forget_type()); // `self.borrow_mut()`, except that we just forgot we're internal now: NodeRef { height: self.height, node: self.node, _marker: PhantomData } @@ -578,7 +585,7 @@ impl NodeRef { /// it will not invalidate other handles or references to the root node. /// /// Panics if there is no internal level, i.e., if the root node is a leaf. - pub fn pop_internal_level(&mut self) { + pub fn pop_internal_level(&mut self, alloc: &A) { assert!(self.height > 0); let top = self.node; @@ -593,7 +600,7 @@ impl NodeRef { self.clear_parent_link(); unsafe { - Global.deallocate(top.cast(), Layout::new::>()); + alloc.deallocate(top.cast(), Layout::new::>()); } } } @@ -862,14 +869,19 @@ impl<'a, K: 'a, V: 'a> Handle, K, V, marker::Leaf>, mark /// this edge. This method splits the node if there isn't enough room. /// /// The returned pointer points to the inserted value. - fn insert(mut self, key: K, val: V) -> (Option>, *mut V) { + fn insert( + mut self, + key: K, + val: V, + alloc: &A, + ) -> (Option>, *mut V) { if self.node.len() < CAPACITY { let val_ptr = self.insert_fit(key, val); (None, val_ptr) } else { let (middle_kv_idx, insertion) = splitpoint(self.idx); let middle = unsafe { Handle::new_kv(self.node, middle_kv_idx) }; - let mut result = middle.split(); + let mut result = middle.split(alloc); let mut insertion_edge = match insertion { LeftOrRight::Left(insert_idx) => unsafe { Handle::new_edge(result.left.reborrow_mut(), insert_idx) @@ -918,11 +930,12 @@ impl<'a, K: 'a, V: 'a> Handle, K, V, marker::Internal>, /// Inserts a new key-value pair and an edge that will go to the right of that new pair /// between this edge and the key-value pair to the right of this edge. This method splits /// the node if there isn't enough room. - fn insert( + fn insert( mut self, key: K, val: V, edge: Root, + alloc: &A, ) -> Option> { assert!(edge.height == self.node.height - 1); @@ -932,7 +945,7 @@ impl<'a, K: 'a, V: 'a> Handle, K, V, marker::Internal>, } else { let (middle_kv_idx, insertion) = splitpoint(self.idx); let middle = unsafe { Handle::new_kv(self.node, middle_kv_idx) }; - let mut result = middle.split(); + let mut result = middle.split(alloc); let mut insertion_edge = match insertion { LeftOrRight::Left(insert_idx) => unsafe { Handle::new_edge(result.left.reborrow_mut(), insert_idx) @@ -955,19 +968,20 @@ impl<'a, K: 'a, V: 'a> Handle, K, V, marker::Leaf>, mark /// If the returned result is some `SplitResult`, the `left` field will be the root node. /// The returned pointer points to the inserted value, which in the case of `SplitResult` /// is in the `left` or `right` tree. - pub fn insert_recursing( + pub fn insert_recursing( self, key: K, value: V, + alloc: &A, ) -> (Option>, *mut V) { - let (mut split, val_ptr) = match self.insert(key, value) { + let (mut split, val_ptr) = match self.insert(key, value, alloc) { (None, val_ptr) => return (None, val_ptr), (Some(split), val_ptr) => (split.forget_node_type(), val_ptr), }; loop { split = match split.left.ascend() { - Ok(parent) => match parent.insert(split.kv.0, split.kv.1, split.right) { + Ok(parent) => match parent.insert(split.kv.0, split.kv.1, split.right, alloc) { None => return (None, val_ptr), Some(split) => split.forget_node_type(), }, @@ -1112,8 +1126,8 @@ impl<'a, K: 'a, V: 'a> Handle, K, V, marker::Leaf>, mark /// - The key and value pointed to by this handle are extracted. /// - All the key-value pairs to the right of this handle are put into a newly /// allocated node. - pub fn split(mut self) -> SplitResult<'a, K, V, marker::Leaf> { - let mut new_node = LeafNode::new(); + pub fn split(mut self, alloc: &A) -> SplitResult<'a, K, V, marker::Leaf> { + let mut new_node = LeafNode::new(alloc); let kv = self.split_leaf_data(&mut new_node); @@ -1144,10 +1158,10 @@ impl<'a, K: 'a, V: 'a> Handle, K, V, marker::Internal>, /// - The key and value pointed to by this handle are extracted. /// - All the edges and key-value pairs to the right of this handle are put into /// a newly allocated node. - pub fn split(mut self) -> SplitResult<'a, K, V, marker::Internal> { + pub fn split(mut self, alloc: &A) -> SplitResult<'a, K, V, marker::Internal> { let old_len = self.node.len(); unsafe { - let mut new_node = InternalNode::new(); + let mut new_node = InternalNode::new(alloc); let kv = self.split_leaf_data(&mut new_node.data); let new_len = usize::from(new_node.data.len); move_to_slice( @@ -1252,9 +1266,11 @@ impl<'a, K: 'a, V: 'a> BalancingContext<'a, K, V> { NodeRef, K, V, marker::LeafOrInternal>, ) -> R, R, + A: Allocator, >( self, result: F, + alloc: &A, ) -> R { let Handle { node: mut parent_node, idx: parent_idx, _marker } = self.parent; let old_parent_len = parent_node.len(); @@ -1299,9 +1315,9 @@ impl<'a, K: 'a, V: 'a> BalancingContext<'a, K, V> { left_node.correct_childrens_parent_links(old_left_len + 1..new_left_len + 1); - Global.deallocate(right_node.node.cast(), Layout::new::>()); + alloc.deallocate(right_node.node.cast(), Layout::new::>()); } else { - Global.deallocate(right_node.node.cast(), Layout::new::>()); + alloc.deallocate(right_node.node.cast(), Layout::new::>()); } } result(parent_node, left_node) @@ -1311,16 +1327,22 @@ impl<'a, K: 'a, V: 'a> BalancingContext<'a, K, V> { /// the left child node and returns the shrunk parent node. /// /// Panics unless we `.can_merge()`. - pub fn merge_tracking_parent(self) -> NodeRef, K, V, marker::Internal> { - self.do_merge(|parent, _child| parent) + pub fn merge_tracking_parent( + self, + alloc: &A, + ) -> NodeRef, K, V, marker::Internal> { + self.do_merge(|parent, _child| parent, alloc) } /// Merges the parent's key-value pair and both adjacent child nodes into /// the left child node and returns that child node. /// /// Panics unless we `.can_merge()`. - pub fn merge_tracking_child(self) -> NodeRef, K, V, marker::LeafOrInternal> { - self.do_merge(|_parent, child| child) + pub fn merge_tracking_child( + self, + alloc: &A, + ) -> NodeRef, K, V, marker::LeafOrInternal> { + self.do_merge(|_parent, child| child, alloc) } /// Merges the parent's key-value pair and both adjacent child nodes into @@ -1328,9 +1350,10 @@ impl<'a, K: 'a, V: 'a> BalancingContext<'a, K, V> { /// where the tracked child edge ended up, /// /// Panics unless we `.can_merge()`. - pub fn merge_tracking_child_edge( + pub fn merge_tracking_child_edge( self, track_edge_idx: LeftOrRight, + alloc: &A, ) -> Handle, K, V, marker::LeafOrInternal>, marker::Edge> { let old_left_len = self.left_child.len(); let right_len = self.right_child.len(); @@ -1338,7 +1361,7 @@ impl<'a, K: 'a, V: 'a> BalancingContext<'a, K, V> { LeftOrRight::Left(idx) => idx <= old_left_len, LeftOrRight::Right(idx) => idx <= right_len, }); - let child = self.merge_tracking_child(); + let child = self.merge_tracking_child(alloc); let new_idx = match track_edge_idx { LeftOrRight::Left(idx) => idx, LeftOrRight::Right(idx) => old_left_len + 1 + idx, diff --git a/library/alloc/src/collections/btree/node/tests.rs b/library/alloc/src/collections/btree/node/tests.rs index 5e44735462282..c8634e6c06fe3 100644 --- a/library/alloc/src/collections/btree/node/tests.rs +++ b/library/alloc/src/collections/btree/node/tests.rs @@ -1,5 +1,6 @@ use super::super::navigate; use super::*; +use crate::alloc::Global; use crate::fmt::Debug; use crate::string::String; @@ -67,10 +68,10 @@ fn test_splitpoint() { #[test] fn test_partial_eq() { - let mut root1 = NodeRef::new_leaf(); + let mut root1 = NodeRef::new_leaf(&Global); root1.borrow_mut().push(1, ()); - let mut root1 = NodeRef::new_internal(root1.forget_type()).forget_type(); - let root2 = Root::new(); + let mut root1 = NodeRef::new_internal(root1.forget_type(), &Global).forget_type(); + let root2 = Root::new(&Global); root1.reborrow().assert_back_pointers(); root2.reborrow().assert_back_pointers(); @@ -86,9 +87,9 @@ fn test_partial_eq() { assert!(top_edge_1 == top_edge_1); assert!(top_edge_1 != top_edge_2); - root1.pop_internal_level(); - unsafe { root1.into_dying().deallocate_and_ascend() }; - unsafe { root2.into_dying().deallocate_and_ascend() }; + root1.pop_internal_level(&Global); + unsafe { root1.into_dying().deallocate_and_ascend(&Global) }; + unsafe { root2.into_dying().deallocate_and_ascend(&Global) }; } #[test] diff --git a/library/alloc/src/collections/btree/remove.rs b/library/alloc/src/collections/btree/remove.rs index 6bc1252b9cb9d..693efd1765460 100644 --- a/library/alloc/src/collections/btree/remove.rs +++ b/library/alloc/src/collections/btree/remove.rs @@ -1,26 +1,29 @@ use super::map::MIN_LEN; use super::node::{marker, ForceResult::*, Handle, LeftOrRight::*, NodeRef}; +use core::alloc::Allocator; impl<'a, K: 'a, V: 'a> Handle, K, V, marker::LeafOrInternal>, marker::KV> { /// Removes a key-value pair from the tree, and returns that pair, as well as /// the leaf edge corresponding to that former pair. It's possible this empties /// a root node that is internal, which the caller should pop from the map /// holding the tree. The caller should also decrement the map's length. - pub fn remove_kv_tracking( + pub fn remove_kv_tracking( self, handle_emptied_internal_root: F, + alloc: &A, ) -> ((K, V), Handle, K, V, marker::Leaf>, marker::Edge>) { match self.force() { - Leaf(node) => node.remove_leaf_kv(handle_emptied_internal_root), - Internal(node) => node.remove_internal_kv(handle_emptied_internal_root), + Leaf(node) => node.remove_leaf_kv(handle_emptied_internal_root, alloc), + Internal(node) => node.remove_internal_kv(handle_emptied_internal_root, alloc), } } } impl<'a, K: 'a, V: 'a> Handle, K, V, marker::Leaf>, marker::KV> { - fn remove_leaf_kv( + fn remove_leaf_kv( self, handle_emptied_internal_root: F, + alloc: &A, ) -> ((K, V), Handle, K, V, marker::Leaf>, marker::Edge>) { let (old_kv, mut pos) = self.remove(); let len = pos.reborrow().into_node().len(); @@ -32,7 +35,7 @@ impl<'a, K: 'a, V: 'a> Handle, K, V, marker::Leaf>, mark Ok(Left(left_parent_kv)) => { debug_assert!(left_parent_kv.right_child_len() == MIN_LEN - 1); if left_parent_kv.can_merge() { - left_parent_kv.merge_tracking_child_edge(Right(idx)) + left_parent_kv.merge_tracking_child_edge(Right(idx), alloc) } else { debug_assert!(left_parent_kv.left_child_len() > MIN_LEN); left_parent_kv.steal_left(idx) @@ -41,7 +44,7 @@ impl<'a, K: 'a, V: 'a> Handle, K, V, marker::Leaf>, mark Ok(Right(right_parent_kv)) => { debug_assert!(right_parent_kv.left_child_len() == MIN_LEN - 1); if right_parent_kv.can_merge() { - right_parent_kv.merge_tracking_child_edge(Left(idx)) + right_parent_kv.merge_tracking_child_edge(Left(idx), alloc) } else { debug_assert!(right_parent_kv.right_child_len() > MIN_LEN); right_parent_kv.steal_right(idx) @@ -60,7 +63,7 @@ impl<'a, K: 'a, V: 'a> Handle, K, V, marker::Leaf>, mark // rearrange the parent through the grandparent, thus change the // link to the parent inside the leaf. if let Ok(parent) = unsafe { pos.reborrow_mut() }.into_node().ascend() { - if !parent.into_node().forget_type().fix_node_and_affected_ancestors() { + if !parent.into_node().forget_type().fix_node_and_affected_ancestors(alloc) { handle_emptied_internal_root(); } } @@ -70,16 +73,17 @@ impl<'a, K: 'a, V: 'a> Handle, K, V, marker::Leaf>, mark } impl<'a, K: 'a, V: 'a> Handle, K, V, marker::Internal>, marker::KV> { - fn remove_internal_kv( + fn remove_internal_kv( self, handle_emptied_internal_root: F, + alloc: &A, ) -> ((K, V), Handle, K, V, marker::Leaf>, marker::Edge>) { // Remove an adjacent KV from its leaf and then put it back in place of // the element we were asked to remove. Prefer the left adjacent KV, // for the reasons listed in `choose_parent_kv`. let left_leaf_kv = self.left_edge().descend().last_leaf_edge().left_kv(); let left_leaf_kv = unsafe { left_leaf_kv.ok().unwrap_unchecked() }; - let (left_kv, left_hole) = left_leaf_kv.remove_leaf_kv(handle_emptied_internal_root); + let (left_kv, left_hole) = left_leaf_kv.remove_leaf_kv(handle_emptied_internal_root, alloc); // The internal node may have been stolen from or merged. Go back right // to find where the original KV ended up. diff --git a/library/alloc/src/collections/btree/set.rs b/library/alloc/src/collections/btree/set.rs index d6733425288d4..2b5a960da6456 100644 --- a/library/alloc/src/collections/btree/set.rs +++ b/library/alloc/src/collections/btree/set.rs @@ -3,16 +3,20 @@ use crate::vec::Vec; use core::borrow::Borrow; -use core::cmp::Ordering::{Equal, Greater, Less}; +use core::cmp::Ordering::{self, Equal, Greater, Less}; use core::cmp::{max, min}; use core::fmt::{self, Debug}; +use core::hash::{Hash, Hasher}; use core::iter::{FromIterator, FusedIterator, Peekable}; +use core::mem::ManuallyDrop; use core::ops::{BitAnd, BitOr, BitXor, RangeBounds, Sub}; use super::map::{BTreeMap, Keys}; use super::merge_iter::MergeIterInner; use super::Recover; +use crate::alloc::{Allocator, Global}; + // FIXME(conventions): implement bounded iterators /// An ordered set based on a B-Tree. @@ -71,15 +75,48 @@ use super::Recover; /// /// let set = BTreeSet::from([1, 2, 3]); /// ``` -#[derive(Hash, PartialEq, Eq, Ord, PartialOrd)] #[stable(feature = "rust1", since = "1.0.0")] #[cfg_attr(not(test), rustc_diagnostic_item = "BTreeSet")] -pub struct BTreeSet { - map: BTreeMap, +pub struct BTreeSet< + T, + #[unstable(feature = "allocator_api", issue = "32838")] A: Allocator = Global, +> { + map: BTreeMap, } #[stable(feature = "rust1", since = "1.0.0")] -impl Clone for BTreeSet { +impl Hash for BTreeSet { + fn hash(&self, state: &mut H) { + self.map.hash(state) + } +} + +#[stable(feature = "rust1", since = "1.0.0")] +impl PartialEq for BTreeSet { + fn eq(&self, other: &BTreeSet) -> bool { + self.map.eq(&other.map) + } +} + +#[stable(feature = "rust1", since = "1.0.0")] +impl Eq for BTreeSet {} + +#[stable(feature = "rust1", since = "1.0.0")] +impl PartialOrd for BTreeSet { + fn partial_cmp(&self, other: &BTreeSet) -> Option { + self.map.partial_cmp(&other.map) + } +} + +#[stable(feature = "rust1", since = "1.0.0")] +impl Ord for BTreeSet { + fn cmp(&self, other: &BTreeSet) -> Ordering { + self.map.cmp(&other.map) + } +} + +#[stable(feature = "rust1", since = "1.0.0")] +impl Clone for BTreeSet { fn clone(&self) -> Self { BTreeSet { map: self.map.clone() } } @@ -117,8 +154,11 @@ impl fmt::Debug for Iter<'_, T> { /// [`IntoIterator`]: core::iter::IntoIterator #[stable(feature = "rust1", since = "1.0.0")] #[derive(Debug)] -pub struct IntoIter { - iter: super::map::IntoIter, +pub struct IntoIter< + T, + #[unstable(feature = "allocator_api", issue = "32838")] A: Allocator = Global, +> { + iter: super::map::IntoIter, } /// An iterator over a sub-range of items in a `BTreeSet`. @@ -143,11 +183,14 @@ pub struct Range<'a, T: 'a> { #[must_use = "this returns the difference as an iterator, \ without modifying either input set"] #[stable(feature = "rust1", since = "1.0.0")] -pub struct Difference<'a, T: 'a> { - inner: DifferenceInner<'a, T>, +pub struct Difference< + 'a, + T: 'a, + #[unstable(feature = "allocator_api", issue = "32838")] A: Allocator = Global, +> { + inner: DifferenceInner<'a, T, A>, } -#[derive(Debug)] -enum DifferenceInner<'a, T: 'a> { +enum DifferenceInner<'a, T: 'a, A: Allocator> { Stitch { // iterate all of `self` and some of `other`, spotting matches along the way self_iter: Iter<'a, T>, @@ -156,13 +199,32 @@ enum DifferenceInner<'a, T: 'a> { Search { // iterate `self`, look up in `other` self_iter: Iter<'a, T>, - other_set: &'a BTreeSet, + other_set: &'a BTreeSet, }, Iterate(Iter<'a, T>), // simply produce all elements in `self` } +// Explicit Debug impl necessary because of issue #26925 +impl Debug for DifferenceInner<'_, T, A> { + fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { + match self { + DifferenceInner::Stitch { self_iter, other_iter } => f + .debug_struct("Stitch") + .field("self_iter", self_iter) + .field("other_iter", other_iter) + .finish(), + DifferenceInner::Search { self_iter, other_set } => f + .debug_struct("Search") + .field("self_iter", self_iter) + .field("other_iter", other_set) + .finish(), + DifferenceInner::Iterate(x) => f.debug_tuple("Iterate").field(x).finish(), + } + } +} + #[stable(feature = "collection_debug", since = "1.17.0")] -impl fmt::Debug for Difference<'_, T> { +impl fmt::Debug for Difference<'_, T, A> { fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { f.debug_tuple("Difference").field(&self.inner).finish() } @@ -195,11 +257,14 @@ impl fmt::Debug for SymmetricDifference<'_, T> { #[must_use = "this returns the intersection as an iterator, \ without modifying either input set"] #[stable(feature = "rust1", since = "1.0.0")] -pub struct Intersection<'a, T: 'a> { - inner: IntersectionInner<'a, T>, +pub struct Intersection< + 'a, + T: 'a, + #[unstable(feature = "allocator_api", issue = "32838")] A: Allocator = Global, +> { + inner: IntersectionInner<'a, T, A>, } -#[derive(Debug)] -enum IntersectionInner<'a, T: 'a> { +enum IntersectionInner<'a, T: 'a, A: Allocator> { Stitch { // iterate similarly sized sets jointly, spotting matches along the way a: Iter<'a, T>, @@ -208,13 +273,30 @@ enum IntersectionInner<'a, T: 'a> { Search { // iterate a small set, look up in the large set small_iter: Iter<'a, T>, - large_set: &'a BTreeSet, + large_set: &'a BTreeSet, }, Answer(Option<&'a T>), // return a specific element or emptiness } +// Explicit Debug impl necessary because of issue #26925 +impl Debug for IntersectionInner<'_, T, A> { + fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { + match self { + IntersectionInner::Stitch { a, b } => { + f.debug_struct("Stitch").field("a", a).field("b", b).finish() + } + IntersectionInner::Search { small_iter, large_set } => f + .debug_struct("Search") + .field("small_iter", small_iter) + .field("large_set", large_set) + .finish(), + IntersectionInner::Answer(x) => f.debug_tuple("Answer").field(x).finish(), + } + } +} + #[stable(feature = "collection_debug", since = "1.17.0")] -impl fmt::Debug for Intersection<'_, T> { +impl Debug for Intersection<'_, T, A> { fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { f.debug_tuple("Intersection").field(&self.inner).finish() } @@ -265,6 +347,26 @@ impl BTreeSet { pub const fn new() -> BTreeSet { BTreeSet { map: BTreeMap::new() } } +} + +impl BTreeSet { + /// Makes a new `BTreeSet` with a reasonable choice of B. + /// + /// # Examples + /// + /// ``` + /// # #![allow(unused_mut)] + /// # #![feature(allocator_api)] + /// # #![feature(btreemap_alloc)] + /// use std::collections::BTreeSet; + /// use std::alloc::Global; + /// + /// let mut set: BTreeSet = BTreeSet::new_in(Global); + /// ``` + #[unstable(feature = "btreemap_alloc", issue = "32838")] + pub fn new_in(alloc: A) -> BTreeSet { + BTreeSet { map: BTreeMap::new_in(alloc) } + } /// Constructs a double-ended iterator over a sub-range of elements in the set. /// The simplest way is to use the range syntax `min..max`, thus `range(min..max)` will @@ -319,7 +421,7 @@ impl BTreeSet { /// assert_eq!(diff, [1]); /// ``` #[stable(feature = "rust1", since = "1.0.0")] - pub fn difference<'a>(&'a self, other: &'a BTreeSet) -> Difference<'a, T> + pub fn difference<'a>(&'a self, other: &'a BTreeSet) -> Difference<'a, T, A> where T: Ord, { @@ -380,7 +482,10 @@ impl BTreeSet { /// assert_eq!(sym_diff, [1, 3]); /// ``` #[stable(feature = "rust1", since = "1.0.0")] - pub fn symmetric_difference<'a>(&'a self, other: &'a BTreeSet) -> SymmetricDifference<'a, T> + pub fn symmetric_difference<'a>( + &'a self, + other: &'a BTreeSet, + ) -> SymmetricDifference<'a, T> where T: Ord, { @@ -408,7 +513,7 @@ impl BTreeSet { /// assert_eq!(intersection, [2]); /// ``` #[stable(feature = "rust1", since = "1.0.0")] - pub fn intersection<'a>(&'a self, other: &'a BTreeSet) -> Intersection<'a, T> + pub fn intersection<'a>(&'a self, other: &'a BTreeSet) -> Intersection<'a, T, A> where T: Ord, { @@ -459,7 +564,7 @@ impl BTreeSet { /// assert_eq!(union, [1, 2]); /// ``` #[stable(feature = "rust1", since = "1.0.0")] - pub fn union<'a>(&'a self, other: &'a BTreeSet) -> Union<'a, T> + pub fn union<'a>(&'a self, other: &'a BTreeSet) -> Union<'a, T> where T: Ord, { @@ -479,7 +584,10 @@ impl BTreeSet { /// assert!(v.is_empty()); /// ``` #[stable(feature = "rust1", since = "1.0.0")] - pub fn clear(&mut self) { + pub fn clear(&mut self) + where + A: Clone, + { self.map.clear() } @@ -551,7 +659,7 @@ impl BTreeSet { /// ``` #[must_use] #[stable(feature = "rust1", since = "1.0.0")] - pub fn is_disjoint(&self, other: &BTreeSet) -> bool + pub fn is_disjoint(&self, other: &BTreeSet) -> bool where T: Ord, { @@ -577,7 +685,7 @@ impl BTreeSet { /// ``` #[must_use] #[stable(feature = "rust1", since = "1.0.0")] - pub fn is_subset(&self, other: &BTreeSet) -> bool + pub fn is_subset(&self, other: &BTreeSet) -> bool where T: Ord, { @@ -657,7 +765,7 @@ impl BTreeSet { /// ``` #[must_use] #[stable(feature = "rust1", since = "1.0.0")] - pub fn is_superset(&self, other: &BTreeSet) -> bool + pub fn is_superset(&self, other: &BTreeSet) -> bool where T: Ord, { @@ -927,6 +1035,7 @@ impl BTreeSet { pub fn append(&mut self, other: &mut Self) where T: Ord, + A: Clone, { self.map.append(&mut other.map); } @@ -964,6 +1073,7 @@ impl BTreeSet { pub fn split_off(&mut self, value: &Q) -> Self where T: Borrow + Ord, + A: Clone, { BTreeSet { map: self.map.split_off(value) } } @@ -998,12 +1108,13 @@ impl BTreeSet { /// assert_eq!(odds.into_iter().collect::>(), vec![1, 3, 5, 7]); /// ``` #[unstable(feature = "btree_drain_filter", issue = "70530")] - pub fn drain_filter<'a, F>(&'a mut self, pred: F) -> DrainFilter<'a, T, F> + pub fn drain_filter<'a, F>(&'a mut self, pred: F) -> DrainFilter<'a, T, F, A> where T: Ord, F: 'a + FnMut(&T) -> bool, { - DrainFilter { pred, inner: self.map.drain_filter_inner() } + let (inner, alloc) = self.map.drain_filter_inner(); + DrainFilter { pred, inner, alloc } } /// Gets an iterator that visits the elements in the `BTreeSet` in ascending @@ -1090,7 +1201,7 @@ impl FromIterator for BTreeSet { // use stable sort to preserve the insertion order. inputs.sort(); let iter = inputs.into_iter().map(|k| (k, ())); - let map = BTreeMap::bulk_build_from_sorted_iter(iter); + let map = BTreeMap::bulk_build_from_sorted_iter(iter, Global); BTreeSet { map } } } @@ -1114,15 +1225,15 @@ impl From<[T; N]> for BTreeSet { // use stable sort to preserve the insertion order. arr.sort(); let iter = IntoIterator::into_iter(arr).map(|k| (k, ())); - let map = BTreeMap::bulk_build_from_sorted_iter(iter); + let map = BTreeMap::bulk_build_from_sorted_iter(iter, Global); BTreeSet { map } } } #[stable(feature = "rust1", since = "1.0.0")] -impl IntoIterator for BTreeSet { +impl IntoIterator for BTreeSet { type Item = T; - type IntoIter = IntoIter; + type IntoIter = IntoIter; /// Gets an iterator for moving out the `BTreeSet`'s contents. /// @@ -1136,13 +1247,13 @@ impl IntoIterator for BTreeSet { /// let v: Vec<_> = set.into_iter().collect(); /// assert_eq!(v, [1, 2, 3, 4]); /// ``` - fn into_iter(self) -> IntoIter { + fn into_iter(self) -> IntoIter { IntoIter { iter: self.map.into_iter() } } } #[stable(feature = "rust1", since = "1.0.0")] -impl<'a, T> IntoIterator for &'a BTreeSet { +impl<'a, T, A: Allocator> IntoIterator for &'a BTreeSet { type Item = &'a T; type IntoIter = Iter<'a, T>; @@ -1153,17 +1264,22 @@ impl<'a, T> IntoIterator for &'a BTreeSet { /// An iterator produced by calling `drain_filter` on BTreeSet. #[unstable(feature = "btree_drain_filter", issue = "70530")] -pub struct DrainFilter<'a, T, F> -where +pub struct DrainFilter< + 'a, + T, + F, + #[unstable(feature = "allocator_api", issue = "32838")] A: Allocator = Global, +> where T: 'a, F: 'a + FnMut(&T) -> bool, { pred: F, inner: super::map::DrainFilterInner<'a, T, ()>, + alloc: &'a A, } #[unstable(feature = "btree_drain_filter", issue = "70530")] -impl Drop for DrainFilter<'_, T, F> +impl Drop for DrainFilter<'_, T, F, A> where F: FnMut(&T) -> bool, { @@ -1173,7 +1289,7 @@ where } #[unstable(feature = "btree_drain_filter", issue = "70530")] -impl fmt::Debug for DrainFilter<'_, T, F> +impl fmt::Debug for DrainFilter<'_, T, F, A> where T: fmt::Debug, F: FnMut(&T) -> bool, @@ -1184,7 +1300,7 @@ where } #[unstable(feature = "btree_drain_filter", issue = "70530")] -impl<'a, T, F> Iterator for DrainFilter<'_, T, F> +impl<'a, T, F, A: Allocator> Iterator for DrainFilter<'_, T, F, A> where F: 'a + FnMut(&T) -> bool, { @@ -1193,7 +1309,7 @@ where fn next(&mut self) -> Option { let pred = &mut self.pred; let mut mapped_pred = |k: &T, _v: &mut ()| pred(k); - self.inner.next(&mut mapped_pred).map(|(k, _)| k) + self.inner.next(&mut mapped_pred, &self.alloc).map(|(k, _)| k) } fn size_hint(&self) -> (usize, Option) { @@ -1202,10 +1318,10 @@ where } #[unstable(feature = "btree_drain_filter", issue = "70530")] -impl FusedIterator for DrainFilter<'_, T, F> where F: FnMut(&T) -> bool {} +impl FusedIterator for DrainFilter<'_, T, F, A> where F: FnMut(&T) -> bool {} #[stable(feature = "rust1", since = "1.0.0")] -impl Extend for BTreeSet { +impl Extend for BTreeSet { #[inline] fn extend>(&mut self, iter: Iter) { iter.into_iter().for_each(move |elem| { @@ -1220,7 +1336,7 @@ impl Extend for BTreeSet { } #[stable(feature = "extend_ref", since = "1.2.0")] -impl<'a, T: 'a + Ord + Copy> Extend<&'a T> for BTreeSet { +impl<'a, T: 'a + Ord + Copy, A: Allocator> Extend<&'a T> for BTreeSet { fn extend>(&mut self, iter: I) { self.extend(iter.into_iter().cloned()); } @@ -1240,8 +1356,8 @@ impl Default for BTreeSet { } #[stable(feature = "rust1", since = "1.0.0")] -impl Sub<&BTreeSet> for &BTreeSet { - type Output = BTreeSet; +impl Sub<&BTreeSet> for &BTreeSet { + type Output = BTreeSet; /// Returns the difference of `self` and `rhs` as a new `BTreeSet`. /// @@ -1257,14 +1373,16 @@ impl Sub<&BTreeSet> for &BTreeSet { /// let result_vec: Vec<_> = result.into_iter().collect(); /// assert_eq!(result_vec, [1, 2]); /// ``` - fn sub(self, rhs: &BTreeSet) -> BTreeSet { - self.difference(rhs).cloned().collect() + fn sub(self, rhs: &BTreeSet) -> BTreeSet { + let mut set = BTreeSet::new_in(ManuallyDrop::into_inner(self.map.alloc.clone())); + set.extend(self.difference(rhs).cloned()); + set } } #[stable(feature = "rust1", since = "1.0.0")] -impl BitXor<&BTreeSet> for &BTreeSet { - type Output = BTreeSet; +impl BitXor<&BTreeSet> for &BTreeSet { + type Output = BTreeSet; /// Returns the symmetric difference of `self` and `rhs` as a new `BTreeSet`. /// @@ -1280,14 +1398,16 @@ impl BitXor<&BTreeSet> for &BTreeSet { /// let result_vec: Vec<_> = result.into_iter().collect(); /// assert_eq!(result_vec, [1, 4]); /// ``` - fn bitxor(self, rhs: &BTreeSet) -> BTreeSet { - self.symmetric_difference(rhs).cloned().collect() + fn bitxor(self, rhs: &BTreeSet) -> BTreeSet { + let mut set = BTreeSet::new_in(ManuallyDrop::into_inner(self.map.alloc.clone())); + set.extend(self.symmetric_difference(rhs).cloned()); + set } } #[stable(feature = "rust1", since = "1.0.0")] -impl BitAnd<&BTreeSet> for &BTreeSet { - type Output = BTreeSet; +impl BitAnd<&BTreeSet> for &BTreeSet { + type Output = BTreeSet; /// Returns the intersection of `self` and `rhs` as a new `BTreeSet`. /// @@ -1303,14 +1423,16 @@ impl BitAnd<&BTreeSet> for &BTreeSet { /// let result_vec: Vec<_> = result.into_iter().collect(); /// assert_eq!(result_vec, [2, 3]); /// ``` - fn bitand(self, rhs: &BTreeSet) -> BTreeSet { - self.intersection(rhs).cloned().collect() + fn bitand(self, rhs: &BTreeSet) -> BTreeSet { + let mut set = BTreeSet::new_in(ManuallyDrop::into_inner(self.map.alloc.clone())); + set.extend(self.intersection(rhs).cloned()); + set } } #[stable(feature = "rust1", since = "1.0.0")] -impl BitOr<&BTreeSet> for &BTreeSet { - type Output = BTreeSet; +impl BitOr<&BTreeSet> for &BTreeSet { + type Output = BTreeSet; /// Returns the union of `self` and `rhs` as a new `BTreeSet`. /// @@ -1326,13 +1448,15 @@ impl BitOr<&BTreeSet> for &BTreeSet { /// let result_vec: Vec<_> = result.into_iter().collect(); /// assert_eq!(result_vec, [1, 2, 3, 4, 5]); /// ``` - fn bitor(self, rhs: &BTreeSet) -> BTreeSet { - self.union(rhs).cloned().collect() + fn bitor(self, rhs: &BTreeSet) -> BTreeSet { + let mut set = BTreeSet::new_in(ManuallyDrop::into_inner(self.map.alloc.clone())); + set.extend(self.union(rhs).cloned()); + set } } #[stable(feature = "rust1", since = "1.0.0")] -impl Debug for BTreeSet { +impl Debug for BTreeSet { fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { f.debug_set().entries(self.iter()).finish() } @@ -1385,7 +1509,7 @@ impl ExactSizeIterator for Iter<'_, T> { impl FusedIterator for Iter<'_, T> {} #[stable(feature = "rust1", since = "1.0.0")] -impl Iterator for IntoIter { +impl Iterator for IntoIter { type Item = T; fn next(&mut self) -> Option { @@ -1397,20 +1521,20 @@ impl Iterator for IntoIter { } } #[stable(feature = "rust1", since = "1.0.0")] -impl DoubleEndedIterator for IntoIter { +impl DoubleEndedIterator for IntoIter { fn next_back(&mut self) -> Option { self.iter.next_back().map(|(k, _)| k) } } #[stable(feature = "rust1", since = "1.0.0")] -impl ExactSizeIterator for IntoIter { +impl ExactSizeIterator for IntoIter { fn len(&self) -> usize { self.iter.len() } } #[stable(feature = "fused", since = "1.26.0")] -impl FusedIterator for IntoIter {} +impl FusedIterator for IntoIter {} #[stable(feature = "btree_range", since = "1.17.0")] impl Clone for Range<'_, T> { @@ -1451,7 +1575,7 @@ impl<'a, T> DoubleEndedIterator for Range<'a, T> { impl FusedIterator for Range<'_, T> {} #[stable(feature = "rust1", since = "1.0.0")] -impl Clone for Difference<'_, T> { +impl Clone for Difference<'_, T, A> { fn clone(&self) -> Self { Difference { inner: match &self.inner { @@ -1468,7 +1592,7 @@ impl Clone for Difference<'_, T> { } } #[stable(feature = "rust1", since = "1.0.0")] -impl<'a, T: Ord> Iterator for Difference<'a, T> { +impl<'a, T: Ord, A: Allocator> Iterator for Difference<'a, T, A> { type Item = &'a T; fn next(&mut self) -> Option<&'a T> { @@ -1515,7 +1639,7 @@ impl<'a, T: Ord> Iterator for Difference<'a, T> { } #[stable(feature = "fused", since = "1.26.0")] -impl FusedIterator for Difference<'_, T> {} +impl FusedIterator for Difference<'_, T, A> {} #[stable(feature = "rust1", since = "1.0.0")] impl Clone for SymmetricDifference<'_, T> { @@ -1553,7 +1677,7 @@ impl<'a, T: Ord> Iterator for SymmetricDifference<'a, T> { impl FusedIterator for SymmetricDifference<'_, T> {} #[stable(feature = "rust1", since = "1.0.0")] -impl Clone for Intersection<'_, T> { +impl Clone for Intersection<'_, T, A> { fn clone(&self) -> Self { Intersection { inner: match &self.inner { @@ -1569,7 +1693,7 @@ impl Clone for Intersection<'_, T> { } } #[stable(feature = "rust1", since = "1.0.0")] -impl<'a, T: Ord> Iterator for Intersection<'a, T> { +impl<'a, T: Ord, A: Allocator> Iterator for Intersection<'a, T, A> { type Item = &'a T; fn next(&mut self) -> Option<&'a T> { @@ -1610,7 +1734,7 @@ impl<'a, T: Ord> Iterator for Intersection<'a, T> { } #[stable(feature = "fused", since = "1.26.0")] -impl FusedIterator for Intersection<'_, T> {} +impl FusedIterator for Intersection<'_, T, A> {} #[stable(feature = "rust1", since = "1.0.0")] impl Clone for Union<'_, T> { diff --git a/library/alloc/src/collections/btree/split.rs b/library/alloc/src/collections/btree/split.rs index bec495a72a4a8..3ccd1d1d8619b 100644 --- a/library/alloc/src/collections/btree/split.rs +++ b/library/alloc/src/collections/btree/split.rs @@ -1,5 +1,6 @@ use super::node::{ForceResult::*, Root}; use super::search::SearchResult::*; +use core::alloc::Allocator; use core::borrow::Borrow; impl Root { @@ -28,12 +29,12 @@ impl Root { /// and if the ordering of `Q` corresponds to that of `K`. /// If `self` respects all `BTreeMap` tree invariants, then both /// `self` and the returned tree will respect those invariants. - pub fn split_off(&mut self, key: &Q) -> Self + pub fn split_off(&mut self, key: &Q, alloc: &A) -> Self where K: Borrow, { let left_root = self; - let mut right_root = Root::new_pillar(left_root.height()); + let mut right_root = Root::new_pillar(left_root.height(), alloc); let mut left_node = left_root.borrow_mut(); let mut right_node = right_root.borrow_mut(); @@ -56,16 +57,16 @@ impl Root { } } - left_root.fix_right_border(); - right_root.fix_left_border(); + left_root.fix_right_border(alloc); + right_root.fix_left_border(alloc); right_root } /// Creates a tree consisting of empty nodes. - fn new_pillar(height: usize) -> Self { - let mut root = Root::new(); + fn new_pillar(height: usize, alloc: &A) -> Self { + let mut root = Root::new(alloc); for _ in 0..height { - root.push_internal_level(); + root.push_internal_level(alloc); } root } From 09511dcdbf8e83a2017c7f3115aef2cf362f1e07 Mon Sep 17 00:00:00 2001 From: Jacob Hughes Date: Mon, 14 Mar 2022 06:55:36 -0400 Subject: [PATCH 2/4] btreemap-alloc: fix clear impl --- library/alloc/src/collections/btree/map.rs | 21 ++++++--------------- 1 file changed, 6 insertions(+), 15 deletions(-) diff --git a/library/alloc/src/collections/btree/map.rs b/library/alloc/src/collections/btree/map.rs index 10a2f37d03743..1ec5edc20dbe5 100644 --- a/library/alloc/src/collections/btree/map.rs +++ b/library/alloc/src/collections/btree/map.rs @@ -571,11 +571,12 @@ impl BTreeMap { /// ``` #[stable(feature = "rust1", since = "1.0.0")] pub fn clear(&mut self) { - let alloc = unsafe { - // drop all elements and retrieve allocator - ptr::read(self).into_iter().into_alloc() - }; - *self = BTreeMap::new_in(alloc); + // avoid moving the allocator + mem::drop(BTreeMap { + root: mem::replace(&mut self.root, None), + length: mem::replace(&mut self.length, 0), + alloc: ManuallyDrop::new(&*self.alloc), + }); } /// Makes a new empty BTreeMap with a reasonable choice for B. @@ -1594,11 +1595,6 @@ impl IntoIterator for BTreeMap { #[stable(feature = "btree_drop", since = "1.7.0")] impl Drop for IntoIter { fn drop(&mut self) { - self.dealloc() - } -} -impl IntoIter { - fn dealloc(&mut self) { struct DropGuard<'a, K, V, A: Allocator>(&'a mut IntoIter); impl<'a, K, V, A: Allocator> Drop for DropGuard<'a, K, V, A> { @@ -1649,11 +1645,6 @@ impl IntoIter { Some(unsafe { self.range.deallocating_next_back_unchecked(&self.alloc) }) } } - fn into_alloc(mut self) -> A { - self.dealloc(); // Deallocate, then don't drop as drop will also call dealloc - let iter = ManuallyDrop::new(self); - unsafe { ptr::read(&iter.alloc) } - } } #[stable(feature = "rust1", since = "1.0.0")] From ea258225605c0cf7dc7986b66146f0fca22795ef Mon Sep 17 00:00:00 2001 From: Jacob Hughes Date: Mon, 14 Mar 2022 06:58:58 -0400 Subject: [PATCH 3/4] btreemap-alloc: adjust ui test --- .../variance-btree-invariant-types.stderr | 32 +++++++++---------- 1 file changed, 16 insertions(+), 16 deletions(-) diff --git a/src/test/ui/variance/variance-btree-invariant-types.stderr b/src/test/ui/variance/variance-btree-invariant-types.stderr index df519e25641c3..c12c1f523060b 100644 --- a/src/test/ui/variance/variance-btree-invariant-types.stderr +++ b/src/test/ui/variance/variance-btree-invariant-types.stderr @@ -124,8 +124,8 @@ error[E0308]: mismatched types LL | v | ^ lifetime mismatch | - = note: expected struct `std::collections::btree_map::OccupiedEntry<'_, &'new (), _>` - found struct `std::collections::btree_map::OccupiedEntry<'_, &'static (), _>` + = note: expected struct `std::collections::btree_map::OccupiedEntry<'_, &'new (), _, _>` + found struct `std::collections::btree_map::OccupiedEntry<'_, &'static (), _, _>` note: the lifetime `'new` as defined here... --> $DIR/variance-btree-invariant-types.rs:29:20 | @@ -139,8 +139,8 @@ error[E0308]: mismatched types LL | v | ^ lifetime mismatch | - = note: expected struct `std::collections::btree_map::OccupiedEntry<'_, _, &'new ()>` - found struct `std::collections::btree_map::OccupiedEntry<'_, _, &'static ()>` + = note: expected struct `std::collections::btree_map::OccupiedEntry<'_, _, &'new (), _>` + found struct `std::collections::btree_map::OccupiedEntry<'_, _, &'static (), _>` note: the lifetime `'new` as defined here... --> $DIR/variance-btree-invariant-types.rs:33:20 | @@ -154,8 +154,8 @@ error[E0308]: mismatched types LL | v | ^ lifetime mismatch | - = note: expected struct `std::collections::btree_map::OccupiedEntry<'_, &'static (), _>` - found struct `std::collections::btree_map::OccupiedEntry<'_, &'new (), _>` + = note: expected struct `std::collections::btree_map::OccupiedEntry<'_, &'static (), _, _>` + found struct `std::collections::btree_map::OccupiedEntry<'_, &'new (), _, _>` note: the lifetime `'new` as defined here... --> $DIR/variance-btree-invariant-types.rs:37:23 | @@ -169,8 +169,8 @@ error[E0308]: mismatched types LL | v | ^ lifetime mismatch | - = note: expected struct `std::collections::btree_map::OccupiedEntry<'_, _, &'static ()>` - found struct `std::collections::btree_map::OccupiedEntry<'_, _, &'new ()>` + = note: expected struct `std::collections::btree_map::OccupiedEntry<'_, _, &'static (), _>` + found struct `std::collections::btree_map::OccupiedEntry<'_, _, &'new (), _>` note: the lifetime `'new` as defined here... --> $DIR/variance-btree-invariant-types.rs:41:23 | @@ -184,8 +184,8 @@ error[E0308]: mismatched types LL | v | ^ lifetime mismatch | - = note: expected struct `std::collections::btree_map::VacantEntry<'_, &'new (), _>` - found struct `std::collections::btree_map::VacantEntry<'_, &'static (), _>` + = note: expected struct `std::collections::btree_map::VacantEntry<'_, &'new (), _, _>` + found struct `std::collections::btree_map::VacantEntry<'_, &'static (), _, _>` note: the lifetime `'new` as defined here... --> $DIR/variance-btree-invariant-types.rs:46:20 | @@ -199,8 +199,8 @@ error[E0308]: mismatched types LL | v | ^ lifetime mismatch | - = note: expected struct `std::collections::btree_map::VacantEntry<'_, _, &'new ()>` - found struct `std::collections::btree_map::VacantEntry<'_, _, &'static ()>` + = note: expected struct `std::collections::btree_map::VacantEntry<'_, _, &'new (), _>` + found struct `std::collections::btree_map::VacantEntry<'_, _, &'static (), _>` note: the lifetime `'new` as defined here... --> $DIR/variance-btree-invariant-types.rs:50:20 | @@ -214,8 +214,8 @@ error[E0308]: mismatched types LL | v | ^ lifetime mismatch | - = note: expected struct `std::collections::btree_map::VacantEntry<'_, &'static (), _>` - found struct `std::collections::btree_map::VacantEntry<'_, &'new (), _>` + = note: expected struct `std::collections::btree_map::VacantEntry<'_, &'static (), _, _>` + found struct `std::collections::btree_map::VacantEntry<'_, &'new (), _, _>` note: the lifetime `'new` as defined here... --> $DIR/variance-btree-invariant-types.rs:54:23 | @@ -229,8 +229,8 @@ error[E0308]: mismatched types LL | v | ^ lifetime mismatch | - = note: expected struct `std::collections::btree_map::VacantEntry<'_, _, &'static ()>` - found struct `std::collections::btree_map::VacantEntry<'_, _, &'new ()>` + = note: expected struct `std::collections::btree_map::VacantEntry<'_, _, &'static (), _>` + found struct `std::collections::btree_map::VacantEntry<'_, _, &'new (), _>` note: the lifetime `'new` as defined here... --> $DIR/variance-btree-invariant-types.rs:58:23 | From 713f4d5cd2feb4682db77662a79d1fd66325e771 Mon Sep 17 00:00:00 2001 From: Jacob Hughes Date: Mon, 14 Mar 2022 21:08:33 -0400 Subject: [PATCH 4/4] hack/workaround: disable LTO --- src/bootstrap/builder.rs | 1 + 1 file changed, 1 insertion(+) diff --git a/src/bootstrap/builder.rs b/src/bootstrap/builder.rs index 32ccca8bcdd2c..e3b7dc670a966 100644 --- a/src/bootstrap/builder.rs +++ b/src/bootstrap/builder.rs @@ -1684,6 +1684,7 @@ impl<'a> Builder<'a> { rustflags.arg(&format!("-Cllvm-args=-import-instr-limit={}", limit)); } } + cargo.env(profile_var("LTO"), "off"); Cargo { command: cargo, rustflags, rustdocflags } }