From e3ac5de377c55ce6b90bbf361fec3275b94dba12 Mon Sep 17 00:00:00 2001 From: Corwin Date: Thu, 4 Aug 2022 19:26:55 +0100 Subject: [PATCH 1/3] make the allocators clonable --- agb/src/agb_alloc/mod.rs | 2 ++ 1 file changed, 2 insertions(+) diff --git a/agb/src/agb_alloc/mod.rs b/agb/src/agb_alloc/mod.rs index d4e59fcc..79b10070 100644 --- a/agb/src/agb_alloc/mod.rs +++ b/agb/src/agb_alloc/mod.rs @@ -78,6 +78,7 @@ macro_rules! impl_zst_allocator { /// ); /// # } /// ``` +#[derive(Clone)] pub struct ExternalAllocator; impl_zst_allocator!(ExternalAllocator, GLOBAL_ALLOC); @@ -102,6 +103,7 @@ impl_zst_allocator!(ExternalAllocator, GLOBAL_ALLOC); /// ); /// # } /// ``` +#[derive(Clone)] pub struct InternalAllocator; impl_zst_allocator!(InternalAllocator, __IWRAM_ALLOC); From 5b3d288079757a4fe72a63522fad4a9d1ad51b70 Mon Sep 17 00:00:00 2001 From: Corwin Date: Thu, 4 Aug 2022 19:27:08 +0100 Subject: [PATCH 2/3] support custom allocators in hashmap --- agb/src/hash_map.rs | 235 ++++++++++++++++++++++++++++++++++++-------- 1 file changed, 192 insertions(+), 43 deletions(-) diff --git a/agb/src/hash_map.rs b/agb/src/hash_map.rs index 2b160703..45939093 100644 --- a/agb/src/hash_map.rs +++ b/agb/src/hash_map.rs @@ -1,10 +1,11 @@ #![deny(missing_docs)] //! A lot of the documentation for this module was copied straight out of the rust //! standard library. The implementation however is not. -use alloc::vec::Vec; +use alloc::{alloc::Global, vec::Vec}; use core::{ + alloc::Allocator, hash::{BuildHasher, BuildHasherDefault, Hash, Hasher}, - iter::{self, FromIterator}, + iter::FromIterator, mem::{self, MaybeUninit}, ops::Index, ptr, @@ -82,36 +83,67 @@ type HashType = u32; /// /// [`Eq`]: https://doc.rust-lang.org/core/cmp/trait.Eq.html /// [`Hash`]: https://doc.rust-lang.org/core/hash/trait.Hash.html -pub struct HashMap { - nodes: NodeStorage, +pub struct HashMap { + nodes: NodeStorage, hasher: BuildHasherDefault, } +/// Trait for allocators that are clonable, blanket implementation for all types that implement Allocator and Clone +pub trait ClonableAllocator: Allocator + Clone {} +impl ClonableAllocator for T {} + impl HashMap { /// Creates a `HashMap` #[must_use] pub fn new() -> Self { - Self::with_size(16) + Self::new_in(Global) } /// Creates an empty `HashMap` with specified internal size. The size must be a power of 2 #[must_use] pub fn with_size(size: usize) -> Self { - Self { - nodes: NodeStorage::with_size(size), - hasher: Default::default(), - } + Self::with_size_in(size, Global) } /// Creates an empty `HashMap` which can hold at least `capacity` elements before resizing. The actual /// internal size may be larger as it must be a power of 2 #[must_use] pub fn with_capacity(capacity: usize) -> Self { + Self::with_capacity_in(capacity, Global) + } +} + +impl HashMap { + #[must_use] + /// Creates an empty `HashMap` with specified internal size using the + /// specified allocator. The size must be a power of 2 + pub fn with_size_in(size: usize, alloc: ALLOCATOR) -> Self { + Self { + nodes: NodeStorage::with_size_in(size, alloc), + hasher: Default::default(), + } + } + + #[must_use] + /// Creates a `HashMap` with a specified allocator + pub fn new_in(alloc: ALLOCATOR) -> Self { + Self::with_size_in(16, alloc) + } + + /// Returns a reference to the underlying allocator + pub fn allocator(&self) -> &ALLOCATOR { + self.nodes.allocator() + } + + /// Creates an empty `HashMap` which can hold at least `capacity` elements before resizing. The actual + /// internal size may be larger as it must be a power of 2 + #[must_use] + pub fn with_capacity_in(capacity: usize, alloc: ALLOCATOR) -> Self { for i in 0..32 { let attempted_size = 1usize << i; if number_before_resize(attempted_size) > capacity { - return Self::with_size(attempted_size); + return Self::with_size_in(attempted_size, alloc); } } @@ -150,7 +182,8 @@ impl HashMap { /// Removes all elements from the map pub fn clear(&mut self) { - self.nodes = NodeStorage::with_size(self.nodes.backing_vec_size()); + self.nodes = + NodeStorage::with_size_in(self.nodes.backing_vec_size(), self.allocator().clone()); } /// An iterator visiting all key-value pairs in an arbitrary order @@ -193,7 +226,7 @@ const fn fast_mod(len: usize, hash: HashType) -> usize { (hash as usize) & (len - 1) } -impl HashMap +impl HashMap where K: Eq + Hash, { @@ -281,7 +314,7 @@ where } } -impl HashMap +impl HashMap where K: Hash, { @@ -296,12 +329,12 @@ where /// /// This struct is created using the `into_iter()` method on [`HashMap`]. See its /// documentation for more. -pub struct Iter<'a, K: 'a, V: 'a> { - map: &'a HashMap, +pub struct Iter<'a, K: 'a, V: 'a, ALLOCATOR: ClonableAllocator> { + map: &'a HashMap, at: usize, } -impl<'a, K, V> Iterator for Iter<'a, K, V> { +impl<'a, K, V, ALLOCATOR: ClonableAllocator> Iterator for Iter<'a, K, V, ALLOCATOR> { type Item = (&'a K, &'a V); fn next(&mut self) -> Option { @@ -320,9 +353,9 @@ impl<'a, K, V> Iterator for Iter<'a, K, V> { } } -impl<'a, K, V> IntoIterator for &'a HashMap { +impl<'a, K, V, ALLOCATOR: ClonableAllocator> IntoIterator for &'a HashMap { type Item = (&'a K, &'a V); - type IntoIter = Iter<'a, K, V>; + type IntoIter = Iter<'a, K, V, ALLOCATOR>; fn into_iter(self) -> Self::IntoIter { Iter { map: self, at: 0 } @@ -333,12 +366,12 @@ impl<'a, K, V> IntoIterator for &'a HashMap { /// /// This struct is created using the `into_iter()` method on [`HashMap`] as part of its implementation /// of the IntoIterator trait. -pub struct IterOwned { - map: HashMap, +pub struct IterOwned { + map: HashMap, at: usize, } -impl Iterator for IterOwned { +impl Iterator for IterOwned { type Item = (K, V); fn next(&mut self) -> Option { @@ -361,9 +394,9 @@ impl Iterator for IterOwned { /// /// This struct is created using the `into_iter()` method on [`HashMap`] as part of its implementation /// of the IntoIterator trait. -impl IntoIterator for HashMap { +impl IntoIterator for HashMap { type Item = (K, V); - type IntoIter = IterOwned; + type IntoIter = IterOwned; fn into_iter(self) -> Self::IntoIter { IterOwned { map: self, at: 0 } @@ -371,13 +404,13 @@ impl IntoIterator for HashMap { } /// A view into an occupied entry in a `HashMap`. This is part of the [`Entry`] enum. -pub struct OccupiedEntry<'a, K: 'a, V: 'a> { +pub struct OccupiedEntry<'a, K: 'a, V: 'a, ALLOCATOR: Allocator> { key: K, - map: &'a mut HashMap, + map: &'a mut HashMap, location: usize, } -impl<'a, K: 'a, V: 'a> OccupiedEntry<'a, K, V> { +impl<'a, K: 'a, V: 'a, ALLOCATOR: ClonableAllocator> OccupiedEntry<'a, K, V, ALLOCATOR> { /// Gets a reference to the key in the entry. pub fn key(&self) -> &K { &self.key @@ -426,12 +459,12 @@ impl<'a, K: 'a, V: 'a> OccupiedEntry<'a, K, V> { } /// A view into a vacant entry in a `HashMap`. It is part of the [`Entry`] enum. -pub struct VacantEntry<'a, K: 'a, V: 'a> { +pub struct VacantEntry<'a, K: 'a, V: 'a, ALLOCATOR: Allocator> { key: K, - map: &'a mut HashMap, + map: &'a mut HashMap, } -impl<'a, K: 'a, V: 'a> VacantEntry<'a, K, V> { +impl<'a, K: 'a, V: 'a, ALLOCATOR: ClonableAllocator> VacantEntry<'a, K, V, ALLOCATOR> { /// Gets a reference to the key that would be used when inserting a value through `VacantEntry` pub fn key(&self) -> &K { &self.key @@ -456,14 +489,14 @@ impl<'a, K: 'a, V: 'a> VacantEntry<'a, K, V> { /// This is constructed using the [`entry`] method on [`HashMap`] /// /// [`entry`]: HashMap::entry() -pub enum Entry<'a, K: 'a, V: 'a> { +pub enum Entry<'a, K: 'a, V: 'a, ALLOCATOR: Allocator = Global> { /// An occupied entry - Occupied(OccupiedEntry<'a, K, V>), + Occupied(OccupiedEntry<'a, K, V, ALLOCATOR>), /// A vacant entry - Vacant(VacantEntry<'a, K, V>), + Vacant(VacantEntry<'a, K, V, ALLOCATOR>), } -impl<'a, K, V> Entry<'a, K, V> +impl<'a, K, V, ALLOCATOR: ClonableAllocator> Entry<'a, K, V, ALLOCATOR> where K: Hash + Eq, { @@ -543,12 +576,12 @@ where } } -impl HashMap +impl HashMap where K: Hash + Eq, { /// Gets the given key's corresponding entry in the map for in-place manipulation. - pub fn entry(&mut self, key: K) -> Entry<'_, K, V> { + pub fn entry(&mut self, key: K) -> Entry<'_, K, V, ALLOCATOR> { let hash = self.hash(&key); let location = self.nodes.location(&key, hash); @@ -586,7 +619,7 @@ where } } -impl Index<&K> for HashMap +impl Index<&K> for HashMap where K: Eq + Hash, { @@ -597,7 +630,7 @@ where } } -impl Index for HashMap +impl Index for HashMap where K: Eq + Hash, { @@ -612,26 +645,35 @@ const fn number_before_resize(capacity: usize) -> usize { capacity * 85 / 100 } -struct NodeStorage { - nodes: Vec>, +struct NodeStorage { + nodes: Vec, ALLOCATOR>, max_distance_to_initial_bucket: i32, number_of_items: usize, max_number_before_resize: usize, } -impl NodeStorage { - fn with_size(capacity: usize) -> Self { +impl NodeStorage { + fn with_size_in(capacity: usize, alloc: ALLOCATOR) -> Self { assert!(capacity.is_power_of_two(), "Capacity must be a power of 2"); + let mut nodes = Vec::with_capacity_in(capacity, alloc); + for _ in 0..capacity { + nodes.push(Default::default()); + } + Self { - nodes: iter::repeat_with(Default::default).take(capacity).collect(), + nodes, max_distance_to_initial_bucket: 0, number_of_items: 0, max_number_before_resize: number_before_resize(capacity), } } + fn allocator(&self) -> &ALLOCATOR { + self.nodes.allocator() + } + fn capacity(&self) -> usize { self.max_number_before_resize } @@ -731,7 +773,7 @@ impl NodeStorage { } fn resized_to(&mut self, new_size: usize) -> Self { - let mut new_node_storage = Self::with_size(new_size); + let mut new_node_storage = Self::with_size_in(new_size, self.allocator().clone()); for mut node in self.nodes.drain(..) { if let Some((key, value, hash)) = node.take_key_value() { @@ -1285,4 +1327,111 @@ mod test { assert_eq!(map[&2], 1); } } + + mod rust_std_tests_custom_allocator { + use crate::{ + hash_map::{Entry::*, HashMap}, + Gba, InternalAllocator, + }; + + #[test_case] + fn test_entry(_gba: &mut Gba) { + let xs = [(1, 10), (2, 20), (3, 30), (4, 40), (5, 50), (6, 60)]; + + let mut map = HashMap::new_in(InternalAllocator); + for (k, v) in xs { + map.insert(k, v); + } + + // Existing key (insert) + match map.entry(1) { + Vacant(_) => unreachable!(), + Occupied(mut view) => { + assert_eq!(view.get(), &10); + assert_eq!(view.insert(100), 10); + } + } + assert_eq!(map.get(&1).unwrap(), &100); + assert_eq!(map.len(), 6); + + // Existing key (update) + match map.entry(2) { + Vacant(_) => unreachable!(), + Occupied(mut view) => { + let v = view.get_mut(); + let new_v = (*v) * 10; + *v = new_v; + } + } + assert_eq!(map.get(&2).unwrap(), &200); + assert_eq!(map.len(), 6); + + // Existing key (take) + match map.entry(3) { + Vacant(_) => unreachable!(), + Occupied(view) => { + assert_eq!(view.remove(), 30); + } + } + assert_eq!(map.get(&3), None); + assert_eq!(map.len(), 5); + + // Inexistent key (insert) + match map.entry(10) { + Occupied(_) => unreachable!(), + Vacant(view) => { + assert_eq!(*view.insert(1000), 1000); + } + } + assert_eq!(map.get(&10).unwrap(), &1000); + assert_eq!(map.len(), 6); + } + + #[test_case] + fn test_occupied_entry_key(_gba: &mut Gba) { + let mut a = HashMap::new_in(InternalAllocator); + let key = "hello there"; + let value = "value goes here"; + assert!(a.is_empty()); + a.insert(key, value); + assert_eq!(a.len(), 1); + assert_eq!(a[key], value); + + match a.entry(key) { + Vacant(_) => panic!(), + Occupied(e) => assert_eq!(key, *e.key()), + } + assert_eq!(a.len(), 1); + assert_eq!(a[key], value); + } + + #[test_case] + fn test_vacant_entry_key(_gba: &mut Gba) { + let mut a = HashMap::new_in(InternalAllocator); + let key = "hello there"; + let value = "value goes here"; + + assert!(a.is_empty()); + match a.entry(key) { + Occupied(_) => panic!(), + Vacant(e) => { + assert_eq!(key, *e.key()); + e.insert(value); + } + } + assert_eq!(a.len(), 1); + assert_eq!(a[key], value); + } + + #[test_case] + fn test_index(_gba: &mut Gba) { + let mut map = HashMap::new_in(InternalAllocator); + + map.insert(1, 2); + map.insert(2, 1); + map.insert(3, 4); + + assert_eq!(map[&2], 1); + } + } } From 194c40dc2e6ff8903e3698494fe25d8b0a84b390 Mon Sep 17 00:00:00 2001 From: Corwin Date: Thu, 4 Aug 2022 20:08:39 +0100 Subject: [PATCH 3/3] add entry for allocators in hashmap in changelog --- CHANGELOG.md | 3 +++ 1 file changed, 3 insertions(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index f0b937ed..6ef9e18b 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -6,6 +6,9 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 ## [Unreleased] +### Added +- Custom allocator support using the `Allocator` trait for `HashMap`. This means the `HashMap` can be used with `InternalAllocator` to allocate to IWRAM or the `ExternalAllocator` to explicitly allocate to EWRAM. + ## [0.11.1] - 2022/08/02 Version 0.11.1 brings documentation for fixed point numbers. We recommend all users upgrade to this version since it also includes fixes to a few functions in fixnum. See changed section for breaking changes.