Skip to content

Commit

Permalink
Fixes rent collection when skipping rewrites (#2910)
Browse files Browse the repository at this point in the history
  • Loading branch information
brooksprumo authored Sep 19, 2024
1 parent 7c54e92 commit c1b465d
Show file tree
Hide file tree
Showing 2 changed files with 55 additions and 61 deletions.
38 changes: 20 additions & 18 deletions runtime/src/bank.rs
Original file line number Diff line number Diff line change
Expand Up @@ -4304,30 +4304,37 @@ impl Bank {
let mut time_collecting_rent_us = 0;
let mut time_storing_accounts_us = 0;
let can_skip_rewrites = self.bank_hash_skips_rent_rewrites();
let test_skip_rewrites_but_include_hash_in_bank_hash = !can_skip_rewrites
&& self
.rc
.accounts
.accounts_db
.test_skip_rewrites_but_include_in_bank_hash;
let test_skip_rewrites_but_include_in_bank_hash = self
.rc
.accounts
.accounts_db
.test_skip_rewrites_but_include_in_bank_hash;
let mut skipped_rewrites = Vec::default();
for (pubkey, account, _loaded_slot) in accounts.iter_mut() {
let rent_epoch_pre = account.rent_epoch();
let (rent_collected_info, collect_rent_us) = measure_us!(collect_rent_from_account(
&self.feature_set,
&self.rent_collector,
pubkey,
account
));
time_collecting_rent_us += collect_rent_us;
let rent_epoch_post = account.rent_epoch();

// did the account change in any way due to rent collection?
let account_changed =
rent_collected_info.rent_amount != 0 || rent_epoch_post != rent_epoch_pre;

// always store the account, regardless if it changed or not
let always_store_accounts =
!can_skip_rewrites && !test_skip_rewrites_but_include_in_bank_hash;

// only store accounts where we collected rent
// but get the hash for all these accounts even if collected rent is 0 (= not updated).
// Also, there's another subtle side-effect from rewrites: this
// ensures we verify the whole on-chain state (= all accounts)
// via the bank delta hash slowly once per an epoch.
if (!can_skip_rewrites && !test_skip_rewrites_but_include_hash_in_bank_hash)
|| !Self::skip_rewrite(rent_collected_info.rent_amount, account)
{
if account_changed || always_store_accounts {
if rent_collected_info.rent_amount > 0 {
if let Some(rent_paying_pubkeys) = rent_paying_pubkeys {
if !rent_paying_pubkeys.contains(pubkey) {
Expand Down Expand Up @@ -4357,7 +4364,10 @@ impl Bank {
}
total_rent_collected_info += rent_collected_info;
accounts_to_store.push((pubkey, account));
} else if test_skip_rewrites_but_include_hash_in_bank_hash {
} else if !account_changed
&& !can_skip_rewrites
&& test_skip_rewrites_but_include_in_bank_hash
{
// include rewrites that we skipped in the accounts delta hash.
// This is what consensus requires prior to activation of bank_hash_skips_rent_rewrites.
// This code path exists to allow us to test the long term effects on validators when the skipped rewrites
Expand Down Expand Up @@ -4513,14 +4523,6 @@ impl Bank {
});
}

/// return true iff storing this account is just a rewrite and can be skipped
fn skip_rewrite(rent_amount: u64, account: &AccountSharedData) -> bool {
// if rent was != 0
// or special case for default rent value
// these cannot be skipped and must be written
rent_amount == 0 && account.rent_epoch() != 0
}

pub(crate) fn fixed_cycle_partitions_between_slots(
&self,
starting_slot: Slot,
Expand Down
78 changes: 35 additions & 43 deletions runtime/src/bank/tests.rs
Original file line number Diff line number Diff line change
Expand Up @@ -1762,7 +1762,9 @@ fn test_collect_rent_from_accounts() {
solana_logger::setup();

for skip_rewrites in [false, true] {
let zero_lamport_pubkey = Pubkey::from([0; 32]);
let address1 = Pubkey::new_unique();
let address2 = Pubkey::new_unique();
let address3 = Pubkey::new_unique();

let (genesis_bank, bank_forks) = create_simple_test_arc_bank(100000);
let mut first_bank = new_from_parent(genesis_bank.clone());
Expand All @@ -1784,12 +1786,20 @@ fn test_collect_rent_from_accounts() {

let data_size = 0; // make sure we're rent exempt
let lamports = later_bank.get_minimum_balance_for_rent_exemption(data_size); // cannot be 0 or we zero out rent_epoch in rent collection and we need to be rent exempt
let mut account = AccountSharedData::new(lamports, data_size, &Pubkey::default());
account.set_rent_epoch(later_bank.epoch() - 1); // non-zero, but less than later_bank's epoch
let mut account1 = AccountSharedData::new(lamports, data_size, &Pubkey::default());
let mut account2 = AccountSharedData::new(lamports, data_size, &Pubkey::default());
let mut account3 = AccountSharedData::new(lamports, data_size, &Pubkey::default());
account1.set_rent_epoch(later_bank.epoch() - 1); // non-zero, but less than later_bank's epoch
account2.set_rent_epoch(RENT_EXEMPT_RENT_EPOCH); // already marked as rent exempt
account3.set_rent_epoch(0); // stake accounts in genesis have a rent epoch of 0

// loaded from previous slot, so we skip rent collection on it
let _result = later_bank.collect_rent_from_accounts(
vec![(zero_lamport_pubkey, account, later_slot - 1)],
vec![
(address1, account1, later_slot - 1),
(address2, account2, later_slot - 1),
(address3, account3, later_slot - 1),
],
None,
PartitionIndex::default(),
);
Expand All @@ -1800,12 +1810,22 @@ fn test_collect_rent_from_accounts() {
.accounts_db
.get_pubkey_hash_for_slot(later_slot)
.0;

// ensure account1 *is* stored because the account *did* change
// (its rent epoch must be updated to RENT_EXEMPT_RENT_EPOCH)
assert!(deltas.iter().map(|(pubkey, _)| pubkey).contains(&address1));

// if doing rewrites, ensure account2 *is* stored
// if skipping rewrites, ensure account2 is *not* stored
// (because the account did *not* change)
assert_eq!(
!deltas
.iter()
.any(|(pubkey, _)| pubkey == &zero_lamport_pubkey),
skip_rewrites
deltas.iter().map(|(pubkey, _)| pubkey).contains(&address2),
!skip_rewrites,
);

// ensure account3 *is* stored because the account *did* change
// (same as account1 above)
assert!(deltas.iter().map(|(pubkey, _)| pubkey).contains(&address3));
}
}

Expand Down Expand Up @@ -11044,39 +11064,6 @@ fn test_update_accounts_data_size() {
}
}

#[test]
fn test_skip_rewrite() {
solana_logger::setup();
let mut account = AccountSharedData::default();
let bank_slot = 10;
for account_rent_epoch in 0..3 {
account.set_rent_epoch(account_rent_epoch);
for rent_amount in [0, 1] {
for loaded_slot in (bank_slot - 1)..=bank_slot {
for old_rent_epoch in account_rent_epoch.saturating_sub(1)..=account_rent_epoch {
let skip = Bank::skip_rewrite(rent_amount, &account);
let mut should_skip = true;
if rent_amount != 0 || account_rent_epoch == 0 {
should_skip = false;
}
assert_eq!(
skip,
should_skip,
"{:?}",
(
account_rent_epoch,
old_rent_epoch,
rent_amount,
loaded_slot,
old_rent_epoch
)
);
}
}
}
}
}

#[derive(Serialize, Deserialize)]
enum MockReallocInstruction {
Realloc(usize, u64, Pubkey),
Expand Down Expand Up @@ -12266,6 +12253,11 @@ where
.transfer_and_confirm(mint_lamports, &mint_keypair, &alice_pubkey)
.unwrap();

// create and freeze a bank a few epochs in the future to trigger rent
// collection to visit (and rewrite) all accounts
let bank = new_from_parent_next_epoch(bank, &bank_forks, 2);
bank.freeze(); // trigger rent collection

// create zero-lamports account to be cleaned
let account = AccountSharedData::new(0, len1, &program);
let slot = bank.slot() + 1;
Expand Down Expand Up @@ -12313,9 +12305,9 @@ fn test_create_zero_lamport_with_clean() {
bank.squash();
bank.force_flush_accounts_cache();
// do clean and assert that it actually did its job
assert_eq!(4, bank.get_snapshot_storages(None).len());
assert_eq!(5, bank.get_snapshot_storages(None).len());
bank.clean_accounts();
assert_eq!(3, bank.get_snapshot_storages(None).len());
assert_eq!(4, bank.get_snapshot_storages(None).len());
});
}

Expand Down

0 comments on commit c1b465d

Please sign in to comment.