diff --git a/bucket_map/src/bucket.rs b/bucket_map/src/bucket.rs index bcca7b605c588b..b92231c6da80de 100644 --- a/bucket_map/src/bucket.rs +++ b/bucket_map/src/bucket.rs @@ -96,6 +96,10 @@ pub struct Bucket { pub data: Vec>, stats: Arc, + /// # entries caller expects the map to need to contain. + /// Used as a hint for the next time we need to grow. + anticipated_size: AtomicU64, + pub reallocated: Reallocated, DataBucket>, } @@ -123,6 +127,7 @@ impl<'b, T: Clone + Copy + 'static> Bucket { data: vec![], stats, reallocated: Reallocated::default(), + anticipated_size: AtomicU64::default(), } } @@ -420,8 +425,20 @@ impl<'b, T: Clone + Copy + 'static> Bucket { } } + pub(crate) fn set_anticipated_count(&self, count: u64) { + self.anticipated_size.store(count, Ordering::Release); + } + pub fn grow_index(&self, current_capacity_pow2: u8) { if self.index.capacity_pow2 == current_capacity_pow2 { + // make sure to grow to at least % more than the anticpated size + let mut starting_size_pow2 = self.index.capacity_pow2; + let anticipated_size = self.anticipated_size.load(Ordering::Acquire); + if anticipated_size > 0 { + // start the growth at the next pow2 larger than what would be required to hold `anticipated_size`. + // This will prevent unnecessary repeated grows at startup. + starting_size_pow2 = starting_size_pow2.max(anticipated_size.ilog2() as u8); + } let mut m = Measure::start("grow_index"); //debug!("GROW_INDEX: {}", current_capacity_pow2); let increment = 1; @@ -434,7 +451,7 @@ impl<'b, T: Clone + Copy + 'static> Bucket { 1, std::mem::size_of::>() as u64, // *2 causes rapid growth of index buckets - self.index.capacity_pow2 + i, // * 2, + starting_size_pow2 + i, // * 2, self.index.max_search, Arc::clone(&self.stats.index), Arc::clone(&self.index.count), diff --git a/bucket_map/src/bucket_api.rs b/bucket_map/src/bucket_api.rs index c4be1cd24e3d8b..6b36e2cfa5b54c 100644 --- a/bucket_map/src/bucket_api.rs +++ b/bucket_map/src/bucket_api.rs @@ -111,6 +111,19 @@ impl BucketApi { } } + /// caller can specify that the index needs to hold approximately `count` entries soon. + /// This gives a hint to the resizing algorithm and prevents repeated incremental resizes. + pub fn set_anticipated_count(&self, count: u64) { + if let Some(bucket) = self.bucket.read().unwrap().as_ref() { + bucket.set_anticipated_count(count); + return; + } + // caller is about to write a lot of elements to this bucket, so we need to allocate it first + // so we have somewhere to store the anticipated count + let mut bucket = self.get_write_bucket(); + bucket.as_mut().unwrap().set_anticipated_count(count); + } + pub fn update(&self, key: &Pubkey, updatefn: F) where F: FnMut(Option<(&[T], RefCount)>) -> Option<(Vec, RefCount)>, diff --git a/runtime/src/in_mem_accounts_index.rs b/runtime/src/in_mem_accounts_index.rs index 5ebdf760227592..8217f76489791d 100644 --- a/runtime/src/in_mem_accounts_index.rs +++ b/runtime/src/in_mem_accounts_index.rs @@ -1061,6 +1061,9 @@ impl + Into> InMemAccountsIndex + Into> InMemAccountsIndex