Allow feature builtins to overwrite existing builtins (#13403)
* Allow feature builtins to overwrite existing builtins * Add feature_builtin ActivationType * Correctly retain idempotent for replacing case * Fix test Co-authored-by: Ryo Onodera <ryoqun@gmail.com>
This commit is contained in:
parent
118ce47b97
commit
bc62313c66
|
@ -1,4 +1,7 @@
|
||||||
use solana_runtime::bank::{Builtin, Builtins};
|
use solana_runtime::{
|
||||||
|
bank::{Builtin, Builtins},
|
||||||
|
builtins::ActivationType,
|
||||||
|
};
|
||||||
use solana_sdk::{feature_set, genesis_config::ClusterType, pubkey::Pubkey};
|
use solana_sdk::{feature_set, genesis_config::ClusterType, pubkey::Pubkey};
|
||||||
|
|
||||||
/// Builtin programs that are always available
|
/// Builtin programs that are always available
|
||||||
|
@ -21,15 +24,16 @@ fn genesis_builtins(cluster_type: ClusterType) -> Vec<Builtin> {
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Builtin programs activated dynamically by feature
|
/// Builtin programs activated dynamically by feature
|
||||||
fn feature_builtins() -> Vec<(Builtin, Pubkey)> {
|
fn feature_builtins() -> Vec<(Builtin, Pubkey, ActivationType)> {
|
||||||
let builtins = vec![(
|
let builtins = vec![(
|
||||||
solana_bpf_loader_program!(),
|
solana_bpf_loader_program!(),
|
||||||
feature_set::bpf_loader2_program::id(),
|
feature_set::bpf_loader2_program::id(),
|
||||||
|
ActivationType::NewProgram,
|
||||||
)];
|
)];
|
||||||
|
|
||||||
builtins
|
builtins
|
||||||
.into_iter()
|
.into_iter()
|
||||||
.map(|(b, p)| (Builtin::new(&b.0, b.1, b.2), p))
|
.map(|(b, p, t)| (Builtin::new(&b.0, b.1, b.2), p, t))
|
||||||
.collect()
|
.collect()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -12,7 +12,7 @@ fn test_program_native_failure() {
|
||||||
let (genesis_config, alice_keypair) = create_genesis_config(50);
|
let (genesis_config, alice_keypair) = create_genesis_config(50);
|
||||||
let program_id = solana_sdk::pubkey::new_rand();
|
let program_id = solana_sdk::pubkey::new_rand();
|
||||||
let bank = Bank::new(&genesis_config);
|
let bank = Bank::new(&genesis_config);
|
||||||
bank.add_native_program("solana_failure_program", &program_id);
|
bank.add_native_program("solana_failure_program", &program_id, false);
|
||||||
|
|
||||||
// Call user program
|
// Call user program
|
||||||
let instruction = create_invoke_instruction(alice_keypair.pubkey(), program_id, &1u8);
|
let instruction = create_invoke_instruction(alice_keypair.pubkey(), program_id, &1u8);
|
||||||
|
|
|
@ -130,7 +130,7 @@ fn do_bench_transactions(
|
||||||
Pubkey::new(&BUILTIN_PROGRAM_ID),
|
Pubkey::new(&BUILTIN_PROGRAM_ID),
|
||||||
process_instruction,
|
process_instruction,
|
||||||
);
|
);
|
||||||
bank.add_native_program("solana_noop_program", &Pubkey::new(&NOOP_PROGRAM_ID));
|
bank.add_native_program("solana_noop_program", &Pubkey::new(&NOOP_PROGRAM_ID), false);
|
||||||
let bank = Arc::new(bank);
|
let bank = Arc::new(bank);
|
||||||
let bank_client = BankClient::new_shared(&bank);
|
let bank_client = BankClient::new_shared(&bank);
|
||||||
let transactions = create_transactions(&bank_client, &mint_keypair);
|
let transactions = create_transactions(&bank_client, &mint_keypair);
|
||||||
|
|
|
@ -10,7 +10,7 @@ use crate::{
|
||||||
accounts_db::{ErrorCounters, SnapshotStorages},
|
accounts_db::{ErrorCounters, SnapshotStorages},
|
||||||
accounts_index::Ancestors,
|
accounts_index::Ancestors,
|
||||||
blockhash_queue::BlockhashQueue,
|
blockhash_queue::BlockhashQueue,
|
||||||
builtins,
|
builtins::{self, ActivationType},
|
||||||
epoch_stakes::{EpochStakes, NodeVoteAccounts},
|
epoch_stakes::{EpochStakes, NodeVoteAccounts},
|
||||||
instruction_recorder::InstructionRecorder,
|
instruction_recorder::InstructionRecorder,
|
||||||
log_collector::LogCollector,
|
log_collector::LogCollector,
|
||||||
|
@ -215,7 +215,7 @@ pub struct Builtins {
|
||||||
pub genesis_builtins: Vec<Builtin>,
|
pub genesis_builtins: Vec<Builtin>,
|
||||||
|
|
||||||
/// Builtin programs activated dynamically by feature
|
/// Builtin programs activated dynamically by feature
|
||||||
pub feature_builtins: Vec<(Builtin, Pubkey)>,
|
pub feature_builtins: Vec<(Builtin, Pubkey, ActivationType)>,
|
||||||
}
|
}
|
||||||
|
|
||||||
const MAX_CACHED_EXECUTORS: usize = 100; // 10 MB assuming programs are around 100k
|
const MAX_CACHED_EXECUTORS: usize = 100; // 10 MB assuming programs are around 100k
|
||||||
|
@ -676,7 +676,7 @@ pub struct Bank {
|
||||||
bpf_compute_budget: Option<BpfComputeBudget>,
|
bpf_compute_budget: Option<BpfComputeBudget>,
|
||||||
|
|
||||||
/// Builtin programs activated dynamically by feature
|
/// Builtin programs activated dynamically by feature
|
||||||
feature_builtins: Arc<Vec<(Builtin, Pubkey)>>,
|
feature_builtins: Arc<Vec<(Builtin, Pubkey, ActivationType)>>,
|
||||||
|
|
||||||
/// Last time when the cluster info vote listener has synced with this bank
|
/// Last time when the cluster info vote listener has synced with this bank
|
||||||
pub last_vote_sync: AtomicU64,
|
pub last_vote_sync: AtomicU64,
|
||||||
|
@ -1715,16 +1715,15 @@ impl Bank {
|
||||||
|
|
||||||
// Add additional native programs specified in the genesis config
|
// Add additional native programs specified in the genesis config
|
||||||
for (name, program_id) in &genesis_config.native_instruction_processors {
|
for (name, program_id) in &genesis_config.native_instruction_processors {
|
||||||
self.add_native_program(name, program_id);
|
self.add_native_program(name, program_id, false);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn add_native_program(&self, name: &str, program_id: &Pubkey) {
|
// NOTE: must hold idempotent for the same set of arguments
|
||||||
let mut already_genuine_program_exists = false;
|
pub fn add_native_program(&self, name: &str, program_id: &Pubkey, must_replace: bool) {
|
||||||
if let Some(mut account) = self.get_account(&program_id) {
|
let mut existing_genuine_program = self.get_account(&program_id);
|
||||||
already_genuine_program_exists = native_loader::check_id(&account.owner);
|
if let Some(account) = &mut existing_genuine_program {
|
||||||
|
if !native_loader::check_id(&account.owner) {
|
||||||
if !already_genuine_program_exists {
|
|
||||||
// malicious account is pre-occupying at program_id
|
// malicious account is pre-occupying at program_id
|
||||||
// forcibly burn and purge it
|
// forcibly burn and purge it
|
||||||
|
|
||||||
|
@ -1737,19 +1736,54 @@ impl Bank {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if !already_genuine_program_exists {
|
if must_replace {
|
||||||
assert!(
|
// updating native program
|
||||||
!self.is_frozen(),
|
|
||||||
"Can't change frozen bank by adding not-existing new native program ({}, {}). \
|
|
||||||
Maybe, inconsistent program activation is detected on snapshot restore?",
|
|
||||||
name,
|
|
||||||
program_id
|
|
||||||
);
|
|
||||||
|
|
||||||
// Add a bogus executable native account, which will be loaded and ignored.
|
match existing_genuine_program {
|
||||||
let account = native_loader::create_loadable_account(name);
|
None => panic!(
|
||||||
self.store_account(&program_id, &account);
|
"There is no account to replace with native program ({}, {}).",
|
||||||
|
name, program_id
|
||||||
|
),
|
||||||
|
Some(account) => {
|
||||||
|
if *name == String::from_utf8_lossy(&account.data) {
|
||||||
|
// nop; it seems that already AccountsDB is updated.
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
// continue to replace account
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// introducing native program
|
||||||
|
|
||||||
|
match existing_genuine_program {
|
||||||
|
None => (), // continue to add account
|
||||||
|
Some(_account) => {
|
||||||
|
// nop; it seems that we already have account
|
||||||
|
|
||||||
|
// before returning here to retain idempotent just make sure
|
||||||
|
// the existing native program name is same with what we're
|
||||||
|
// supposed to add here (but skipping) But I can't:
|
||||||
|
// following assertion already catches several different names for same
|
||||||
|
// program_id
|
||||||
|
// depending on clusters...
|
||||||
|
// assert_eq!(name.to_owned(), String::from_utf8_lossy(&account.data));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
assert!(
|
||||||
|
!self.is_frozen(),
|
||||||
|
"Can't change frozen bank by adding not-existing new native program ({}, {}). \
|
||||||
|
Maybe, inconsistent program activation is detected on snapshot restore?",
|
||||||
|
name,
|
||||||
|
program_id
|
||||||
|
);
|
||||||
|
|
||||||
|
// Add a bogus executable native account, which will be loaded and ignored.
|
||||||
|
let account = native_loader::create_loadable_account(name);
|
||||||
|
self.store_account(&program_id, &account);
|
||||||
|
|
||||||
debug!("Added native program {} under {:?}", name, program_id);
|
debug!("Added native program {} under {:?}", name, program_id);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -3886,8 +3920,21 @@ impl Bank {
|
||||||
program_id: Pubkey,
|
program_id: Pubkey,
|
||||||
process_instruction_with_context: ProcessInstructionWithContext,
|
process_instruction_with_context: ProcessInstructionWithContext,
|
||||||
) {
|
) {
|
||||||
debug!("Added program {} under {:?}", name, program_id);
|
debug!("Adding program {} under {:?}", name, program_id);
|
||||||
self.add_native_program(name, &program_id);
|
self.add_native_program(name, &program_id, false);
|
||||||
|
self.message_processor
|
||||||
|
.add_program(program_id, process_instruction_with_context);
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Replace a builtin instruction processor if it already exists
|
||||||
|
pub fn replace_builtin(
|
||||||
|
&mut self,
|
||||||
|
name: &str,
|
||||||
|
program_id: Pubkey,
|
||||||
|
process_instruction_with_context: ProcessInstructionWithContext,
|
||||||
|
) {
|
||||||
|
debug!("Replacing program {} under {:?}", name, program_id);
|
||||||
|
self.add_native_program(name, &program_id, true);
|
||||||
self.message_processor
|
self.message_processor
|
||||||
.add_program(program_id, process_instruction_with_context);
|
.add_program(program_id, process_instruction_with_context);
|
||||||
}
|
}
|
||||||
|
@ -4021,15 +4068,22 @@ impl Bank {
|
||||||
new_feature_activations: &HashSet<Pubkey>,
|
new_feature_activations: &HashSet<Pubkey>,
|
||||||
) {
|
) {
|
||||||
let feature_builtins = self.feature_builtins.clone();
|
let feature_builtins = self.feature_builtins.clone();
|
||||||
for (builtin, feature) in feature_builtins.iter() {
|
for (builtin, feature, activation_type) in feature_builtins.iter() {
|
||||||
let should_populate = init_or_warp && self.feature_set.is_active(&feature)
|
let should_populate = init_or_warp && self.feature_set.is_active(&feature)
|
||||||
|| !init_or_warp && new_feature_activations.contains(&feature);
|
|| !init_or_warp && new_feature_activations.contains(&feature);
|
||||||
if should_populate {
|
if should_populate {
|
||||||
self.add_builtin(
|
match activation_type {
|
||||||
&builtin.name,
|
ActivationType::NewProgram => self.add_builtin(
|
||||||
builtin.id,
|
&builtin.name,
|
||||||
builtin.process_instruction_with_context,
|
builtin.id,
|
||||||
);
|
builtin.process_instruction_with_context,
|
||||||
|
),
|
||||||
|
ActivationType::NewVersion => self.replace_builtin(
|
||||||
|
&builtin.name,
|
||||||
|
builtin.id,
|
||||||
|
builtin.process_instruction_with_context,
|
||||||
|
),
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -9232,6 +9286,16 @@ mod tests {
|
||||||
.unwrap()
|
.unwrap()
|
||||||
.add_builtin("mock_program", program_id, mock_ix_processor);
|
.add_builtin("mock_program", program_id, mock_ix_processor);
|
||||||
assert_eq!(bank.get_account_modified_slot(&program_id).unwrap().1, slot);
|
assert_eq!(bank.get_account_modified_slot(&program_id).unwrap().1, slot);
|
||||||
|
|
||||||
|
Arc::get_mut(&mut bank).unwrap().replace_builtin(
|
||||||
|
"mock_program v2",
|
||||||
|
program_id,
|
||||||
|
mock_ix_processor,
|
||||||
|
);
|
||||||
|
assert_eq!(
|
||||||
|
bank.get_account_modified_slot(&program_id).unwrap().1,
|
||||||
|
bank.slot()
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
|
@ -9285,14 +9349,35 @@ mod tests {
|
||||||
|
|
||||||
Arc::get_mut(&mut bank)
|
Arc::get_mut(&mut bank)
|
||||||
.unwrap()
|
.unwrap()
|
||||||
.add_native_program("mock_program", &program_id);
|
.add_native_program("mock_program", &program_id, false);
|
||||||
assert_eq!(bank.get_account_modified_slot(&program_id).unwrap().1, slot);
|
assert_eq!(bank.get_account_modified_slot(&program_id).unwrap().1, slot);
|
||||||
|
|
||||||
let mut bank = Arc::new(new_from_parent(&bank));
|
let mut bank = Arc::new(new_from_parent(&bank));
|
||||||
Arc::get_mut(&mut bank)
|
Arc::get_mut(&mut bank)
|
||||||
.unwrap()
|
.unwrap()
|
||||||
.add_native_program("mock_program", &program_id);
|
.add_native_program("mock_program", &program_id, false);
|
||||||
assert_eq!(bank.get_account_modified_slot(&program_id).unwrap().1, slot);
|
assert_eq!(bank.get_account_modified_slot(&program_id).unwrap().1, slot);
|
||||||
|
|
||||||
|
let mut bank = Arc::new(new_from_parent(&bank));
|
||||||
|
// When replacing native_program, name must change to disambiguate from repeated
|
||||||
|
// invocations.
|
||||||
|
Arc::get_mut(&mut bank)
|
||||||
|
.unwrap()
|
||||||
|
.add_native_program("mock_program v2", &program_id, true);
|
||||||
|
assert_eq!(
|
||||||
|
bank.get_account_modified_slot(&program_id).unwrap().1,
|
||||||
|
bank.slot()
|
||||||
|
);
|
||||||
|
|
||||||
|
let mut bank = Arc::new(new_from_parent(&bank));
|
||||||
|
Arc::get_mut(&mut bank)
|
||||||
|
.unwrap()
|
||||||
|
.add_native_program("mock_program v2", &program_id, true);
|
||||||
|
// replacing with same name shouldn't update account
|
||||||
|
assert_eq!(
|
||||||
|
bank.get_account_modified_slot(&program_id).unwrap().1,
|
||||||
|
bank.parent_slot()
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
|
@ -9308,16 +9393,35 @@ mod tests {
|
||||||
let slot = 123;
|
let slot = 123;
|
||||||
let program_id = Pubkey::from_str("CiXgo2KHKSDmDnV1F6B69eWFgNAPiSBjjYvfB4cvRNre").unwrap();
|
let program_id = Pubkey::from_str("CiXgo2KHKSDmDnV1F6B69eWFgNAPiSBjjYvfB4cvRNre").unwrap();
|
||||||
|
|
||||||
let mut bank = Arc::new(Bank::new_from_parent(
|
let bank = Bank::new_from_parent(
|
||||||
&Arc::new(Bank::new(&genesis_config)),
|
&Arc::new(Bank::new(&genesis_config)),
|
||||||
&Pubkey::default(),
|
&Pubkey::default(),
|
||||||
slot,
|
slot,
|
||||||
));
|
);
|
||||||
bank.freeze();
|
bank.freeze();
|
||||||
|
|
||||||
Arc::get_mut(&mut bank)
|
bank.add_native_program("mock_program", &program_id, false);
|
||||||
.unwrap()
|
}
|
||||||
.add_native_program("mock_program", &program_id);
|
|
||||||
|
#[test]
|
||||||
|
#[should_panic(
|
||||||
|
expected = "There is no account to replace with native program (mock_program, \
|
||||||
|
CiXgo2KHKSDmDnV1F6B69eWFgNAPiSBjjYvfB4cvRNre)."
|
||||||
|
)]
|
||||||
|
fn test_add_native_program_replace_none() {
|
||||||
|
use std::str::FromStr;
|
||||||
|
let (genesis_config, _mint_keypair) = create_genesis_config(100_000);
|
||||||
|
|
||||||
|
let slot = 123;
|
||||||
|
let program_id = Pubkey::from_str("CiXgo2KHKSDmDnV1F6B69eWFgNAPiSBjjYvfB4cvRNre").unwrap();
|
||||||
|
|
||||||
|
let bank = Bank::new_from_parent(
|
||||||
|
&Arc::new(Bank::new(&genesis_config)),
|
||||||
|
&Pubkey::default(),
|
||||||
|
slot,
|
||||||
|
);
|
||||||
|
|
||||||
|
bank.add_native_program("mock_program", &program_id, true);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
|
|
|
@ -30,8 +30,14 @@ fn genesis_builtins() -> Vec<Builtin> {
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[derive(AbiExample, Debug, Clone)]
|
||||||
|
pub enum ActivationType {
|
||||||
|
NewProgram,
|
||||||
|
NewVersion,
|
||||||
|
}
|
||||||
|
|
||||||
/// Builtin programs activated dynamically by feature
|
/// Builtin programs activated dynamically by feature
|
||||||
fn feature_builtins() -> Vec<(Builtin, Pubkey)> {
|
fn feature_builtins() -> Vec<(Builtin, Pubkey, ActivationType)> {
|
||||||
vec![(
|
vec![(
|
||||||
Builtin::new(
|
Builtin::new(
|
||||||
"secp256k1_program",
|
"secp256k1_program",
|
||||||
|
@ -39,6 +45,7 @@ fn feature_builtins() -> Vec<(Builtin, Pubkey)> {
|
||||||
solana_secp256k1_program::process_instruction,
|
solana_secp256k1_program::process_instruction,
|
||||||
),
|
),
|
||||||
feature_set::secp256k1_program_enabled::id(),
|
feature_set::secp256k1_program_enabled::id(),
|
||||||
|
ActivationType::NewProgram,
|
||||||
)]
|
)]
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -10,7 +10,7 @@ fn test_program_native_noop() {
|
||||||
let (genesis_config, alice_keypair) = create_genesis_config(50);
|
let (genesis_config, alice_keypair) = create_genesis_config(50);
|
||||||
let program_id = solana_sdk::pubkey::new_rand();
|
let program_id = solana_sdk::pubkey::new_rand();
|
||||||
let bank = Bank::new(&genesis_config);
|
let bank = Bank::new(&genesis_config);
|
||||||
bank.add_native_program("solana_noop_program", &program_id);
|
bank.add_native_program("solana_noop_program", &program_id, false);
|
||||||
|
|
||||||
// Call user program
|
// Call user program
|
||||||
let instruction = create_invoke_instruction(alice_keypair.pubkey(), program_id, &1u8);
|
let instruction = create_invoke_instruction(alice_keypair.pubkey(), program_id, &1u8);
|
||||||
|
|
Loading…
Reference in New Issue