#[cfg(feature = "metrics")]
use {crate::loaded_programs::LoadProgramMetrics, solana_svm_measure::measure::Measure};
use {
crate::{
invoke_context::InvokeContext,
loaded_programs::{
DELAY_VISIBILITY_SLOT_OFFSET, ProgramCacheEntry, ProgramCacheForTxBatch,
ProgramRuntimeEnvironment,
},
},
solana_clock::Slot,
solana_instruction::error::InstructionError,
solana_pubkey::Pubkey,
solana_sbpf::{
elf::{ElfError, Executable},
program::BuiltinProgram,
verifier::RequisiteVerifier,
},
solana_svm_log_collector::{LogCollector, ic_logger_msg},
solana_svm_type_overrides::sync::{Arc, atomic::Ordering},
std::{cell::RefCell, rc::Rc},
};
fn morph_into_deployment_environment_v1<'a>(
from: Arc<BuiltinProgram<InvokeContext<'a, 'a>>>,
) -> Result<BuiltinProgram<InvokeContext<'a, 'a>>, ElfError> {
let mut config = from.get_config().clone();
config.reject_broken_elfs = true;
let mut result = BuiltinProgram::new_loader(config);
for (_key, (name, value)) in from.get_function_registry().iter() {
if name != *b"sol_alloc_free_" {
result.register_function(unsafe { std::str::from_utf8_unchecked(name) }, value)?;
}
}
Ok(result)
}
#[allow(clippy::too_many_arguments)]
pub fn deploy_program(
log_collector: Option<Rc<RefCell<LogCollector>>>,
#[cfg(feature = "metrics")] load_program_metrics: &mut LoadProgramMetrics,
program_cache_for_tx_batch: &mut ProgramCacheForTxBatch,
program_runtime_environment: ProgramRuntimeEnvironment,
program_id: &Pubkey,
loader_key: &Pubkey,
account_size: usize,
programdata: &[u8],
deployment_slot: Slot,
) -> Result<(), InstructionError> {
#[cfg(feature = "metrics")]
let mut register_syscalls_time = Measure::start("register_syscalls_time");
let deployment_program_runtime_environment =
morph_into_deployment_environment_v1(program_runtime_environment.clone()).map_err(|e| {
ic_logger_msg!(log_collector, "Failed to register syscalls: {}", e);
InstructionError::ProgramEnvironmentSetupFailure
})?;
#[cfg(feature = "metrics")]
{
register_syscalls_time.stop();
load_program_metrics.register_syscalls_us = register_syscalls_time.as_us();
}
#[cfg(feature = "metrics")]
let mut load_elf_time = Measure::start("load_elf_time");
let executable = Executable::<InvokeContext>::load(
programdata,
Arc::new(deployment_program_runtime_environment),
)
.map_err(|err| {
ic_logger_msg!(log_collector, "{}", err);
InstructionError::InvalidAccountData
})?;
#[cfg(feature = "metrics")]
{
load_elf_time.stop();
load_program_metrics.load_elf_us = load_elf_time.as_us();
}
#[cfg(feature = "metrics")]
let mut verify_code_time = Measure::start("verify_code_time");
executable.verify::<RequisiteVerifier>().map_err(|err| {
ic_logger_msg!(log_collector, "{}", err);
InstructionError::InvalidAccountData
})?;
#[cfg(feature = "metrics")]
{
verify_code_time.stop();
load_program_metrics.verify_code_us = verify_code_time.as_us();
}
let executor = unsafe {
ProgramCacheEntry::reload(
loader_key,
program_runtime_environment,
deployment_slot,
deployment_slot.saturating_add(DELAY_VISIBILITY_SLOT_OFFSET),
programdata,
account_size,
#[cfg(feature = "metrics")]
load_program_metrics,
)
}
.map_err(|err| {
ic_logger_msg!(log_collector, "{}", err);
InstructionError::InvalidAccountData
})?;
if let Some(old_entry) = program_cache_for_tx_batch.find(program_id) {
executor.tx_usage_counter.store(
old_entry.tx_usage_counter.load(Ordering::Relaxed),
Ordering::Relaxed,
);
}
#[cfg(feature = "metrics")]
{
load_program_metrics.program_id = program_id.to_string();
}
program_cache_for_tx_batch.store_modified_entry(*program_id, Arc::new(executor));
Ok(())
}
#[macro_export]
macro_rules! deploy_program {
($invoke_context:expr, $program_id:expr, $loader_key:expr, $account_size:expr, $programdata:expr, $deployment_slot:expr $(,)?) => {
assert_eq!(
$deployment_slot,
$invoke_context.program_cache_for_tx_batch.slot()
);
#[cfg(feature = "metrics")]
let mut load_program_metrics = $crate::loaded_programs::LoadProgramMetrics::default();
$crate::deploy::deploy_program(
$invoke_context.get_log_collector(),
#[cfg(feature = "metrics")]
&mut load_program_metrics,
$invoke_context.program_cache_for_tx_batch,
$invoke_context
.get_program_runtime_environments_for_deployment()
.program_runtime_v1
.clone(),
$program_id,
$loader_key,
$account_size,
$programdata,
$deployment_slot,
)?;
#[cfg(feature = "metrics")]
load_program_metrics.submit_datapoint(&mut $invoke_context.timings);
};
}