Refactor - create_executor() to be usable outside of bpf_loader (#28369)
* Moves disable_deploy_of_alloc_free_syscall parameter inside create_executor(). * Lets register_syscalls() take &FeatureSet only instead of the entire InvokeContext. * Uses ic_logger_msg!() instead of ic_msg!() inside create_executor(). Inlines map_ebpf_error(). * Adds register_syscalls_us to executor_metrics::CreateMetrics. * Moves timings accumulation into executor_metrics::CreateMetrics::submit_datapoint(). * Moves &invoke_context.feature_set into a variable. * Lets create_executor() return executor_metrics::CreateMetrics via a mutable parameter. * Dissolves invoke_context parameter in create_executor(). * Pulls assignment of create_executor_metrics.program_id outside of create_executor(). * Makes create_executor() take a byte slice instead of a BorrowedAccount. * Adds create_executor_from_account().
This commit is contained in:
parent
d949f4f42f
commit
547f07526b
|
@ -2083,7 +2083,7 @@ fn read_and_verify_elf(program_location: &str) -> Result<Vec<u8>, Box<dyn std::e
|
||||||
file.read_to_end(&mut program_data)
|
file.read_to_end(&mut program_data)
|
||||||
.map_err(|err| format!("Unable to read program file: {}", err))?;
|
.map_err(|err| format!("Unable to read program file: {}", err))?;
|
||||||
let mut transaction_context = TransactionContext::new(Vec::new(), Some(Rent::default()), 1, 1);
|
let mut transaction_context = TransactionContext::new(Vec::new(), Some(Rent::default()), 1, 1);
|
||||||
let mut invoke_context = InvokeContext::new_mock(&mut transaction_context, &[]);
|
let invoke_context = InvokeContext::new_mock(&mut transaction_context, &[]);
|
||||||
|
|
||||||
// Verify the program
|
// Verify the program
|
||||||
let executable = Executable::<ThisInstructionMeter>::from_elf(
|
let executable = Executable::<ThisInstructionMeter>::from_elf(
|
||||||
|
@ -2092,7 +2092,7 @@ fn read_and_verify_elf(program_location: &str) -> Result<Vec<u8>, Box<dyn std::e
|
||||||
reject_broken_elfs: true,
|
reject_broken_elfs: true,
|
||||||
..Config::default()
|
..Config::default()
|
||||||
},
|
},
|
||||||
register_syscalls(&mut invoke_context, true).unwrap(),
|
register_syscalls(&invoke_context.feature_set, true).unwrap(),
|
||||||
)
|
)
|
||||||
.map_err(|err| format!("ELF error: {}", err))?;
|
.map_err(|err| format!("ELF error: {}", err))?;
|
||||||
|
|
||||||
|
|
|
@ -21,12 +21,14 @@ use {
|
||||||
log::{log_enabled, trace, Level::Trace},
|
log::{log_enabled, trace, Level::Trace},
|
||||||
solana_measure::measure::Measure,
|
solana_measure::measure::Measure,
|
||||||
solana_program_runtime::{
|
solana_program_runtime::{
|
||||||
executor_cache::Executor,
|
compute_budget::ComputeBudget,
|
||||||
|
executor_cache::{Executor, TransactionExecutorCache},
|
||||||
ic_logger_msg, ic_msg,
|
ic_logger_msg, ic_msg,
|
||||||
invoke_context::{ComputeMeter, InvokeContext},
|
invoke_context::{ComputeMeter, InvokeContext},
|
||||||
log_collector::LogCollector,
|
log_collector::LogCollector,
|
||||||
stable_log,
|
stable_log,
|
||||||
sysvar_cache::get_sysvar_with_account_check,
|
sysvar_cache::get_sysvar_with_account_check,
|
||||||
|
timings::ExecuteDetailsTimings,
|
||||||
},
|
},
|
||||||
solana_rbpf::{
|
solana_rbpf::{
|
||||||
aligned_memory::AlignedMemory,
|
aligned_memory::AlignedMemory,
|
||||||
|
@ -47,7 +49,7 @@ use {
|
||||||
check_slice_translation_size, disable_deploy_of_alloc_free_syscall,
|
check_slice_translation_size, disable_deploy_of_alloc_free_syscall,
|
||||||
disable_deprecated_loader, enable_bpf_loader_extend_program_ix,
|
disable_deprecated_loader, enable_bpf_loader_extend_program_ix,
|
||||||
error_on_syscall_bpf_function_hash_collisions, limit_max_instruction_trace_length,
|
error_on_syscall_bpf_function_hash_collisions, limit_max_instruction_trace_length,
|
||||||
reject_callx_r10,
|
reject_callx_r10, FeatureSet,
|
||||||
},
|
},
|
||||||
instruction::{AccountMeta, InstructionError},
|
instruction::{AccountMeta, InstructionError},
|
||||||
loader_instruction::LoaderInstruction,
|
loader_instruction::LoaderInstruction,
|
||||||
|
@ -63,7 +65,12 @@ use {
|
||||||
BorrowedAccount, IndexOfAccount, InstructionContext, TransactionContext,
|
BorrowedAccount, IndexOfAccount, InstructionContext, TransactionContext,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
std::{cell::RefCell, fmt::Debug, rc::Rc, sync::Arc},
|
std::{
|
||||||
|
cell::{RefCell, RefMut},
|
||||||
|
fmt::Debug,
|
||||||
|
rc::Rc,
|
||||||
|
sync::Arc,
|
||||||
|
},
|
||||||
thiserror::Error,
|
thiserror::Error,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -83,25 +90,31 @@ pub enum BpfError {
|
||||||
}
|
}
|
||||||
impl UserDefinedError for BpfError {}
|
impl UserDefinedError for BpfError {}
|
||||||
|
|
||||||
fn map_ebpf_error(invoke_context: &InvokeContext, e: EbpfError) -> InstructionError {
|
|
||||||
ic_msg!(invoke_context, "{}", e);
|
|
||||||
InstructionError::InvalidAccountData
|
|
||||||
}
|
|
||||||
|
|
||||||
mod executor_metrics {
|
mod executor_metrics {
|
||||||
|
use super::*;
|
||||||
|
|
||||||
#[derive(Debug, Default)]
|
#[derive(Debug, Default)]
|
||||||
pub struct CreateMetrics {
|
pub struct CreateMetrics {
|
||||||
pub program_id: String,
|
pub program_id: String,
|
||||||
|
pub register_syscalls_us: u64,
|
||||||
pub load_elf_us: u64,
|
pub load_elf_us: u64,
|
||||||
pub verify_code_us: u64,
|
pub verify_code_us: u64,
|
||||||
pub jit_compile_us: u64,
|
pub jit_compile_us: u64,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl CreateMetrics {
|
impl CreateMetrics {
|
||||||
pub fn submit_datapoint(&self) {
|
pub fn submit_datapoint(&self, timings: &mut ExecuteDetailsTimings) {
|
||||||
|
saturating_add_assign!(
|
||||||
|
timings.create_executor_register_syscalls_us,
|
||||||
|
self.register_syscalls_us
|
||||||
|
);
|
||||||
|
saturating_add_assign!(timings.create_executor_load_elf_us, self.load_elf_us);
|
||||||
|
saturating_add_assign!(timings.create_executor_verify_code_us, self.verify_code_us);
|
||||||
|
saturating_add_assign!(timings.create_executor_jit_compile_us, self.jit_compile_us);
|
||||||
datapoint_trace!(
|
datapoint_trace!(
|
||||||
"create_executor_trace",
|
"create_executor_trace",
|
||||||
("program_id", self.program_id, String),
|
("program_id", self.program_id, String),
|
||||||
|
("register_syscalls_us", self.register_syscalls_us, i64),
|
||||||
("load_elf_us", self.load_elf_us, i64),
|
("load_elf_us", self.load_elf_us, i64),
|
||||||
("verify_code_us", self.verify_code_us, i64),
|
("verify_code_us", self.verify_code_us, i64),
|
||||||
("jit_compile_us", self.jit_compile_us, i64),
|
("jit_compile_us", self.jit_compile_us, i64),
|
||||||
|
@ -144,27 +157,26 @@ fn try_borrow_account<'a>(
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn create_executor(
|
fn create_executor_from_bytes(
|
||||||
programdata_account_index: IndexOfAccount,
|
feature_set: &FeatureSet,
|
||||||
programdata_offset: usize,
|
compute_budget: &ComputeBudget,
|
||||||
invoke_context: &mut InvokeContext,
|
log_collector: Option<Rc<RefCell<LogCollector>>>,
|
||||||
|
create_executor_metrics: &mut executor_metrics::CreateMetrics,
|
||||||
|
programdata: &[u8],
|
||||||
use_jit: bool,
|
use_jit: bool,
|
||||||
reject_deployment_of_broken_elfs: bool,
|
reject_deployment_of_broken_elfs: bool,
|
||||||
disable_deploy_of_alloc_free_syscall: bool,
|
|
||||||
) -> Result<Arc<BpfExecutor>, InstructionError> {
|
) -> Result<Arc<BpfExecutor>, InstructionError> {
|
||||||
let mut register_syscalls_time = Measure::start("register_syscalls_time");
|
let mut register_syscalls_time = Measure::start("register_syscalls_time");
|
||||||
|
let disable_deploy_of_alloc_free_syscall = reject_deployment_of_broken_elfs
|
||||||
|
&& feature_set.is_active(&disable_deploy_of_alloc_free_syscall::id());
|
||||||
let register_syscall_result =
|
let register_syscall_result =
|
||||||
syscalls::register_syscalls(invoke_context, disable_deploy_of_alloc_free_syscall);
|
syscalls::register_syscalls(feature_set, disable_deploy_of_alloc_free_syscall);
|
||||||
register_syscalls_time.stop();
|
register_syscalls_time.stop();
|
||||||
invoke_context.timings.create_executor_register_syscalls_us = invoke_context
|
create_executor_metrics.register_syscalls_us = register_syscalls_time.as_us();
|
||||||
.timings
|
|
||||||
.create_executor_register_syscalls_us
|
|
||||||
.saturating_add(register_syscalls_time.as_us());
|
|
||||||
let syscall_registry = register_syscall_result.map_err(|e| {
|
let syscall_registry = register_syscall_result.map_err(|e| {
|
||||||
ic_msg!(invoke_context, "Failed to register syscalls: {}", e);
|
ic_logger_msg!(log_collector, "Failed to register syscalls: {}", e);
|
||||||
InstructionError::ProgramEnvironmentSetupFailure
|
InstructionError::ProgramEnvironmentSetupFailure
|
||||||
})?;
|
})?;
|
||||||
let compute_budget = invoke_context.get_compute_budget();
|
|
||||||
let config = Config {
|
let config = Config {
|
||||||
max_call_depth: compute_budget.max_call_depth,
|
max_call_depth: compute_budget.max_call_depth,
|
||||||
stack_frame_size: compute_budget.stack_frame_size,
|
stack_frame_size: compute_budget.stack_frame_size,
|
||||||
|
@ -177,12 +189,9 @@ pub fn create_executor(
|
||||||
noop_instruction_rate: 256,
|
noop_instruction_rate: 256,
|
||||||
sanitize_user_provided_values: true,
|
sanitize_user_provided_values: true,
|
||||||
encrypt_environment_registers: true,
|
encrypt_environment_registers: true,
|
||||||
syscall_bpf_function_hash_collision: invoke_context
|
syscall_bpf_function_hash_collision: feature_set
|
||||||
.feature_set
|
|
||||||
.is_active(&error_on_syscall_bpf_function_hash_collisions::id()),
|
.is_active(&error_on_syscall_bpf_function_hash_collisions::id()),
|
||||||
reject_callx_r10: invoke_context
|
reject_callx_r10: feature_set.is_active(&reject_callx_r10::id()),
|
||||||
.feature_set
|
|
||||||
.is_active(&reject_callx_r10::id()),
|
|
||||||
dynamic_stack_frames: false,
|
dynamic_stack_frames: false,
|
||||||
enable_sdiv: false,
|
enable_sdiv: false,
|
||||||
optimize_rodata: false,
|
optimize_rodata: false,
|
||||||
|
@ -193,65 +202,117 @@ pub fn create_executor(
|
||||||
aligned_memory_mapping: true,
|
aligned_memory_mapping: true,
|
||||||
// Warning, do not use `Config::default()` so that configuration here is explicit.
|
// Warning, do not use `Config::default()` so that configuration here is explicit.
|
||||||
};
|
};
|
||||||
let mut create_executor_metrics = executor_metrics::CreateMetrics::default();
|
|
||||||
let executable = {
|
|
||||||
let transaction_context = &invoke_context.transaction_context;
|
|
||||||
let instruction_context = transaction_context.get_current_instruction_context()?;
|
|
||||||
let programdata = try_borrow_account(
|
|
||||||
transaction_context,
|
|
||||||
instruction_context,
|
|
||||||
programdata_account_index,
|
|
||||||
)?;
|
|
||||||
create_executor_metrics.program_id = programdata.get_key().to_string();
|
|
||||||
let mut load_elf_time = Measure::start("load_elf_time");
|
let mut load_elf_time = Measure::start("load_elf_time");
|
||||||
let executable = Executable::<ThisInstructionMeter>::from_elf(
|
let executable =
|
||||||
programdata
|
Executable::<ThisInstructionMeter>::from_elf(programdata, config, syscall_registry)
|
||||||
.get_data()
|
.map_err(|err| {
|
||||||
.get(programdata_offset..)
|
ic_logger_msg!(log_collector, "{}", err);
|
||||||
.ok_or(InstructionError::AccountDataTooSmall)?,
|
InstructionError::InvalidAccountData
|
||||||
config,
|
});
|
||||||
syscall_registry,
|
|
||||||
);
|
|
||||||
load_elf_time.stop();
|
load_elf_time.stop();
|
||||||
create_executor_metrics.load_elf_us = load_elf_time.as_us();
|
create_executor_metrics.load_elf_us = load_elf_time.as_us();
|
||||||
invoke_context.timings.create_executor_load_elf_us = invoke_context
|
let executable = executable?;
|
||||||
.timings
|
|
||||||
.create_executor_load_elf_us
|
|
||||||
.saturating_add(create_executor_metrics.load_elf_us);
|
|
||||||
executable
|
|
||||||
}
|
|
||||||
.map_err(|e| map_ebpf_error(invoke_context, e))?;
|
|
||||||
let mut verify_code_time = Measure::start("verify_code_time");
|
let mut verify_code_time = Measure::start("verify_code_time");
|
||||||
let mut verified_executable =
|
let mut verified_executable =
|
||||||
VerifiedExecutable::<RequisiteVerifier, ThisInstructionMeter>::from_executable(executable)
|
VerifiedExecutable::<RequisiteVerifier, ThisInstructionMeter>::from_executable(executable)
|
||||||
.map_err(|e| map_ebpf_error(invoke_context, e))?;
|
.map_err(|err| {
|
||||||
|
ic_logger_msg!(log_collector, "{}", err);
|
||||||
|
InstructionError::InvalidAccountData
|
||||||
|
})?;
|
||||||
verify_code_time.stop();
|
verify_code_time.stop();
|
||||||
create_executor_metrics.verify_code_us = verify_code_time.as_us();
|
create_executor_metrics.verify_code_us = verify_code_time.as_us();
|
||||||
invoke_context.timings.create_executor_verify_code_us = invoke_context
|
|
||||||
.timings
|
|
||||||
.create_executor_verify_code_us
|
|
||||||
.saturating_add(create_executor_metrics.verify_code_us);
|
|
||||||
if use_jit {
|
if use_jit {
|
||||||
let mut jit_compile_time = Measure::start("jit_compile_time");
|
let mut jit_compile_time = Measure::start("jit_compile_time");
|
||||||
let jit_compile_result = verified_executable.jit_compile();
|
let jit_compile_result = verified_executable.jit_compile();
|
||||||
jit_compile_time.stop();
|
jit_compile_time.stop();
|
||||||
create_executor_metrics.jit_compile_us = jit_compile_time.as_us();
|
create_executor_metrics.jit_compile_us = jit_compile_time.as_us();
|
||||||
invoke_context.timings.create_executor_jit_compile_us = invoke_context
|
|
||||||
.timings
|
|
||||||
.create_executor_jit_compile_us
|
|
||||||
.saturating_add(create_executor_metrics.jit_compile_us);
|
|
||||||
if let Err(err) = jit_compile_result {
|
if let Err(err) = jit_compile_result {
|
||||||
ic_msg!(invoke_context, "Failed to compile program {:?}", err);
|
ic_logger_msg!(log_collector, "Failed to compile program {:?}", err);
|
||||||
return Err(InstructionError::ProgramFailedToCompile);
|
return Err(InstructionError::ProgramFailedToCompile);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
create_executor_metrics.submit_datapoint();
|
|
||||||
Ok(Arc::new(BpfExecutor {
|
Ok(Arc::new(BpfExecutor {
|
||||||
verified_executable,
|
verified_executable,
|
||||||
use_jit,
|
use_jit,
|
||||||
}))
|
}))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn create_executor_from_account(
|
||||||
|
feature_set: &FeatureSet,
|
||||||
|
compute_budget: &ComputeBudget,
|
||||||
|
log_collector: Option<Rc<RefCell<LogCollector>>>,
|
||||||
|
tx_executor_cache: Option<RefMut<TransactionExecutorCache>>,
|
||||||
|
program: &BorrowedAccount,
|
||||||
|
programdata: &BorrowedAccount,
|
||||||
|
use_jit: bool,
|
||||||
|
) -> Result<(Arc<dyn Executor>, Option<executor_metrics::CreateMetrics>), InstructionError> {
|
||||||
|
if !check_loader_id(program.get_owner()) {
|
||||||
|
ic_logger_msg!(
|
||||||
|
log_collector,
|
||||||
|
"Executable account not owned by the BPF loader"
|
||||||
|
);
|
||||||
|
return Err(InstructionError::IncorrectProgramId);
|
||||||
|
}
|
||||||
|
|
||||||
|
let programdata_offset = if bpf_loader_upgradeable::check_id(program.get_owner()) {
|
||||||
|
if let UpgradeableLoaderState::Program {
|
||||||
|
programdata_address,
|
||||||
|
} = program.get_state()?
|
||||||
|
{
|
||||||
|
if &programdata_address != programdata.get_key() {
|
||||||
|
ic_logger_msg!(
|
||||||
|
log_collector,
|
||||||
|
"Wrong ProgramData account for this Program account"
|
||||||
|
);
|
||||||
|
return Err(InstructionError::InvalidArgument);
|
||||||
|
}
|
||||||
|
if !matches!(
|
||||||
|
programdata.get_state()?,
|
||||||
|
UpgradeableLoaderState::ProgramData {
|
||||||
|
slot: _,
|
||||||
|
upgrade_authority_address: _,
|
||||||
|
}
|
||||||
|
) {
|
||||||
|
ic_logger_msg!(log_collector, "Program has been closed");
|
||||||
|
return Err(InstructionError::InvalidAccountData);
|
||||||
|
}
|
||||||
|
UpgradeableLoaderState::size_of_programdata_metadata()
|
||||||
|
} else {
|
||||||
|
ic_logger_msg!(log_collector, "Invalid Program account");
|
||||||
|
return Err(InstructionError::InvalidAccountData);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
0
|
||||||
|
};
|
||||||
|
|
||||||
|
if let Some(ref tx_executor_cache) = tx_executor_cache {
|
||||||
|
if let Some(executor) = tx_executor_cache.get(program.get_key()) {
|
||||||
|
return Ok((executor, None));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
let mut create_executor_metrics = executor_metrics::CreateMetrics {
|
||||||
|
program_id: program.get_key().to_string(),
|
||||||
|
..executor_metrics::CreateMetrics::default()
|
||||||
|
};
|
||||||
|
let executor = create_executor_from_bytes(
|
||||||
|
feature_set,
|
||||||
|
compute_budget,
|
||||||
|
log_collector,
|
||||||
|
&mut create_executor_metrics,
|
||||||
|
programdata
|
||||||
|
.get_data()
|
||||||
|
.get(programdata_offset..)
|
||||||
|
.ok_or(InstructionError::AccountDataTooSmall)?,
|
||||||
|
use_jit,
|
||||||
|
false, /* reject_deployment_of_broken_elfs */
|
||||||
|
)?;
|
||||||
|
if let Some(mut tx_executor_cache) = tx_executor_cache {
|
||||||
|
tx_executor_cache.set(*program.get_key(), executor.clone(), false);
|
||||||
|
}
|
||||||
|
Ok((executor, Some(create_executor_metrics)))
|
||||||
|
}
|
||||||
|
|
||||||
fn write_program_data(
|
fn write_program_data(
|
||||||
program_account_index: IndexOfAccount,
|
program_account_index: IndexOfAccount,
|
||||||
program_data_offset: usize,
|
program_data_offset: usize,
|
||||||
|
@ -397,76 +458,35 @@ fn process_instruction_common(
|
||||||
invoke_context.get_stack_height() > 1
|
invoke_context.get_stack_height() > 1
|
||||||
);
|
);
|
||||||
|
|
||||||
if !check_loader_id(program.get_owner()) {
|
let programdata = if program_account_index == first_instruction_account {
|
||||||
ic_logger_msg!(
|
None
|
||||||
log_collector,
|
|
||||||
"Executable account not owned by the BPF loader"
|
|
||||||
);
|
|
||||||
return Err(InstructionError::IncorrectProgramId);
|
|
||||||
}
|
|
||||||
|
|
||||||
let program_data_offset = if bpf_loader_upgradeable::check_id(program.get_owner()) {
|
|
||||||
if let UpgradeableLoaderState::Program {
|
|
||||||
programdata_address,
|
|
||||||
} = program.get_state()?
|
|
||||||
{
|
|
||||||
if programdata_address != *first_account_key {
|
|
||||||
ic_logger_msg!(
|
|
||||||
log_collector,
|
|
||||||
"Wrong ProgramData account for this Program account"
|
|
||||||
);
|
|
||||||
return Err(InstructionError::InvalidArgument);
|
|
||||||
}
|
|
||||||
if !matches!(
|
|
||||||
instruction_context
|
|
||||||
.try_borrow_program_account(transaction_context, first_instruction_account)?
|
|
||||||
.get_state()?,
|
|
||||||
UpgradeableLoaderState::ProgramData {
|
|
||||||
slot: _,
|
|
||||||
upgrade_authority_address: _,
|
|
||||||
}
|
|
||||||
) {
|
|
||||||
ic_logger_msg!(log_collector, "Program has been closed");
|
|
||||||
return Err(InstructionError::InvalidAccountData);
|
|
||||||
}
|
|
||||||
UpgradeableLoaderState::size_of_programdata_metadata()
|
|
||||||
} else {
|
} else {
|
||||||
ic_logger_msg!(log_collector, "Invalid Program account");
|
Some(try_borrow_account(
|
||||||
return Err(InstructionError::InvalidAccountData);
|
transaction_context,
|
||||||
}
|
instruction_context,
|
||||||
} else {
|
|
||||||
0
|
|
||||||
};
|
|
||||||
drop(program);
|
|
||||||
|
|
||||||
let mut get_or_create_executor_time = Measure::start("get_or_create_executor_time");
|
|
||||||
let cached_executor = invoke_context.tx_executor_cache.borrow().get(program_id);
|
|
||||||
let executor = if let Some(executor) = cached_executor {
|
|
||||||
executor
|
|
||||||
} else {
|
|
||||||
let executor = create_executor(
|
|
||||||
first_instruction_account,
|
first_instruction_account,
|
||||||
program_data_offset,
|
)?)
|
||||||
invoke_context,
|
|
||||||
use_jit,
|
|
||||||
false, /* reject_deployment_of_broken_elfs */
|
|
||||||
// allow _sol_alloc_free syscall for execution
|
|
||||||
false, /* disable_sol_alloc_free_syscall */
|
|
||||||
)?;
|
|
||||||
let transaction_context = &invoke_context.transaction_context;
|
|
||||||
let instruction_context = transaction_context.get_current_instruction_context()?;
|
|
||||||
let program_id = instruction_context.get_last_program_key(transaction_context)?;
|
|
||||||
invoke_context
|
|
||||||
.tx_executor_cache
|
|
||||||
.borrow_mut()
|
|
||||||
.set(*program_id, executor.clone(), false);
|
|
||||||
executor
|
|
||||||
};
|
};
|
||||||
|
let mut get_or_create_executor_time = Measure::start("get_or_create_executor_time");
|
||||||
|
let (executor, create_executor_metrics) = create_executor_from_account(
|
||||||
|
&invoke_context.feature_set,
|
||||||
|
invoke_context.get_compute_budget(),
|
||||||
|
log_collector,
|
||||||
|
Some(invoke_context.tx_executor_cache.borrow_mut()),
|
||||||
|
&program,
|
||||||
|
programdata.as_ref().unwrap_or(&program),
|
||||||
|
use_jit,
|
||||||
|
)?;
|
||||||
|
drop(program);
|
||||||
|
drop(programdata);
|
||||||
get_or_create_executor_time.stop();
|
get_or_create_executor_time.stop();
|
||||||
saturating_add_assign!(
|
saturating_add_assign!(
|
||||||
invoke_context.timings.get_or_create_executor_us,
|
invoke_context.timings.get_or_create_executor_us,
|
||||||
get_or_create_executor_time.as_us()
|
get_or_create_executor_time.as_us()
|
||||||
);
|
);
|
||||||
|
if let Some(create_executor_metrics) = create_executor_metrics {
|
||||||
|
create_executor_metrics.submit_datapoint(&mut invoke_context.timings);
|
||||||
|
}
|
||||||
|
|
||||||
executor.execute(program_account_index, invoke_context)
|
executor.execute(program_account_index, invoke_context)
|
||||||
} else {
|
} else {
|
||||||
|
@ -677,16 +697,26 @@ fn process_loader_upgradeable_instruction(
|
||||||
invoke_context.native_invoke(instruction, signers.as_slice())?;
|
invoke_context.native_invoke(instruction, signers.as_slice())?;
|
||||||
|
|
||||||
// Load and verify the program bits
|
// Load and verify the program bits
|
||||||
let executor = create_executor(
|
let transaction_context = &invoke_context.transaction_context;
|
||||||
first_instruction_account.saturating_add(3),
|
let instruction_context = transaction_context.get_current_instruction_context()?;
|
||||||
buffer_data_offset,
|
let buffer =
|
||||||
invoke_context,
|
instruction_context.try_borrow_instruction_account(transaction_context, 3)?;
|
||||||
|
let mut create_executor_metrics = executor_metrics::CreateMetrics::default();
|
||||||
|
let executor = create_executor_from_bytes(
|
||||||
|
&invoke_context.feature_set,
|
||||||
|
invoke_context.get_compute_budget(),
|
||||||
|
invoke_context.get_log_collector(),
|
||||||
|
&mut create_executor_metrics,
|
||||||
|
buffer
|
||||||
|
.get_data()
|
||||||
|
.get(buffer_data_offset..)
|
||||||
|
.ok_or(InstructionError::AccountDataTooSmall)?,
|
||||||
use_jit,
|
use_jit,
|
||||||
true,
|
true,
|
||||||
invoke_context
|
|
||||||
.feature_set
|
|
||||||
.is_active(&disable_deploy_of_alloc_free_syscall::id()),
|
|
||||||
)?;
|
)?;
|
||||||
|
drop(buffer);
|
||||||
|
create_executor_metrics.program_id = new_program_id.to_string();
|
||||||
|
create_executor_metrics.submit_datapoint(&mut invoke_context.timings);
|
||||||
invoke_context
|
invoke_context
|
||||||
.tx_executor_cache
|
.tx_executor_cache
|
||||||
.borrow_mut()
|
.borrow_mut()
|
||||||
|
@ -848,16 +878,24 @@ fn process_loader_upgradeable_instruction(
|
||||||
drop(programdata);
|
drop(programdata);
|
||||||
|
|
||||||
// Load and verify the program bits
|
// Load and verify the program bits
|
||||||
let executor = create_executor(
|
let buffer =
|
||||||
first_instruction_account.saturating_add(2),
|
instruction_context.try_borrow_instruction_account(transaction_context, 2)?;
|
||||||
buffer_data_offset,
|
let mut create_executor_metrics = executor_metrics::CreateMetrics::default();
|
||||||
invoke_context,
|
let executor = create_executor_from_bytes(
|
||||||
|
&invoke_context.feature_set,
|
||||||
|
invoke_context.get_compute_budget(),
|
||||||
|
invoke_context.get_log_collector(),
|
||||||
|
&mut create_executor_metrics,
|
||||||
|
buffer
|
||||||
|
.get_data()
|
||||||
|
.get(buffer_data_offset..)
|
||||||
|
.ok_or(InstructionError::AccountDataTooSmall)?,
|
||||||
use_jit,
|
use_jit,
|
||||||
true,
|
true,
|
||||||
invoke_context
|
|
||||||
.feature_set
|
|
||||||
.is_active(&disable_deploy_of_alloc_free_syscall::id()),
|
|
||||||
)?;
|
)?;
|
||||||
|
drop(buffer);
|
||||||
|
create_executor_metrics.program_id = new_program_id.to_string();
|
||||||
|
create_executor_metrics.submit_datapoint(&mut invoke_context.timings);
|
||||||
invoke_context
|
invoke_context
|
||||||
.tx_executor_cache
|
.tx_executor_cache
|
||||||
.borrow_mut()
|
.borrow_mut()
|
||||||
|
@ -1238,7 +1276,7 @@ fn process_loader_instruction(
|
||||||
let instruction_context = transaction_context.get_current_instruction_context()?;
|
let instruction_context = transaction_context.get_current_instruction_context()?;
|
||||||
let instruction_data = instruction_context.get_instruction_data();
|
let instruction_data = instruction_context.get_instruction_data();
|
||||||
let program_id = instruction_context.get_last_program_key(transaction_context)?;
|
let program_id = instruction_context.get_last_program_key(transaction_context)?;
|
||||||
let program = instruction_context.try_borrow_instruction_account(transaction_context, 0)?;
|
let mut program = instruction_context.try_borrow_instruction_account(transaction_context, 0)?;
|
||||||
if program.get_owner() != program_id {
|
if program.get_owner() != program_id {
|
||||||
ic_msg!(
|
ic_msg!(
|
||||||
invoke_context,
|
invoke_context,
|
||||||
|
@ -1247,13 +1285,13 @@ fn process_loader_instruction(
|
||||||
return Err(InstructionError::IncorrectProgramId);
|
return Err(InstructionError::IncorrectProgramId);
|
||||||
}
|
}
|
||||||
let is_program_signer = program.is_signer();
|
let is_program_signer = program.is_signer();
|
||||||
drop(program);
|
|
||||||
match limited_deserialize(instruction_data)? {
|
match limited_deserialize(instruction_data)? {
|
||||||
LoaderInstruction::Write { offset, bytes } => {
|
LoaderInstruction::Write { offset, bytes } => {
|
||||||
if !is_program_signer {
|
if !is_program_signer {
|
||||||
ic_msg!(invoke_context, "Program account did not sign");
|
ic_msg!(invoke_context, "Program account did not sign");
|
||||||
return Err(InstructionError::MissingRequiredSignature);
|
return Err(InstructionError::MissingRequiredSignature);
|
||||||
}
|
}
|
||||||
|
drop(program);
|
||||||
write_program_data(
|
write_program_data(
|
||||||
first_instruction_account,
|
first_instruction_account,
|
||||||
offset as usize,
|
offset as usize,
|
||||||
|
@ -1266,20 +1304,18 @@ fn process_loader_instruction(
|
||||||
ic_msg!(invoke_context, "key[0] did not sign the transaction");
|
ic_msg!(invoke_context, "key[0] did not sign the transaction");
|
||||||
return Err(InstructionError::MissingRequiredSignature);
|
return Err(InstructionError::MissingRequiredSignature);
|
||||||
}
|
}
|
||||||
let executor = create_executor(
|
let mut create_executor_metrics = executor_metrics::CreateMetrics::default();
|
||||||
first_instruction_account,
|
let executor = create_executor_from_bytes(
|
||||||
0,
|
&invoke_context.feature_set,
|
||||||
invoke_context,
|
invoke_context.get_compute_budget(),
|
||||||
|
invoke_context.get_log_collector(),
|
||||||
|
&mut create_executor_metrics,
|
||||||
|
program.get_data(),
|
||||||
use_jit,
|
use_jit,
|
||||||
true,
|
true,
|
||||||
invoke_context
|
|
||||||
.feature_set
|
|
||||||
.is_active(&disable_deploy_of_alloc_free_syscall::id()),
|
|
||||||
)?;
|
)?;
|
||||||
let transaction_context = &invoke_context.transaction_context;
|
create_executor_metrics.program_id = program.get_key().to_string();
|
||||||
let instruction_context = transaction_context.get_current_instruction_context()?;
|
create_executor_metrics.submit_datapoint(&mut invoke_context.timings);
|
||||||
let mut program =
|
|
||||||
instruction_context.try_borrow_instruction_account(transaction_context, 0)?;
|
|
||||||
invoke_context
|
invoke_context
|
||||||
.tx_executor_cache
|
.tx_executor_cache
|
||||||
.borrow_mut()
|
.borrow_mut()
|
||||||
|
|
|
@ -28,6 +28,7 @@ use {
|
||||||
account_info::AccountInfo,
|
account_info::AccountInfo,
|
||||||
blake3, bpf_loader, bpf_loader_deprecated, bpf_loader_upgradeable,
|
blake3, bpf_loader, bpf_loader_deprecated, bpf_loader_upgradeable,
|
||||||
entrypoint::{BPF_ALIGN_OF_U128, MAX_PERMITTED_DATA_INCREASE, SUCCESS},
|
entrypoint::{BPF_ALIGN_OF_U128, MAX_PERMITTED_DATA_INCREASE, SUCCESS},
|
||||||
|
feature_set::FeatureSet,
|
||||||
feature_set::{
|
feature_set::{
|
||||||
self, blake3_syscall_enabled, check_physical_overlapping, curve25519_syscall_enabled,
|
self, blake3_syscall_enabled, check_physical_overlapping, curve25519_syscall_enabled,
|
||||||
disable_cpi_setting_executable_and_rent_epoch, disable_fees_sysvar,
|
disable_cpi_setting_executable_and_rent_epoch, disable_fees_sysvar,
|
||||||
|
@ -151,18 +152,12 @@ macro_rules! register_feature_gated_syscall {
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn register_syscalls(
|
pub fn register_syscalls(
|
||||||
invoke_context: &mut InvokeContext,
|
feature_set: &FeatureSet,
|
||||||
disable_deploy_of_alloc_free_syscall: bool,
|
disable_deploy_of_alloc_free_syscall: bool,
|
||||||
) -> Result<SyscallRegistry, EbpfError> {
|
) -> Result<SyscallRegistry, EbpfError> {
|
||||||
let blake3_syscall_enabled = invoke_context
|
let blake3_syscall_enabled = feature_set.is_active(&blake3_syscall_enabled::id());
|
||||||
.feature_set
|
let curve25519_syscall_enabled = feature_set.is_active(&curve25519_syscall_enabled::id());
|
||||||
.is_active(&blake3_syscall_enabled::id());
|
let disable_fees_sysvar = feature_set.is_active(&disable_fees_sysvar::id());
|
||||||
let curve25519_syscall_enabled = invoke_context
|
|
||||||
.feature_set
|
|
||||||
.is_active(&curve25519_syscall_enabled::id());
|
|
||||||
let disable_fees_sysvar = invoke_context
|
|
||||||
.feature_set
|
|
||||||
.is_active(&disable_fees_sysvar::id());
|
|
||||||
let is_abi_v2 = false;
|
let is_abi_v2 = false;
|
||||||
|
|
||||||
let mut syscall_registry = SyscallRegistry::default();
|
let mut syscall_registry = SyscallRegistry::default();
|
||||||
|
|
|
@ -110,7 +110,7 @@ fn bench_program_alu(bencher: &mut Bencher) {
|
||||||
let executable = Executable::<ThisInstructionMeter>::from_elf(
|
let executable = Executable::<ThisInstructionMeter>::from_elf(
|
||||||
&elf,
|
&elf,
|
||||||
Config::default(),
|
Config::default(),
|
||||||
register_syscalls(invoke_context, true).unwrap(),
|
register_syscalls(&invoke_context.feature_set, true).unwrap(),
|
||||||
)
|
)
|
||||||
.unwrap();
|
.unwrap();
|
||||||
|
|
||||||
|
@ -238,7 +238,7 @@ fn bench_create_vm(bencher: &mut Bencher) {
|
||||||
let executable = Executable::<ThisInstructionMeter>::from_elf(
|
let executable = Executable::<ThisInstructionMeter>::from_elf(
|
||||||
&elf,
|
&elf,
|
||||||
Config::default(),
|
Config::default(),
|
||||||
register_syscalls(invoke_context, true).unwrap(),
|
register_syscalls(&invoke_context.feature_set, true).unwrap(),
|
||||||
)
|
)
|
||||||
.unwrap();
|
.unwrap();
|
||||||
|
|
||||||
|
@ -285,7 +285,7 @@ fn bench_instruction_count_tuner(_bencher: &mut Bencher) {
|
||||||
let executable = Executable::<ThisInstructionMeter>::from_elf(
|
let executable = Executable::<ThisInstructionMeter>::from_elf(
|
||||||
&elf,
|
&elf,
|
||||||
Config::default(),
|
Config::default(),
|
||||||
register_syscalls(invoke_context, true).unwrap(),
|
register_syscalls(&invoke_context.feature_set, true).unwrap(),
|
||||||
)
|
)
|
||||||
.unwrap();
|
.unwrap();
|
||||||
|
|
||||||
|
|
|
@ -234,7 +234,11 @@ fn run_program(name: &str) -> u64 {
|
||||||
let executable = Executable::<ThisInstructionMeter>::from_elf(
|
let executable = Executable::<ThisInstructionMeter>::from_elf(
|
||||||
&data,
|
&data,
|
||||||
config,
|
config,
|
||||||
register_syscalls(invoke_context, true /* no sol_alloc_free */).unwrap(),
|
register_syscalls(
|
||||||
|
&invoke_context.feature_set,
|
||||||
|
true, /* no sol_alloc_free */
|
||||||
|
)
|
||||||
|
.unwrap(),
|
||||||
)
|
)
|
||||||
.unwrap();
|
.unwrap();
|
||||||
|
|
||||||
|
|
|
@ -252,7 +252,7 @@ native machine code before execting it in the virtual machine.",
|
||||||
file.seek(SeekFrom::Start(0)).unwrap();
|
file.seek(SeekFrom::Start(0)).unwrap();
|
||||||
let mut contents = Vec::new();
|
let mut contents = Vec::new();
|
||||||
file.read_to_end(&mut contents).unwrap();
|
file.read_to_end(&mut contents).unwrap();
|
||||||
let syscall_registry = register_syscalls(&mut invoke_context, true).unwrap();
|
let syscall_registry = register_syscalls(&invoke_context.feature_set, true).unwrap();
|
||||||
let executable = if magic == [0x7f, 0x45, 0x4c, 0x46] {
|
let executable = if magic == [0x7f, 0x45, 0x4c, 0x46] {
|
||||||
Executable::<ThisInstructionMeter>::from_elf(&contents, config, syscall_registry)
|
Executable::<ThisInstructionMeter>::from_elf(&contents, config, syscall_registry)
|
||||||
.map_err(|err| format!("Executable constructor failed: {:?}", err))
|
.map_err(|err| format!("Executable constructor failed: {:?}", err))
|
||||||
|
|
Loading…
Reference in New Issue