load_accounts_index_for_shrink uses scan (#26260)
This commit is contained in:
parent
b3406b5b2a
commit
ca83f143aa
|
@ -29,9 +29,9 @@ use {
|
||||||
},
|
},
|
||||||
accounts_index::{
|
accounts_index::{
|
||||||
AccountIndexGetResult, AccountSecondaryIndexes, AccountsIndex, AccountsIndexConfig,
|
AccountIndexGetResult, AccountSecondaryIndexes, AccountsIndex, AccountsIndexConfig,
|
||||||
AccountsIndexRootsStats, IndexKey, IndexValue, IsCached, RefCount, ScanConfig,
|
AccountsIndexRootsStats, AccountsIndexScanResult, IndexKey, IndexValue, IsCached,
|
||||||
ScanResult, SlotList, SlotSlice, ZeroLamport, ACCOUNTS_INDEX_CONFIG_FOR_BENCHMARKS,
|
RefCount, ScanConfig, ScanResult, SlotList, SlotSlice, ZeroLamport,
|
||||||
ACCOUNTS_INDEX_CONFIG_FOR_TESTING,
|
ACCOUNTS_INDEX_CONFIG_FOR_BENCHMARKS, ACCOUNTS_INDEX_CONFIG_FOR_TESTING,
|
||||||
},
|
},
|
||||||
accounts_index_storage::Startup,
|
accounts_index_storage::Startup,
|
||||||
accounts_update_notifier_interface::AccountsUpdateNotifier,
|
accounts_update_notifier_interface::AccountsUpdateNotifier,
|
||||||
|
@ -2592,7 +2592,11 @@ impl AccountsDb {
|
||||||
if !useless {
|
if !useless {
|
||||||
useful += 1;
|
useful += 1;
|
||||||
}
|
}
|
||||||
!useless
|
if useless {
|
||||||
|
AccountsIndexScanResult::None
|
||||||
|
} else {
|
||||||
|
AccountsIndexScanResult::KeepInMemory
|
||||||
|
}
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
found_not_zero_accum.fetch_add(found_not_zero, Ordering::Relaxed);
|
found_not_zero_accum.fetch_add(found_not_zero, Ordering::Relaxed);
|
||||||
|
@ -3014,15 +3018,18 @@ impl AccountsDb {
|
||||||
|
|
||||||
let mut alive = 0;
|
let mut alive = 0;
|
||||||
let mut dead = 0;
|
let mut dead = 0;
|
||||||
|
let mut index = 0;
|
||||||
|
self.accounts_index.scan(
|
||||||
accounts[..std::cmp::min(accounts.len(), count)]
|
accounts[..std::cmp::min(accounts.len(), count)]
|
||||||
.iter()
|
.iter()
|
||||||
.for_each(|pair| {
|
.map(|(key, _)| key),
|
||||||
let pubkey = &pair.0;
|
// return true if we want this item to remain in the cache
|
||||||
|
|exists, slot_list, pubkey, _ref_count| {
|
||||||
|
let mut result = AccountsIndexScanResult::None;
|
||||||
|
if exists {
|
||||||
|
let pair = &accounts[index];
|
||||||
let stored_account = &pair.1;
|
let stored_account = &pair.1;
|
||||||
let lookup = self.accounts_index.get_account_read_entry(pubkey);
|
let is_alive = slot_list.iter().any(|(_slot, acct_info)| {
|
||||||
if let Some(locked_entry) = lookup {
|
|
||||||
let is_alive = locked_entry.slot_list().iter().any(|(_slot, acct_info)| {
|
|
||||||
acct_info.matches_storage_location(
|
acct_info.matches_storage_location(
|
||||||
stored_account.store_id,
|
stored_account.store_id,
|
||||||
stored_account.account.offset,
|
stored_account.account.offset,
|
||||||
|
@ -3036,7 +3043,7 @@ impl AccountsDb {
|
||||||
if let Some(unrefed_pubkeys) = &mut unrefed_pubkeys {
|
if let Some(unrefed_pubkeys) = &mut unrefed_pubkeys {
|
||||||
unrefed_pubkeys.push(pubkey);
|
unrefed_pubkeys.push(pubkey);
|
||||||
}
|
}
|
||||||
locked_entry.unref();
|
result = AccountsIndexScanResult::Unref;
|
||||||
dead += 1;
|
dead += 1;
|
||||||
} else {
|
} else {
|
||||||
alive_accounts.push(pair);
|
alive_accounts.push(pair);
|
||||||
|
@ -3044,7 +3051,11 @@ impl AccountsDb {
|
||||||
alive += 1;
|
alive += 1;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
});
|
index += 1;
|
||||||
|
result
|
||||||
|
},
|
||||||
|
);
|
||||||
|
assert_eq!(index, std::cmp::min(accounts.len(), count));
|
||||||
self.shrink_stats
|
self.shrink_stats
|
||||||
.alive_accounts
|
.alive_accounts
|
||||||
.fetch_add(alive, Ordering::Relaxed);
|
.fetch_add(alive, Ordering::Relaxed);
|
||||||
|
|
|
@ -639,6 +639,15 @@ impl ScanSlotTracker {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub enum AccountsIndexScanResult {
|
||||||
|
/// if the entry is not in the in-memory index, do not add it, make no modifications to it
|
||||||
|
None,
|
||||||
|
/// keep the entry in the in-memory index
|
||||||
|
KeepInMemory,
|
||||||
|
/// reduce refcount by 1
|
||||||
|
Unref,
|
||||||
|
}
|
||||||
|
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
pub struct AccountsIndex<T: IndexValue> {
|
pub struct AccountsIndex<T: IndexValue> {
|
||||||
pub account_maps: LockMapType<T>,
|
pub account_maps: LockMapType<T>,
|
||||||
|
@ -1295,13 +1304,12 @@ impl<T: IndexValue> AccountsIndex<T> {
|
||||||
/// call `callback`
|
/// call `callback`
|
||||||
pub(crate) fn scan<'a, F, I>(&'a self, pubkeys: I, mut callback: F)
|
pub(crate) fn scan<'a, F, I>(&'a self, pubkeys: I, mut callback: F)
|
||||||
where
|
where
|
||||||
// return true if accounts index entry should be put in in_mem cache
|
|
||||||
// params:
|
// params:
|
||||||
// exists: false if not in index at all
|
// exists: false if not in index at all
|
||||||
// index in slot list where best slot was found or None if nothing found by root criteria
|
// index in slot list where best slot was found or None if nothing found by root criteria
|
||||||
// pubkey looked up
|
// pubkey looked up
|
||||||
// refcount of entry in index
|
// refcount of entry in index
|
||||||
F: FnMut(bool, &SlotList<T>, &Pubkey, RefCount) -> bool,
|
F: FnMut(bool, &SlotList<T>, &'a Pubkey, RefCount) -> AccountsIndexScanResult,
|
||||||
I: IntoIterator<Item = &'a Pubkey>,
|
I: IntoIterator<Item = &'a Pubkey>,
|
||||||
{
|
{
|
||||||
let empty_slot_list = vec![];
|
let empty_slot_list = vec![];
|
||||||
|
@ -1315,13 +1323,24 @@ impl<T: IndexValue> AccountsIndex<T> {
|
||||||
last_bin = bin;
|
last_bin = bin;
|
||||||
}
|
}
|
||||||
lock.as_ref().unwrap().get_internal(pubkey, |entry| {
|
lock.as_ref().unwrap().get_internal(pubkey, |entry| {
|
||||||
let cache = match entry {
|
let mut cache = false;
|
||||||
|
match entry {
|
||||||
Some(locked_entry) => {
|
Some(locked_entry) => {
|
||||||
let slot_list = &locked_entry.slot_list.read().unwrap();
|
let slot_list = &locked_entry.slot_list.read().unwrap();
|
||||||
callback(true, slot_list, pubkey, locked_entry.ref_count())
|
let result = callback(true, slot_list, pubkey, locked_entry.ref_count());
|
||||||
|
cache = match result {
|
||||||
|
AccountsIndexScanResult::Unref => {
|
||||||
|
locked_entry.add_un_ref(false);
|
||||||
|
true
|
||||||
}
|
}
|
||||||
None => callback(false, &empty_slot_list, pubkey, RefCount::MAX),
|
AccountsIndexScanResult::KeepInMemory => true,
|
||||||
|
AccountsIndexScanResult::None => false,
|
||||||
};
|
};
|
||||||
|
}
|
||||||
|
None => {
|
||||||
|
callback(false, &empty_slot_list, pubkey, RefCount::MAX);
|
||||||
|
}
|
||||||
|
}
|
||||||
(cache, ())
|
(cache, ())
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
Loading…
Reference in New Issue