This repository has been archived by the owner on Jan 22, 2025. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 4.5k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
v1.17: Bank: Add function to replace empty account with upgradeable p…
…rogram on feature activation (backport of #32783) (#33527) Bank: Add function to replace empty account with upgradeable program on feature activation (#32783) * replace program account * modify for all cases * remove non-data swap * address tests & conditional feedback * get the rent involved * mix in owner & executable * feature-related cases * stripped back to feature-specific case only * added feature * address initial feedback * added more lamport checks * condense tests * using test_case * add fail cases to tests * more cleanup * add verifiably built program * update program account state * cleaned up serializing logic * use full word capitalization * rename old & new to dst & src * swap src and dst in parameters * add warnings and errors * rename feature to programify * test suite description clarity * remove strings from datapoints * spell out source and destination * more verbose comments in account replace functions * move lamport calculation * swap lamport check for state check * move replace functions to helper module * make replace_account methods fallible * refactor error handling * add test for source program state (cherry picked from commit 25460f7) Co-authored-by: Joe C <jcaulfield135@gmail.com>
- Loading branch information
1 parent
2bfe99c
commit 9318394
Showing
6 changed files
with
613 additions
and
62 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,191 @@ | ||
use { | ||
super::Bank, | ||
log::*, | ||
solana_accounts_db::accounts_index::ZeroLamport, | ||
solana_sdk::{ | ||
account::{Account, AccountSharedData, ReadableAccount}, | ||
bpf_loader_upgradeable::{self, UpgradeableLoaderState}, | ||
pubkey::Pubkey, | ||
}, | ||
std::sync::atomic::Ordering::Relaxed, | ||
thiserror::Error, | ||
}; | ||
|
||
/// Errors returned by `replace_account` methods | ||
#[derive(Debug, Error)] | ||
pub enum ReplaceAccountError { | ||
/// Account not found | ||
#[error("Account not found: {0:?}")] | ||
AccountNotFound(Pubkey), | ||
/// Account exists | ||
#[error("Account exists: {0:?}")] | ||
AccountExists(Pubkey), | ||
#[error("Bincode Error: {0}")] | ||
BincodeError(#[from] bincode::Error), | ||
/// Not an upgradeable program | ||
#[error("Not an upgradeable program")] | ||
NotAnUpgradeableProgram, | ||
} | ||
|
||
/// Moves one account in place of another | ||
/// `source`: the account to replace with | ||
/// `destination`: the account to be replaced | ||
fn move_account<U, V>( | ||
bank: &Bank, | ||
source_address: &Pubkey, | ||
source_account: &V, | ||
destination_address: &Pubkey, | ||
destination_account: Option<&U>, | ||
) where | ||
U: ReadableAccount + Sync + ZeroLamport, | ||
V: ReadableAccount + Sync + ZeroLamport, | ||
{ | ||
let (destination_lamports, destination_len) = match destination_account { | ||
Some(destination_account) => ( | ||
destination_account.lamports(), | ||
destination_account.data().len(), | ||
), | ||
None => (0, 0), | ||
}; | ||
|
||
// Burn lamports in the destination account | ||
bank.capitalization.fetch_sub(destination_lamports, Relaxed); | ||
|
||
// Transfer source account to destination account | ||
bank.store_account(destination_address, source_account); | ||
|
||
// Clear source account | ||
bank.store_account(source_address, &AccountSharedData::default()); | ||
|
||
bank.calculate_and_update_accounts_data_size_delta_off_chain( | ||
destination_len, | ||
source_account.data().len(), | ||
); | ||
} | ||
|
||
/// Use to replace non-upgradeable programs by feature activation | ||
/// `source`: the non-upgradeable program account to replace with | ||
/// `destination`: the non-upgradeable program account to be replaced | ||
#[allow(dead_code)] | ||
pub(crate) fn replace_non_upgradeable_program_account( | ||
bank: &Bank, | ||
source_address: &Pubkey, | ||
destination_address: &Pubkey, | ||
datapoint_name: &'static str, | ||
) -> Result<(), ReplaceAccountError> { | ||
let destination_account = bank | ||
.get_account_with_fixed_root(destination_address) | ||
.ok_or(ReplaceAccountError::AccountNotFound(*destination_address))?; | ||
let source_account = bank | ||
.get_account_with_fixed_root(source_address) | ||
.ok_or(ReplaceAccountError::AccountNotFound(*source_address))?; | ||
|
||
datapoint_info!(datapoint_name, ("slot", bank.slot, i64)); | ||
|
||
move_account( | ||
bank, | ||
source_address, | ||
&source_account, | ||
destination_address, | ||
Some(&destination_account), | ||
); | ||
|
||
// Unload a program from the bank's cache | ||
bank.loaded_programs_cache | ||
.write() | ||
.unwrap() | ||
.remove_programs([*destination_address].into_iter()); | ||
|
||
Ok(()) | ||
} | ||
|
||
/// Use to replace an empty account with a program by feature activation | ||
/// Note: The upgradeable program should have both: | ||
/// - Program account | ||
/// - Program data account | ||
/// `source`: the upgradeable program account to replace with | ||
/// `destination`: the empty account to be replaced | ||
pub(crate) fn replace_empty_account_with_upgradeable_program( | ||
bank: &Bank, | ||
source_address: &Pubkey, | ||
destination_address: &Pubkey, | ||
datapoint_name: &'static str, | ||
) -> Result<(), ReplaceAccountError> { | ||
// Must be attempting to replace an empty account with a program | ||
// account _and_ data account | ||
let source_account = bank | ||
.get_account_with_fixed_root(source_address) | ||
.ok_or(ReplaceAccountError::AccountNotFound(*source_address))?; | ||
|
||
let (destination_data_address, _) = Pubkey::find_program_address( | ||
&[destination_address.as_ref()], | ||
&bpf_loader_upgradeable::id(), | ||
); | ||
let (source_data_address, _) = | ||
Pubkey::find_program_address(&[source_address.as_ref()], &bpf_loader_upgradeable::id()); | ||
|
||
// Make sure the data within the source account is the PDA of its | ||
// data account. This also means it has at least the necessary | ||
// lamports for rent. | ||
let source_state = bincode::deserialize::<UpgradeableLoaderState>(source_account.data())?; | ||
if !matches!(source_state, UpgradeableLoaderState::Program { .. }) { | ||
return Err(ReplaceAccountError::NotAnUpgradeableProgram); | ||
} | ||
|
||
let source_data_account = bank | ||
.get_account_with_fixed_root(&source_data_address) | ||
.ok_or(ReplaceAccountError::AccountNotFound(source_data_address))?; | ||
|
||
// Make sure the destination account is empty | ||
// We aren't going to check that there isn't a data account at | ||
// the known program-derived address (ie. `destination_data_address`), | ||
// because if it exists, it will be overwritten | ||
if bank | ||
.get_account_with_fixed_root(destination_address) | ||
.is_some() | ||
{ | ||
return Err(ReplaceAccountError::AccountExists(*destination_address)); | ||
} | ||
let state = UpgradeableLoaderState::Program { | ||
programdata_address: destination_data_address, | ||
}; | ||
let data = bincode::serialize(&state)?; | ||
let lamports = bank.get_minimum_balance_for_rent_exemption(data.len()); | ||
let created_program_account = Account { | ||
lamports, | ||
data, | ||
owner: bpf_loader_upgradeable::id(), | ||
executable: true, | ||
rent_epoch: source_account.rent_epoch(), | ||
}; | ||
|
||
datapoint_info!(datapoint_name, ("slot", bank.slot, i64)); | ||
let change_in_capitalization = source_account.lamports().saturating_sub(lamports); | ||
|
||
// Replace the destination data account with the source one | ||
// If the destination data account does not exist, it will be created | ||
// If it does exist, it will be overwritten | ||
move_account( | ||
bank, | ||
&source_data_address, | ||
&source_data_account, | ||
&destination_data_address, | ||
bank.get_account_with_fixed_root(&destination_data_address) | ||
.as_ref(), | ||
); | ||
|
||
// Write the source data account's PDA into the destination program account | ||
move_account( | ||
bank, | ||
source_address, | ||
&created_program_account, | ||
destination_address, | ||
None::<&AccountSharedData>, | ||
); | ||
|
||
// Any remaining lamports in the source program account are burnt | ||
bank.capitalization | ||
.fetch_sub(change_in_capitalization, Relaxed); | ||
|
||
Ok(()) | ||
} |
Oops, something went wrong.