parent
5f6755f58b
commit
0335ea7249
|
@ -7144,9 +7144,9 @@ dependencies = [
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "solana_rbpf"
|
name = "solana_rbpf"
|
||||||
version = "0.2.39"
|
version = "0.2.40"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "199a2e8ced67a32b47695256509d5a62b39cb7340f52f0a5d8de99f4dc2aeb43"
|
checksum = "1a5735b8c9defc3723162321a61ef738d34168401eeef213f62a32809739b0f5"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"byteorder",
|
"byteorder",
|
||||||
"combine",
|
"combine",
|
||||||
|
|
|
@ -287,7 +287,7 @@ signal-hook = "0.3.14"
|
||||||
smpl_jwt = "0.7.1"
|
smpl_jwt = "0.7.1"
|
||||||
socket2 = "0.4.7"
|
socket2 = "0.4.7"
|
||||||
soketto = "0.7"
|
soketto = "0.7"
|
||||||
solana_rbpf = "=0.2.39"
|
solana_rbpf = "=0.2.40"
|
||||||
solana-account-decoder = { path = "account-decoder", version = "=1.16.0" }
|
solana-account-decoder = { path = "account-decoder", version = "=1.16.0" }
|
||||||
solana-address-lookup-table-program = { path = "programs/address-lookup-table", version = "=1.16.0" }
|
solana-address-lookup-table-program = { path = "programs/address-lookup-table", version = "=1.16.0" }
|
||||||
solana-banks-client = { path = "banks-client", version = "=1.16.0" }
|
solana-banks-client = { path = "banks-client", version = "=1.16.0" }
|
||||||
|
|
|
@ -26,8 +26,8 @@ impl BpfAllocator {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn get_heap(&mut self) -> &mut [u8] {
|
pub fn heap_mut(&mut self) -> &mut AlignedMemory<HOST_ALIGN> {
|
||||||
self.heap.as_slice_mut()
|
&mut self.heap
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -28,9 +28,10 @@ use {
|
||||||
},
|
},
|
||||||
solana_rbpf::{
|
solana_rbpf::{
|
||||||
aligned_memory::AlignedMemory,
|
aligned_memory::AlignedMemory,
|
||||||
ebpf::{HOST_ALIGN, MM_HEAP_START},
|
ebpf::{self, HOST_ALIGN, MM_HEAP_START},
|
||||||
|
elf::Executable,
|
||||||
error::{EbpfError, UserDefinedError},
|
error::{EbpfError, UserDefinedError},
|
||||||
memory_region::MemoryRegion,
|
memory_region::{MemoryCowCallback, MemoryMapping, MemoryRegion},
|
||||||
verifier::{RequisiteVerifier, VerifierError},
|
verifier::{RequisiteVerifier, VerifierError},
|
||||||
vm::{ContextObject, EbpfVm, ProgramResult, VerifiedExecutable},
|
vm::{ContextObject, EbpfVm, ProgramResult, VerifiedExecutable},
|
||||||
},
|
},
|
||||||
|
@ -38,7 +39,7 @@ use {
|
||||||
bpf_loader, bpf_loader_deprecated,
|
bpf_loader, bpf_loader_deprecated,
|
||||||
bpf_loader_upgradeable::{self, UpgradeableLoaderState},
|
bpf_loader_upgradeable::{self, UpgradeableLoaderState},
|
||||||
clock::Slot,
|
clock::Slot,
|
||||||
entrypoint::{HEAP_LENGTH, SUCCESS},
|
entrypoint::SUCCESS,
|
||||||
feature_set::{
|
feature_set::{
|
||||||
cap_accounts_data_allocations_per_transaction, cap_bpf_program_instruction_accounts,
|
cap_accounts_data_allocations_per_transaction, cap_bpf_program_instruction_accounts,
|
||||||
check_slice_translation_size, delay_visibility_of_program_deployment,
|
check_slice_translation_size, delay_visibility_of_program_deployment,
|
||||||
|
@ -64,6 +65,7 @@ use {
|
||||||
std::{
|
std::{
|
||||||
cell::{RefCell, RefMut},
|
cell::{RefCell, RefMut},
|
||||||
fmt::Debug,
|
fmt::Debug,
|
||||||
|
mem,
|
||||||
rc::Rc,
|
rc::Rc,
|
||||||
sync::Arc,
|
sync::Arc,
|
||||||
},
|
},
|
||||||
|
@ -296,21 +298,19 @@ fn check_loader_id(id: &Pubkey) -> bool {
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Create the SBF virtual machine
|
/// Create the SBF virtual machine
|
||||||
pub fn create_vm<'a, 'b>(
|
pub fn create_ebpf_vm<'a, 'b>(
|
||||||
program: &'a VerifiedExecutable<RequisiteVerifier, InvokeContext<'b>>,
|
program: &'a VerifiedExecutable<RequisiteVerifier, InvokeContext<'b>>,
|
||||||
|
stack: &'a mut AlignedMemory<HOST_ALIGN>,
|
||||||
|
heap: AlignedMemory<HOST_ALIGN>,
|
||||||
regions: Vec<MemoryRegion>,
|
regions: Vec<MemoryRegion>,
|
||||||
orig_account_lengths: Vec<usize>,
|
orig_account_lengths: Vec<usize>,
|
||||||
invoke_context: &'a mut InvokeContext<'b>,
|
invoke_context: &'a mut InvokeContext<'b>,
|
||||||
) -> Result<EbpfVm<'a, RequisiteVerifier, InvokeContext<'b>>, EbpfError> {
|
) -> Result<EbpfVm<'a, RequisiteVerifier, InvokeContext<'b>>, EbpfError> {
|
||||||
let compute_budget = invoke_context.get_compute_budget();
|
|
||||||
let heap_size = compute_budget.heap_size.unwrap_or(HEAP_LENGTH);
|
|
||||||
let _ = invoke_context.consume_checked(
|
let _ = invoke_context.consume_checked(
|
||||||
((heap_size as u64).saturating_div(32_u64.saturating_mul(1024)))
|
((heap.len() as u64).saturating_div(32_u64.saturating_mul(1024)))
|
||||||
.saturating_sub(1)
|
.saturating_sub(1)
|
||||||
.saturating_mul(compute_budget.heap_cost),
|
.saturating_mul(invoke_context.get_compute_budget().heap_cost),
|
||||||
);
|
);
|
||||||
let heap =
|
|
||||||
AlignedMemory::<HOST_ALIGN>::zero_filled(compute_budget.heap_size.unwrap_or(HEAP_LENGTH));
|
|
||||||
let check_aligned = bpf_loader_deprecated::id()
|
let check_aligned = bpf_loader_deprecated::id()
|
||||||
!= invoke_context
|
!= invoke_context
|
||||||
.transaction_context
|
.transaction_context
|
||||||
|
@ -333,13 +333,76 @@ pub fn create_vm<'a, 'b>(
|
||||||
allocator.clone(),
|
allocator.clone(),
|
||||||
)
|
)
|
||||||
.map_err(SyscallError::InstructionError)?;
|
.map_err(SyscallError::InstructionError)?;
|
||||||
let result = EbpfVm::new(
|
let stack_len = stack.len();
|
||||||
program,
|
let memory_mapping = create_memory_mapping(
|
||||||
invoke_context,
|
program.get_executable(),
|
||||||
allocator.borrow_mut().get_heap(),
|
stack,
|
||||||
|
allocator.borrow_mut().heap_mut(),
|
||||||
regions,
|
regions,
|
||||||
|
None,
|
||||||
|
)?;
|
||||||
|
|
||||||
|
EbpfVm::new(program, invoke_context, memory_mapping, stack_len)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[macro_export]
|
||||||
|
macro_rules! create_vm {
|
||||||
|
($vm_name:ident, $executable:expr, $stack:ident, $heap:ident, $additional_regions:expr, $orig_account_lengths:expr, $invoke_context:expr) => {
|
||||||
|
let mut $stack = solana_rbpf::aligned_memory::AlignedMemory::<
|
||||||
|
{ solana_rbpf::ebpf::HOST_ALIGN },
|
||||||
|
>::zero_filled($executable.get_executable().get_config().stack_size());
|
||||||
|
|
||||||
|
// this is needed if the caller passes "&mut invoke_context" to the
|
||||||
|
// macro. The lint complains that (&mut invoke_context).get_compute_budget()
|
||||||
|
// does an unnecessary mutable borrow
|
||||||
|
#[allow(clippy::unnecessary_mut_passed)]
|
||||||
|
let heap_size = $invoke_context
|
||||||
|
.get_compute_budget()
|
||||||
|
.heap_size
|
||||||
|
.unwrap_or(solana_sdk::entrypoint::HEAP_LENGTH);
|
||||||
|
let $heap = solana_rbpf::aligned_memory::AlignedMemory::<{ solana_rbpf::ebpf::HOST_ALIGN }>::zero_filled(heap_size);
|
||||||
|
|
||||||
|
let $vm_name = create_ebpf_vm(
|
||||||
|
$executable,
|
||||||
|
&mut $stack,
|
||||||
|
$heap,
|
||||||
|
$additional_regions,
|
||||||
|
$orig_account_lengths,
|
||||||
|
$invoke_context,
|
||||||
);
|
);
|
||||||
result
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
fn create_memory_mapping<'a, 'b, C: ContextObject>(
|
||||||
|
executable: &'a Executable<C>,
|
||||||
|
stack: &'b mut AlignedMemory<{ HOST_ALIGN }>,
|
||||||
|
heap: &'b mut AlignedMemory<{ HOST_ALIGN }>,
|
||||||
|
additional_regions: Vec<MemoryRegion>,
|
||||||
|
cow_cb: Option<MemoryCowCallback>,
|
||||||
|
) -> Result<MemoryMapping<'a>, EbpfError> {
|
||||||
|
let config = executable.get_config();
|
||||||
|
let regions: Vec<MemoryRegion> = vec![
|
||||||
|
executable.get_ro_region(),
|
||||||
|
MemoryRegion::new_writable_gapped(
|
||||||
|
stack.as_slice_mut(),
|
||||||
|
ebpf::MM_STACK_START,
|
||||||
|
if !config.dynamic_stack_frames && config.enable_stack_frame_gaps {
|
||||||
|
config.stack_frame_size as u64
|
||||||
|
} else {
|
||||||
|
0
|
||||||
|
},
|
||||||
|
),
|
||||||
|
MemoryRegion::new_writable(heap.as_slice_mut(), ebpf::MM_HEAP_START),
|
||||||
|
]
|
||||||
|
.into_iter()
|
||||||
|
.chain(additional_regions.into_iter())
|
||||||
|
.collect();
|
||||||
|
|
||||||
|
Ok(if let Some(cow_cb) = cow_cb {
|
||||||
|
MemoryMapping::new_with_cow(regions, cow_cb, config)?
|
||||||
|
} else {
|
||||||
|
MemoryMapping::new(regions, config)?
|
||||||
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn process_instruction(invoke_context: &mut InvokeContext) -> Result<(), InstructionError> {
|
pub fn process_instruction(invoke_context: &mut InvokeContext) -> Result<(), InstructionError> {
|
||||||
|
@ -1345,9 +1408,9 @@ fn process_loader_instruction(
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
fn execute(
|
fn execute<'a, 'b: 'a>(
|
||||||
executable: &VerifiedExecutable<RequisiteVerifier, InvokeContext<'static>>,
|
executable: &'a VerifiedExecutable<RequisiteVerifier, InvokeContext<'static>>,
|
||||||
invoke_context: &mut InvokeContext,
|
invoke_context: &'a mut InvokeContext<'b>,
|
||||||
) -> Result<(), InstructionError> {
|
) -> Result<(), InstructionError> {
|
||||||
let log_collector = invoke_context.get_log_collector();
|
let log_collector = invoke_context.get_log_collector();
|
||||||
let stack_height = invoke_context.get_stack_height();
|
let stack_height = invoke_context.get_stack_height();
|
||||||
|
@ -1373,14 +1436,22 @@ fn execute(
|
||||||
let mut execute_time;
|
let mut execute_time;
|
||||||
let execution_result = {
|
let execution_result = {
|
||||||
let compute_meter_prev = invoke_context.get_remaining();
|
let compute_meter_prev = invoke_context.get_remaining();
|
||||||
let mut vm = match create_vm(
|
create_vm!(
|
||||||
|
vm,
|
||||||
// We dropped the lifetime tracking in the Executor by setting it to 'static,
|
// We dropped the lifetime tracking in the Executor by setting it to 'static,
|
||||||
// thus we need to reintroduce the correct lifetime of InvokeContext here again.
|
// thus we need to reintroduce the correct lifetime of InvokeContext here again.
|
||||||
unsafe { std::mem::transmute(executable) },
|
unsafe {
|
||||||
|
mem::transmute::<_, &'a VerifiedExecutable<RequisiteVerifier, InvokeContext<'b>>>(
|
||||||
|
executable,
|
||||||
|
)
|
||||||
|
},
|
||||||
|
stack,
|
||||||
|
heap,
|
||||||
regions,
|
regions,
|
||||||
account_lengths,
|
account_lengths,
|
||||||
invoke_context,
|
invoke_context
|
||||||
) {
|
);
|
||||||
|
let mut vm = match vm {
|
||||||
Ok(info) => info,
|
Ok(info) => info,
|
||||||
Err(e) => {
|
Err(e) => {
|
||||||
ic_logger_msg!(log_collector, "Failed to create SBF VM: {}", e);
|
ic_logger_msg!(log_collector, "Failed to create SBF VM: {}", e);
|
||||||
|
@ -1592,11 +1663,28 @@ mod tests {
|
||||||
.unwrap();
|
.unwrap();
|
||||||
let input_region = MemoryRegion::new_writable(&mut input_mem, MM_INPUT_START);
|
let input_region = MemoryRegion::new_writable(&mut input_mem, MM_INPUT_START);
|
||||||
let mut context_object = TestContextObject { remaining: 10 };
|
let mut context_object = TestContextObject { remaining: 10 };
|
||||||
|
let mut stack = AlignedMemory::zero_filled(
|
||||||
|
verified_executable
|
||||||
|
.get_executable()
|
||||||
|
.get_config()
|
||||||
|
.stack_size(),
|
||||||
|
);
|
||||||
|
let mut heap = AlignedMemory::with_capacity(0);
|
||||||
|
let stack_len = stack.len();
|
||||||
|
|
||||||
|
let memory_mapping = create_memory_mapping(
|
||||||
|
verified_executable.get_executable(),
|
||||||
|
&mut stack,
|
||||||
|
&mut heap,
|
||||||
|
vec![input_region],
|
||||||
|
None,
|
||||||
|
)
|
||||||
|
.unwrap();
|
||||||
let mut vm = EbpfVm::new(
|
let mut vm = EbpfVm::new(
|
||||||
&verified_executable,
|
&verified_executable,
|
||||||
&mut context_object,
|
&mut context_object,
|
||||||
&mut [],
|
memory_mapping,
|
||||||
vec![input_region],
|
stack_len,
|
||||||
)
|
)
|
||||||
.unwrap();
|
.unwrap();
|
||||||
vm.execute_program(true).1.unwrap();
|
vm.execute_program(true).1.unwrap();
|
||||||
|
|
|
@ -1333,7 +1333,7 @@ mod tests {
|
||||||
..Config::default()
|
..Config::default()
|
||||||
};
|
};
|
||||||
let memory_mapping =
|
let memory_mapping =
|
||||||
MemoryMapping::new(vec![mock_caller_account.region.clone()], &config).unwrap();
|
MemoryMapping::new(mock_caller_account.regions.split_off(0), &config).unwrap();
|
||||||
|
|
||||||
let mut caller_account = mock_caller_account.caller_account();
|
let mut caller_account = mock_caller_account.caller_account();
|
||||||
|
|
||||||
|
@ -1390,7 +1390,7 @@ mod tests {
|
||||||
..Config::default()
|
..Config::default()
|
||||||
};
|
};
|
||||||
let memory_mapping =
|
let memory_mapping =
|
||||||
MemoryMapping::new(vec![mock_caller_account.region.clone()], &config).unwrap();
|
MemoryMapping::new(mock_caller_account.regions.split_off(0), &config).unwrap();
|
||||||
|
|
||||||
let data_slice = mock_caller_account.data_slice();
|
let data_slice = mock_caller_account.data_slice();
|
||||||
let len_ptr = unsafe {
|
let len_ptr = unsafe {
|
||||||
|
@ -1608,7 +1608,7 @@ mod tests {
|
||||||
vm_addr: u64,
|
vm_addr: u64,
|
||||||
data: Vec<u8>,
|
data: Vec<u8>,
|
||||||
len: u64,
|
len: u64,
|
||||||
region: MemoryRegion,
|
regions: Vec<MemoryRegion>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl MockCallerAccount {
|
impl MockCallerAccount {
|
||||||
|
@ -1620,7 +1620,7 @@ mod tests {
|
||||||
d[mem::size_of::<u64>()..][..data.len()].copy_from_slice(data);
|
d[mem::size_of::<u64>()..][..data.len()].copy_from_slice(data);
|
||||||
|
|
||||||
// the memory region must include the realloc data
|
// the memory region must include the realloc data
|
||||||
let region = MemoryRegion::new_writable(d.as_mut_slice(), vm_addr);
|
let regions = vec![MemoryRegion::new_writable(d.as_mut_slice(), vm_addr)];
|
||||||
|
|
||||||
// caller_account.data must have the actual data length
|
// caller_account.data must have the actual data length
|
||||||
d.truncate(mem::size_of::<u64>() + data.len());
|
d.truncate(mem::size_of::<u64>() + data.len());
|
||||||
|
@ -1631,7 +1631,7 @@ mod tests {
|
||||||
vm_addr,
|
vm_addr,
|
||||||
data: d,
|
data: d,
|
||||||
len: data.len() as u64,
|
len: data.len() as u64,
|
||||||
region,
|
regions,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1803,12 +1803,15 @@ declare_syscall!(
|
||||||
);
|
);
|
||||||
|
|
||||||
#[cfg(test)]
|
#[cfg(test)]
|
||||||
|
#[allow(clippy::integer_arithmetic)]
|
||||||
|
#[allow(clippy::indexing_slicing)]
|
||||||
mod tests {
|
mod tests {
|
||||||
#[allow(deprecated)]
|
#[allow(deprecated)]
|
||||||
use solana_sdk::sysvar::fees::Fees;
|
use solana_sdk::sysvar::fees::Fees;
|
||||||
use {
|
use {
|
||||||
super::*,
|
super::*,
|
||||||
crate::BpfAllocator,
|
crate::BpfAllocator,
|
||||||
|
core::slice,
|
||||||
solana_program_runtime::{invoke_context::InvokeContext, sysvar_cache::SysvarCache},
|
solana_program_runtime::{invoke_context::InvokeContext, sysvar_cache::SysvarCache},
|
||||||
solana_rbpf::{
|
solana_rbpf::{
|
||||||
aligned_memory::AlignedMemory,
|
aligned_memory::AlignedMemory,
|
||||||
|
@ -1827,7 +1830,7 @@ mod tests {
|
||||||
sysvar::{clock::Clock, epoch_schedule::EpochSchedule, rent::Rent},
|
sysvar::{clock::Clock, epoch_schedule::EpochSchedule, rent::Rent},
|
||||||
transaction_context::TransactionContext,
|
transaction_context::TransactionContext,
|
||||||
},
|
},
|
||||||
std::{borrow::Cow, cell::RefCell, rc::Rc, str::FromStr},
|
std::{borrow::Cow, cell::RefCell, mem, rc::Rc, str::FromStr},
|
||||||
};
|
};
|
||||||
|
|
||||||
macro_rules! assert_access_violation {
|
macro_rules! assert_access_violation {
|
||||||
|
@ -1877,6 +1880,7 @@ mod tests {
|
||||||
fn test_translate() {
|
fn test_translate() {
|
||||||
const START: u64 = 0x100000000;
|
const START: u64 = 0x100000000;
|
||||||
const LENGTH: u64 = 1000;
|
const LENGTH: u64 = 1000;
|
||||||
|
|
||||||
let data = vec![0u8; LENGTH as usize];
|
let data = vec![0u8; LENGTH as usize];
|
||||||
let addr = data.as_ptr() as u64;
|
let addr = data.as_ptr() as u64;
|
||||||
let config = Config::default();
|
let config = Config::default();
|
||||||
|
@ -1913,19 +1917,12 @@ mod tests {
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn test_translate_type() {
|
fn test_translate_type() {
|
||||||
|
let config = Config::default();
|
||||||
|
|
||||||
// Pubkey
|
// Pubkey
|
||||||
let pubkey = solana_sdk::pubkey::new_rand();
|
let pubkey = solana_sdk::pubkey::new_rand();
|
||||||
let addr = &pubkey as *const _ as u64;
|
|
||||||
let config = Config::default();
|
|
||||||
let memory_mapping = MemoryMapping::new(
|
let memory_mapping = MemoryMapping::new(
|
||||||
vec![MemoryRegion {
|
vec![MemoryRegion::new_readonly(bytes_of(&pubkey), 0x100000000)],
|
||||||
host_addr: addr,
|
|
||||||
vm_addr: 0x100000000,
|
|
||||||
vm_addr_end: 0x100000000_u64.saturating_add(std::mem::size_of::<Pubkey>() as u64),
|
|
||||||
len: std::mem::size_of::<Pubkey>() as u64,
|
|
||||||
vm_gap_shift: 63,
|
|
||||||
is_writable: false,
|
|
||||||
}],
|
|
||||||
&config,
|
&config,
|
||||||
)
|
)
|
||||||
.unwrap();
|
.unwrap();
|
||||||
|
@ -1940,48 +1937,27 @@ mod tests {
|
||||||
vec![AccountMeta::new(solana_sdk::pubkey::new_rand(), false)],
|
vec![AccountMeta::new(solana_sdk::pubkey::new_rand(), false)],
|
||||||
);
|
);
|
||||||
let instruction = StableInstruction::from(instruction);
|
let instruction = StableInstruction::from(instruction);
|
||||||
let addr = &instruction as *const _ as u64;
|
let memory_region = MemoryRegion::new_readonly(bytes_of(&instruction), 0x100000000);
|
||||||
let mut memory_region = MemoryRegion {
|
let memory_mapping = MemoryMapping::new(vec![memory_region], &config).unwrap();
|
||||||
host_addr: addr,
|
|
||||||
vm_addr: 0x100000000,
|
|
||||||
vm_addr_end: 0x100000000_u64.saturating_add(std::mem::size_of::<Instruction>() as u64),
|
|
||||||
len: std::mem::size_of::<StableInstruction>() as u64,
|
|
||||||
vm_gap_shift: 63,
|
|
||||||
is_writable: false,
|
|
||||||
};
|
|
||||||
let mut memory_mapping = MemoryMapping::new(vec![memory_region.clone()], &config).unwrap();
|
|
||||||
let translated_instruction =
|
let translated_instruction =
|
||||||
translate_type::<StableInstruction>(&memory_mapping, 0x100000000, true).unwrap();
|
translate_type::<StableInstruction>(&memory_mapping, 0x100000000, true).unwrap();
|
||||||
assert_eq!(instruction, *translated_instruction);
|
assert_eq!(instruction, *translated_instruction);
|
||||||
memory_region.len = 1;
|
|
||||||
let memory_region_index = memory_mapping
|
let memory_region = MemoryRegion::new_readonly(&bytes_of(&instruction)[..1], 0x100000000);
|
||||||
.get_regions()
|
let memory_mapping = MemoryMapping::new(vec![memory_region], &config).unwrap();
|
||||||
.iter()
|
assert!(translate_type::<Instruction>(&memory_mapping, 0x100000000, true).is_err());
|
||||||
.position(|memory_region| memory_region.vm_addr == 0x100000000)
|
|
||||||
.unwrap();
|
|
||||||
memory_mapping
|
|
||||||
.replace_region(memory_region_index, memory_region)
|
|
||||||
.unwrap();
|
|
||||||
assert!(translate_type::<StableInstruction>(&memory_mapping, 0x100000000, true).is_err());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn test_translate_slice() {
|
fn test_translate_slice() {
|
||||||
|
let config = Config::default();
|
||||||
|
|
||||||
// zero len
|
// zero len
|
||||||
let good_data = vec![1u8, 2, 3, 4, 5];
|
let good_data = vec![1u8, 2, 3, 4, 5];
|
||||||
let data: Vec<u8> = vec![];
|
let data: Vec<u8> = vec![];
|
||||||
assert_eq!(0x1 as *const u8, data.as_ptr());
|
assert_eq!(0x1 as *const u8, data.as_ptr());
|
||||||
let addr = good_data.as_ptr() as *const _ as u64;
|
|
||||||
let config = Config::default();
|
|
||||||
let memory_mapping = MemoryMapping::new(
|
let memory_mapping = MemoryMapping::new(
|
||||||
vec![MemoryRegion {
|
vec![MemoryRegion::new_readonly(&good_data, 0x100000000)],
|
||||||
host_addr: addr,
|
|
||||||
vm_addr: 0x100000000,
|
|
||||||
vm_addr_end: 0x100000000_u64.saturating_add(good_data.len() as u64),
|
|
||||||
len: good_data.len() as u64,
|
|
||||||
vm_gap_shift: 63,
|
|
||||||
is_writable: false,
|
|
||||||
}],
|
|
||||||
&config,
|
&config,
|
||||||
)
|
)
|
||||||
.unwrap();
|
.unwrap();
|
||||||
|
@ -1992,16 +1968,8 @@ mod tests {
|
||||||
|
|
||||||
// u8
|
// u8
|
||||||
let mut data = vec![1u8, 2, 3, 4, 5];
|
let mut data = vec![1u8, 2, 3, 4, 5];
|
||||||
let addr = data.as_ptr() as *const _ as u64;
|
|
||||||
let memory_mapping = MemoryMapping::new(
|
let memory_mapping = MemoryMapping::new(
|
||||||
vec![MemoryRegion {
|
vec![MemoryRegion::new_readonly(&data, 0x100000000)],
|
||||||
host_addr: addr,
|
|
||||||
vm_addr: 0x100000000,
|
|
||||||
vm_addr_end: 0x100000000_u64.saturating_add(data.len() as u64),
|
|
||||||
len: data.len() as u64,
|
|
||||||
vm_gap_shift: 63,
|
|
||||||
is_writable: false,
|
|
||||||
}],
|
|
||||||
&config,
|
&config,
|
||||||
)
|
)
|
||||||
.unwrap();
|
.unwrap();
|
||||||
|
@ -2027,16 +1995,11 @@ mod tests {
|
||||||
|
|
||||||
// u64
|
// u64
|
||||||
let mut data = vec![1u64, 2, 3, 4, 5];
|
let mut data = vec![1u64, 2, 3, 4, 5];
|
||||||
let addr = data.as_ptr() as *const _ as u64;
|
|
||||||
let memory_mapping = MemoryMapping::new(
|
let memory_mapping = MemoryMapping::new(
|
||||||
vec![MemoryRegion {
|
vec![MemoryRegion::new_readonly(
|
||||||
host_addr: addr,
|
bytes_of_slice(&data),
|
||||||
vm_addr: 0x100000000,
|
0x100000000,
|
||||||
vm_addr_end: 0x100000000_u64.saturating_add((data.len() * size_of::<u64>()) as u64),
|
)],
|
||||||
len: (data.len() * size_of::<u64>()) as u64,
|
|
||||||
vm_gap_shift: 63,
|
|
||||||
is_writable: false,
|
|
||||||
}],
|
|
||||||
&config,
|
&config,
|
||||||
)
|
)
|
||||||
.unwrap();
|
.unwrap();
|
||||||
|
@ -2052,17 +2015,13 @@ mod tests {
|
||||||
|
|
||||||
// Pubkeys
|
// Pubkeys
|
||||||
let mut data = vec![solana_sdk::pubkey::new_rand(); 5];
|
let mut data = vec![solana_sdk::pubkey::new_rand(); 5];
|
||||||
let addr = data.as_ptr() as *const _ as u64;
|
|
||||||
let memory_mapping = MemoryMapping::new(
|
let memory_mapping = MemoryMapping::new(
|
||||||
vec![MemoryRegion {
|
vec![MemoryRegion::new_readonly(
|
||||||
host_addr: addr,
|
unsafe {
|
||||||
vm_addr: 0x100000000,
|
slice::from_raw_parts(data.as_ptr() as *const u8, mem::size_of::<Pubkey>() * 5)
|
||||||
vm_addr_end: 0x100000000_u64
|
},
|
||||||
.saturating_add((data.len() * std::mem::size_of::<Pubkey>()) as u64),
|
0x100000000,
|
||||||
len: (data.len() * std::mem::size_of::<Pubkey>()) as u64,
|
)],
|
||||||
vm_gap_shift: 63,
|
|
||||||
is_writable: false,
|
|
||||||
}],
|
|
||||||
&config,
|
&config,
|
||||||
)
|
)
|
||||||
.unwrap();
|
.unwrap();
|
||||||
|
@ -2077,17 +2036,9 @@ mod tests {
|
||||||
#[test]
|
#[test]
|
||||||
fn test_translate_string_and_do() {
|
fn test_translate_string_and_do() {
|
||||||
let string = "Gaggablaghblagh!";
|
let string = "Gaggablaghblagh!";
|
||||||
let addr = string.as_ptr() as *const _ as u64;
|
|
||||||
let config = Config::default();
|
let config = Config::default();
|
||||||
let memory_mapping = MemoryMapping::new(
|
let memory_mapping = MemoryMapping::new(
|
||||||
vec![MemoryRegion {
|
vec![MemoryRegion::new_readonly(string.as_bytes(), 0x100000000)],
|
||||||
host_addr: addr,
|
|
||||||
vm_addr: 0x100000000,
|
|
||||||
len: string.len() as u64,
|
|
||||||
vm_addr_end: 0x100000000_u64.saturating_add(string.len() as u64),
|
|
||||||
vm_gap_shift: 63,
|
|
||||||
is_writable: false,
|
|
||||||
}],
|
|
||||||
&config,
|
&config,
|
||||||
)
|
)
|
||||||
.unwrap();
|
.unwrap();
|
||||||
|
@ -2144,17 +2095,9 @@ mod tests {
|
||||||
);
|
);
|
||||||
|
|
||||||
let string = "Gaggablaghblagh!";
|
let string = "Gaggablaghblagh!";
|
||||||
let addr = string.as_ptr() as *const _ as u64;
|
|
||||||
let config = Config::default();
|
let config = Config::default();
|
||||||
let mut memory_mapping = MemoryMapping::new(
|
let mut memory_mapping = MemoryMapping::new(
|
||||||
vec![MemoryRegion {
|
vec![MemoryRegion::new_readonly(string.as_bytes(), 0x100000000)],
|
||||||
host_addr: addr,
|
|
||||||
vm_addr: 0x100000000,
|
|
||||||
len: string.len() as u64,
|
|
||||||
vm_addr_end: 0x100000000_u64.saturating_add(string.len() as u64),
|
|
||||||
vm_gap_shift: 63,
|
|
||||||
is_writable: false,
|
|
||||||
}],
|
|
||||||
&config,
|
&config,
|
||||||
)
|
)
|
||||||
.unwrap();
|
.unwrap();
|
||||||
|
@ -2203,17 +2146,9 @@ mod tests {
|
||||||
);
|
);
|
||||||
|
|
||||||
let string = "Gaggablaghblagh!";
|
let string = "Gaggablaghblagh!";
|
||||||
let addr = string.as_ptr() as *const _ as u64;
|
|
||||||
let config = Config::default();
|
let config = Config::default();
|
||||||
let mut memory_mapping = MemoryMapping::new(
|
let mut memory_mapping = MemoryMapping::new(
|
||||||
vec![MemoryRegion {
|
vec![MemoryRegion::new_readonly(string.as_bytes(), 0x100000000)],
|
||||||
host_addr: addr,
|
|
||||||
vm_addr: 0x100000000,
|
|
||||||
len: string.len() as u64,
|
|
||||||
vm_addr_end: 0x100000000_u64.saturating_add(string.len() as u64),
|
|
||||||
vm_gap_shift: 63,
|
|
||||||
is_writable: false,
|
|
||||||
}],
|
|
||||||
&config,
|
&config,
|
||||||
)
|
)
|
||||||
.unwrap();
|
.unwrap();
|
||||||
|
@ -2331,17 +2266,9 @@ mod tests {
|
||||||
let cost = invoke_context.get_compute_budget().log_pubkey_units;
|
let cost = invoke_context.get_compute_budget().log_pubkey_units;
|
||||||
|
|
||||||
let pubkey = Pubkey::from_str("MoqiU1vryuCGQSxFKA1SZ316JdLEFFhoAu6cKUNk7dN").unwrap();
|
let pubkey = Pubkey::from_str("MoqiU1vryuCGQSxFKA1SZ316JdLEFFhoAu6cKUNk7dN").unwrap();
|
||||||
let addr = pubkey.as_ref().first().unwrap() as *const _ as u64;
|
|
||||||
let config = Config::default();
|
let config = Config::default();
|
||||||
let mut memory_mapping = MemoryMapping::new(
|
let mut memory_mapping = MemoryMapping::new(
|
||||||
vec![MemoryRegion {
|
vec![MemoryRegion::new_readonly(bytes_of(&pubkey), 0x100000000)],
|
||||||
host_addr: addr,
|
|
||||||
vm_addr: 0x100000000,
|
|
||||||
len: 32,
|
|
||||||
vm_addr_end: 0x100000000_u64.saturating_add(32),
|
|
||||||
vm_gap_shift: 63,
|
|
||||||
is_writable: false,
|
|
||||||
}],
|
|
||||||
&config,
|
&config,
|
||||||
)
|
)
|
||||||
.unwrap();
|
.unwrap();
|
||||||
|
@ -2657,44 +2584,16 @@ mod tests {
|
||||||
len: bytes2.len(),
|
len: bytes2.len(),
|
||||||
};
|
};
|
||||||
let bytes_to_hash = [mock_slice1, mock_slice2];
|
let bytes_to_hash = [mock_slice1, mock_slice2];
|
||||||
let hash_result = [0; HASH_BYTES];
|
let mut hash_result = [0; HASH_BYTES];
|
||||||
let ro_len = bytes_to_hash.len() as u64;
|
let ro_len = bytes_to_hash.len() as u64;
|
||||||
let ro_va = 0x100000000;
|
let ro_va = 0x100000000;
|
||||||
let rw_va = 0x200000000;
|
let rw_va = 0x200000000;
|
||||||
let mut memory_mapping = MemoryMapping::new(
|
let mut memory_mapping = MemoryMapping::new(
|
||||||
vec![
|
vec![
|
||||||
MemoryRegion {
|
MemoryRegion::new_readonly(bytes_of_slice(&bytes_to_hash), ro_va),
|
||||||
host_addr: bytes_to_hash.as_ptr() as *const _ as u64,
|
MemoryRegion::new_writable(bytes_of_slice_mut(&mut hash_result), rw_va),
|
||||||
vm_addr: ro_va,
|
MemoryRegion::new_readonly(bytes1.as_bytes(), bytes_to_hash[0].vm_addr),
|
||||||
len: 32,
|
MemoryRegion::new_readonly(bytes2.as_bytes(), bytes_to_hash[1].vm_addr),
|
||||||
vm_addr_end: ro_va.saturating_add(32),
|
|
||||||
vm_gap_shift: 63,
|
|
||||||
is_writable: false,
|
|
||||||
},
|
|
||||||
MemoryRegion {
|
|
||||||
host_addr: hash_result.as_ptr() as *const _ as u64,
|
|
||||||
vm_addr: rw_va,
|
|
||||||
len: HASH_BYTES as u64,
|
|
||||||
vm_addr_end: rw_va.saturating_add(HASH_BYTES as u64),
|
|
||||||
vm_gap_shift: 63,
|
|
||||||
is_writable: true,
|
|
||||||
},
|
|
||||||
MemoryRegion {
|
|
||||||
host_addr: bytes1.as_ptr() as *const _ as u64,
|
|
||||||
vm_addr: bytes_to_hash[0].vm_addr,
|
|
||||||
len: bytes1.len() as u64,
|
|
||||||
vm_addr_end: bytes_to_hash[0].vm_addr.saturating_add(bytes1.len() as u64),
|
|
||||||
vm_gap_shift: 63,
|
|
||||||
is_writable: false,
|
|
||||||
},
|
|
||||||
MemoryRegion {
|
|
||||||
host_addr: bytes2.as_ptr() as *const _ as u64,
|
|
||||||
vm_addr: bytes_to_hash[1].vm_addr,
|
|
||||||
len: bytes2.len() as u64,
|
|
||||||
vm_addr_end: bytes_to_hash[1].vm_addr.saturating_add(bytes2.len() as u64),
|
|
||||||
vm_gap_shift: 63,
|
|
||||||
is_writable: false,
|
|
||||||
},
|
|
||||||
],
|
],
|
||||||
&config,
|
&config,
|
||||||
)
|
)
|
||||||
|
@ -2807,22 +2706,8 @@ mod tests {
|
||||||
|
|
||||||
let mut memory_mapping = MemoryMapping::new(
|
let mut memory_mapping = MemoryMapping::new(
|
||||||
vec![
|
vec![
|
||||||
MemoryRegion {
|
MemoryRegion::new_readonly(&valid_bytes, valid_bytes_va),
|
||||||
host_addr: valid_bytes.as_ptr() as *const _ as u64,
|
MemoryRegion::new_readonly(&invalid_bytes, invalid_bytes_va),
|
||||||
vm_addr: valid_bytes_va,
|
|
||||||
len: 32,
|
|
||||||
vm_addr_end: valid_bytes_va.saturating_add(32),
|
|
||||||
vm_gap_shift: 63,
|
|
||||||
is_writable: false,
|
|
||||||
},
|
|
||||||
MemoryRegion {
|
|
||||||
host_addr: invalid_bytes.as_ptr() as *const _ as u64,
|
|
||||||
vm_addr: invalid_bytes_va,
|
|
||||||
len: 32,
|
|
||||||
vm_addr_end: invalid_bytes_va.saturating_add(32),
|
|
||||||
vm_gap_shift: 63,
|
|
||||||
is_writable: false,
|
|
||||||
},
|
|
||||||
],
|
],
|
||||||
&config,
|
&config,
|
||||||
)
|
)
|
||||||
|
@ -2906,22 +2791,8 @@ mod tests {
|
||||||
|
|
||||||
let mut memory_mapping = MemoryMapping::new(
|
let mut memory_mapping = MemoryMapping::new(
|
||||||
vec![
|
vec![
|
||||||
MemoryRegion {
|
MemoryRegion::new_readonly(&valid_bytes, valid_bytes_va),
|
||||||
host_addr: valid_bytes.as_ptr() as *const _ as u64,
|
MemoryRegion::new_readonly(&invalid_bytes, invalid_bytes_va),
|
||||||
vm_addr: valid_bytes_va,
|
|
||||||
len: 32,
|
|
||||||
vm_addr_end: valid_bytes_va.saturating_add(32),
|
|
||||||
vm_gap_shift: 63,
|
|
||||||
is_writable: false,
|
|
||||||
},
|
|
||||||
MemoryRegion {
|
|
||||||
host_addr: invalid_bytes.as_ptr() as *const _ as u64,
|
|
||||||
vm_addr: invalid_bytes_va,
|
|
||||||
len: 32,
|
|
||||||
vm_addr_end: invalid_bytes_va.saturating_add(32),
|
|
||||||
vm_gap_shift: 63,
|
|
||||||
is_writable: false,
|
|
||||||
},
|
|
||||||
],
|
],
|
||||||
&config,
|
&config,
|
||||||
)
|
)
|
||||||
|
@ -3013,51 +2884,16 @@ mod tests {
|
||||||
60, 87, 144, 161, 146, 42, 34, 91, 155, 158, 189, 121, 79,
|
60, 87, 144, 161, 146, 42, 34, 91, 155, 158, 189, 121, 79,
|
||||||
];
|
];
|
||||||
let invalid_point_va = 0x400000000;
|
let invalid_point_va = 0x400000000;
|
||||||
let result_point: [u8; 32] = [0; 32];
|
let mut result_point: [u8; 32] = [0; 32];
|
||||||
let result_point_va = 0x500000000;
|
let result_point_va = 0x500000000;
|
||||||
|
|
||||||
let mut memory_mapping = MemoryMapping::new(
|
let mut memory_mapping = MemoryMapping::new(
|
||||||
vec![
|
vec![
|
||||||
MemoryRegion {
|
MemoryRegion::new_readonly(bytes_of_slice(&left_point), left_point_va),
|
||||||
host_addr: left_point.as_ptr() as *const _ as u64,
|
MemoryRegion::new_readonly(bytes_of_slice(&right_point), right_point_va),
|
||||||
vm_addr: left_point_va,
|
MemoryRegion::new_readonly(bytes_of_slice(&scalar), scalar_va),
|
||||||
len: 32,
|
MemoryRegion::new_readonly(bytes_of_slice(&invalid_point), invalid_point_va),
|
||||||
vm_addr_end: left_point_va.saturating_add(32),
|
MemoryRegion::new_writable(bytes_of_slice_mut(&mut result_point), result_point_va),
|
||||||
vm_gap_shift: 63,
|
|
||||||
is_writable: false,
|
|
||||||
},
|
|
||||||
MemoryRegion {
|
|
||||||
host_addr: right_point.as_ptr() as *const _ as u64,
|
|
||||||
vm_addr: right_point_va,
|
|
||||||
len: 32,
|
|
||||||
vm_addr_end: right_point_va.saturating_add(32),
|
|
||||||
vm_gap_shift: 63,
|
|
||||||
is_writable: false,
|
|
||||||
},
|
|
||||||
MemoryRegion {
|
|
||||||
host_addr: scalar.as_ptr() as *const _ as u64,
|
|
||||||
vm_addr: scalar_va,
|
|
||||||
len: 32,
|
|
||||||
vm_addr_end: scalar_va.saturating_add(32),
|
|
||||||
vm_gap_shift: 63,
|
|
||||||
is_writable: false,
|
|
||||||
},
|
|
||||||
MemoryRegion {
|
|
||||||
host_addr: invalid_point.as_ptr() as *const _ as u64,
|
|
||||||
vm_addr: invalid_point_va,
|
|
||||||
len: 32,
|
|
||||||
vm_addr_end: invalid_point_va.saturating_add(32),
|
|
||||||
vm_gap_shift: 63,
|
|
||||||
is_writable: false,
|
|
||||||
},
|
|
||||||
MemoryRegion {
|
|
||||||
host_addr: result_point.as_ptr() as *const _ as u64,
|
|
||||||
vm_addr: result_point_va,
|
|
||||||
len: 32,
|
|
||||||
vm_addr_end: result_point_va.saturating_add(32),
|
|
||||||
vm_gap_shift: 63,
|
|
||||||
is_writable: true,
|
|
||||||
},
|
|
||||||
],
|
],
|
||||||
&config,
|
&config,
|
||||||
)
|
)
|
||||||
|
@ -3225,51 +3061,16 @@ mod tests {
|
||||||
60, 87, 144, 161, 146, 42, 34, 91, 155, 158, 189, 121, 79,
|
60, 87, 144, 161, 146, 42, 34, 91, 155, 158, 189, 121, 79,
|
||||||
];
|
];
|
||||||
let invalid_point_va = 0x400000000;
|
let invalid_point_va = 0x400000000;
|
||||||
let result_point: [u8; 32] = [0; 32];
|
let mut result_point: [u8; 32] = [0; 32];
|
||||||
let result_point_va = 0x500000000;
|
let result_point_va = 0x500000000;
|
||||||
|
|
||||||
let mut memory_mapping = MemoryMapping::new(
|
let mut memory_mapping = MemoryMapping::new(
|
||||||
vec![
|
vec![
|
||||||
MemoryRegion {
|
MemoryRegion::new_readonly(bytes_of_slice(&left_point), left_point_va),
|
||||||
host_addr: left_point.as_ptr() as *const _ as u64,
|
MemoryRegion::new_readonly(bytes_of_slice(&right_point), right_point_va),
|
||||||
vm_addr: left_point_va,
|
MemoryRegion::new_readonly(bytes_of_slice(&scalar), scalar_va),
|
||||||
len: 32,
|
MemoryRegion::new_readonly(bytes_of_slice(&invalid_point), invalid_point_va),
|
||||||
vm_addr_end: left_point_va.saturating_add(32),
|
MemoryRegion::new_writable(bytes_of_slice_mut(&mut result_point), result_point_va),
|
||||||
vm_gap_shift: 63,
|
|
||||||
is_writable: false,
|
|
||||||
},
|
|
||||||
MemoryRegion {
|
|
||||||
host_addr: right_point.as_ptr() as *const _ as u64,
|
|
||||||
vm_addr: right_point_va,
|
|
||||||
len: 32,
|
|
||||||
vm_addr_end: right_point_va.saturating_add(32),
|
|
||||||
vm_gap_shift: 63,
|
|
||||||
is_writable: false,
|
|
||||||
},
|
|
||||||
MemoryRegion {
|
|
||||||
host_addr: scalar.as_ptr() as *const _ as u64,
|
|
||||||
vm_addr: scalar_va,
|
|
||||||
len: 32,
|
|
||||||
vm_addr_end: scalar_va.saturating_add(32),
|
|
||||||
vm_gap_shift: 63,
|
|
||||||
is_writable: false,
|
|
||||||
},
|
|
||||||
MemoryRegion {
|
|
||||||
host_addr: invalid_point.as_ptr() as *const _ as u64,
|
|
||||||
vm_addr: invalid_point_va,
|
|
||||||
len: 32,
|
|
||||||
vm_addr_end: invalid_point_va.saturating_add(32),
|
|
||||||
vm_gap_shift: 63,
|
|
||||||
is_writable: false,
|
|
||||||
},
|
|
||||||
MemoryRegion {
|
|
||||||
host_addr: result_point.as_ptr() as *const _ as u64,
|
|
||||||
vm_addr: result_point_va,
|
|
||||||
len: 32,
|
|
||||||
vm_addr_end: result_point_va.saturating_add(32),
|
|
||||||
vm_gap_shift: 63,
|
|
||||||
is_writable: true,
|
|
||||||
},
|
|
||||||
],
|
],
|
||||||
&config,
|
&config,
|
||||||
)
|
)
|
||||||
|
@ -3453,43 +3254,15 @@ mod tests {
|
||||||
let ristretto_points = [ristretto_point_x, ristretto_point_y];
|
let ristretto_points = [ristretto_point_x, ristretto_point_y];
|
||||||
let ristretto_points_va = 0x300000000;
|
let ristretto_points_va = 0x300000000;
|
||||||
|
|
||||||
let result_point: [u8; 32] = [0; 32];
|
let mut result_point: [u8; 32] = [0; 32];
|
||||||
let result_point_va = 0x400000000;
|
let result_point_va = 0x400000000;
|
||||||
|
|
||||||
let mut memory_mapping = MemoryMapping::new(
|
let mut memory_mapping = MemoryMapping::new(
|
||||||
vec![
|
vec![
|
||||||
MemoryRegion {
|
MemoryRegion::new_readonly(bytes_of_slice(&scalars), scalars_va),
|
||||||
host_addr: scalars.as_ptr() as *const _ as u64,
|
MemoryRegion::new_readonly(bytes_of_slice(&edwards_points), edwards_points_va),
|
||||||
vm_addr: scalars_va,
|
MemoryRegion::new_readonly(bytes_of_slice(&ristretto_points), ristretto_points_va),
|
||||||
len: 64,
|
MemoryRegion::new_writable(bytes_of_slice_mut(&mut result_point), result_point_va),
|
||||||
vm_addr_end: scalars_va.saturating_add(64),
|
|
||||||
vm_gap_shift: 63,
|
|
||||||
is_writable: false,
|
|
||||||
},
|
|
||||||
MemoryRegion {
|
|
||||||
host_addr: edwards_points.as_ptr() as *const _ as u64,
|
|
||||||
vm_addr: edwards_points_va,
|
|
||||||
len: 64,
|
|
||||||
vm_addr_end: edwards_points_va.saturating_add(64),
|
|
||||||
vm_gap_shift: 63,
|
|
||||||
is_writable: false,
|
|
||||||
},
|
|
||||||
MemoryRegion {
|
|
||||||
host_addr: ristretto_points.as_ptr() as *const _ as u64,
|
|
||||||
vm_addr: ristretto_points_va,
|
|
||||||
len: 64,
|
|
||||||
vm_addr_end: ristretto_points_va.saturating_add(64),
|
|
||||||
vm_gap_shift: 63,
|
|
||||||
is_writable: false,
|
|
||||||
},
|
|
||||||
MemoryRegion {
|
|
||||||
host_addr: result_point.as_ptr() as *const _ as u64,
|
|
||||||
vm_addr: result_point_va,
|
|
||||||
len: 32,
|
|
||||||
vm_addr_end: result_point_va.saturating_add(32),
|
|
||||||
vm_gap_shift: 63,
|
|
||||||
is_writable: true,
|
|
||||||
},
|
|
||||||
],
|
],
|
||||||
&config,
|
&config,
|
||||||
)
|
)
|
||||||
|
@ -3615,18 +3388,14 @@ mod tests {
|
||||||
|
|
||||||
// Test clock sysvar
|
// Test clock sysvar
|
||||||
{
|
{
|
||||||
let got_clock = Clock::default();
|
let mut got_clock = Clock::default();
|
||||||
let got_clock_va = 0x100000000;
|
let got_clock_va = 0x100000000;
|
||||||
|
|
||||||
let mut memory_mapping = MemoryMapping::new(
|
let mut memory_mapping = MemoryMapping::new(
|
||||||
vec![MemoryRegion {
|
vec![MemoryRegion::new_writable(
|
||||||
host_addr: &got_clock as *const _ as u64,
|
bytes_of_mut(&mut got_clock),
|
||||||
vm_addr: got_clock_va,
|
got_clock_va,
|
||||||
len: size_of::<Clock>() as u64,
|
)],
|
||||||
vm_addr_end: got_clock_va.saturating_add(size_of::<Clock>() as u64),
|
|
||||||
vm_gap_shift: 63,
|
|
||||||
is_writable: true,
|
|
||||||
}],
|
|
||||||
&config,
|
&config,
|
||||||
)
|
)
|
||||||
.unwrap();
|
.unwrap();
|
||||||
|
@ -3656,19 +3425,14 @@ mod tests {
|
||||||
|
|
||||||
// Test epoch_schedule sysvar
|
// Test epoch_schedule sysvar
|
||||||
{
|
{
|
||||||
let got_epochschedule = EpochSchedule::default();
|
let mut got_epochschedule = EpochSchedule::default();
|
||||||
let got_epochschedule_va = 0x100000000;
|
let got_epochschedule_va = 0x100000000;
|
||||||
|
|
||||||
let mut memory_mapping = MemoryMapping::new(
|
let mut memory_mapping = MemoryMapping::new(
|
||||||
vec![MemoryRegion {
|
vec![MemoryRegion::new_writable(
|
||||||
host_addr: &got_epochschedule as *const _ as u64,
|
bytes_of_mut(&mut got_epochschedule),
|
||||||
vm_addr: got_epochschedule_va,
|
got_epochschedule_va,
|
||||||
len: size_of::<EpochSchedule>() as u64,
|
)],
|
||||||
vm_addr_end: got_epochschedule_va
|
|
||||||
.saturating_add(size_of::<EpochSchedule>() as u64),
|
|
||||||
vm_gap_shift: 63,
|
|
||||||
is_writable: true,
|
|
||||||
}],
|
|
||||||
&config,
|
&config,
|
||||||
)
|
)
|
||||||
.unwrap();
|
.unwrap();
|
||||||
|
@ -3699,18 +3463,14 @@ mod tests {
|
||||||
|
|
||||||
// Test fees sysvar
|
// Test fees sysvar
|
||||||
{
|
{
|
||||||
let got_fees = Fees::default();
|
let mut got_fees = Fees::default();
|
||||||
let got_fees_va = 0x100000000;
|
let got_fees_va = 0x100000000;
|
||||||
|
|
||||||
let mut memory_mapping = MemoryMapping::new(
|
let mut memory_mapping = MemoryMapping::new(
|
||||||
vec![MemoryRegion {
|
vec![MemoryRegion::new_writable(
|
||||||
host_addr: &got_fees as *const _ as u64,
|
bytes_of_mut(&mut got_fees),
|
||||||
vm_addr: got_fees_va,
|
got_fees_va,
|
||||||
len: size_of::<Fees>() as u64,
|
)],
|
||||||
vm_addr_end: got_fees_va.saturating_add(size_of::<Fees>() as u64),
|
|
||||||
vm_gap_shift: 63,
|
|
||||||
is_writable: true,
|
|
||||||
}],
|
|
||||||
&config,
|
&config,
|
||||||
)
|
)
|
||||||
.unwrap();
|
.unwrap();
|
||||||
|
@ -3736,18 +3496,14 @@ mod tests {
|
||||||
|
|
||||||
// Test rent sysvar
|
// Test rent sysvar
|
||||||
{
|
{
|
||||||
let got_rent = create_filled_type::<Rent>(true);
|
let mut got_rent = create_filled_type::<Rent>(true);
|
||||||
let got_rent_va = 0x100000000;
|
let got_rent_va = 0x100000000;
|
||||||
|
|
||||||
let mut memory_mapping = MemoryMapping::new(
|
let mut memory_mapping = MemoryMapping::new(
|
||||||
vec![MemoryRegion {
|
vec![MemoryRegion::new_writable(
|
||||||
host_addr: &got_rent as *const _ as u64,
|
bytes_of_mut(&mut got_rent),
|
||||||
vm_addr: got_rent_va,
|
got_rent_va,
|
||||||
len: size_of::<Rent>() as u64,
|
)],
|
||||||
vm_addr_end: got_rent_va.saturating_add(size_of::<Rent>() as u64),
|
|
||||||
vm_gap_shift: 63,
|
|
||||||
is_writable: true,
|
|
||||||
}],
|
|
||||||
&config,
|
&config,
|
||||||
)
|
)
|
||||||
.unwrap();
|
.unwrap();
|
||||||
|
@ -3788,45 +3544,15 @@ mod tests {
|
||||||
const SEED_VA: u64 = 0x500000000;
|
const SEED_VA: u64 = 0x500000000;
|
||||||
|
|
||||||
let config = Config::default();
|
let config = Config::default();
|
||||||
let address = Pubkey::default();
|
let mut address = Pubkey::default();
|
||||||
let bump_seed = 0;
|
let mut bump_seed = 0;
|
||||||
let mut mock_slices = Vec::with_capacity(seeds.len());
|
|
||||||
let mut regions = vec![
|
let mut regions = vec![
|
||||||
MemoryRegion {
|
MemoryRegion::new_readonly(bytes_of(program_id), PROGRAM_ID_VA),
|
||||||
host_addr: mock_slices.as_ptr() as u64,
|
MemoryRegion::new_writable(bytes_of_mut(&mut address), ADDRESS_VA),
|
||||||
vm_addr: SEEDS_VA,
|
MemoryRegion::new_writable(bytes_of_mut(&mut bump_seed), BUMP_SEED_VA),
|
||||||
len: (seeds.len().saturating_mul(size_of::<MockSlice>()) as u64),
|
|
||||||
vm_addr_end: SEEDS_VA
|
|
||||||
.saturating_add(seeds.len().saturating_mul(size_of::<MockSlice>()) as u64),
|
|
||||||
vm_gap_shift: 63,
|
|
||||||
is_writable: false,
|
|
||||||
},
|
|
||||||
MemoryRegion {
|
|
||||||
host_addr: program_id.as_ref().as_ptr() as u64,
|
|
||||||
vm_addr: PROGRAM_ID_VA,
|
|
||||||
len: 32,
|
|
||||||
vm_addr_end: PROGRAM_ID_VA.saturating_add(32),
|
|
||||||
vm_gap_shift: 63,
|
|
||||||
is_writable: false,
|
|
||||||
},
|
|
||||||
MemoryRegion {
|
|
||||||
host_addr: address.as_ref().as_ptr() as u64,
|
|
||||||
vm_addr: ADDRESS_VA,
|
|
||||||
len: 32,
|
|
||||||
vm_addr_end: ADDRESS_VA.saturating_add(32),
|
|
||||||
vm_gap_shift: 63,
|
|
||||||
is_writable: true,
|
|
||||||
},
|
|
||||||
MemoryRegion {
|
|
||||||
host_addr: &bump_seed as *const u8 as u64,
|
|
||||||
vm_addr: BUMP_SEED_VA,
|
|
||||||
len: 32,
|
|
||||||
vm_addr_end: BUMP_SEED_VA.saturating_add(32),
|
|
||||||
vm_gap_shift: 63,
|
|
||||||
is_writable: true,
|
|
||||||
},
|
|
||||||
];
|
];
|
||||||
|
|
||||||
|
let mut mock_slices = Vec::with_capacity(seeds.len());
|
||||||
for (i, seed) in seeds.iter().enumerate() {
|
for (i, seed) in seeds.iter().enumerate() {
|
||||||
let vm_addr = SEED_VA.saturating_add((i as u64).saturating_mul(0x100000000));
|
let vm_addr = SEED_VA.saturating_add((i as u64).saturating_mul(0x100000000));
|
||||||
let mock_slice = MockSlice {
|
let mock_slice = MockSlice {
|
||||||
|
@ -3834,15 +3560,12 @@ mod tests {
|
||||||
len: seed.len(),
|
len: seed.len(),
|
||||||
};
|
};
|
||||||
mock_slices.push(mock_slice);
|
mock_slices.push(mock_slice);
|
||||||
regions.push(MemoryRegion {
|
regions.push(MemoryRegion::new_readonly(bytes_of_slice(seed), vm_addr));
|
||||||
host_addr: seed.as_ptr() as u64,
|
|
||||||
vm_addr,
|
|
||||||
len: seed.len() as u64,
|
|
||||||
vm_addr_end: vm_addr.saturating_add(seed.len() as u64),
|
|
||||||
vm_gap_shift: 63,
|
|
||||||
is_writable: false,
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
regions.push(MemoryRegion::new_readonly(
|
||||||
|
bytes_of_slice(&mock_slices),
|
||||||
|
SEEDS_VA,
|
||||||
|
));
|
||||||
let mut memory_mapping = MemoryMapping::new(regions, &config).unwrap();
|
let mut memory_mapping = MemoryMapping::new(regions, &config).unwrap();
|
||||||
|
|
||||||
let mut result = ProgramResult::Ok(0);
|
let mut result = ProgramResult::Ok(0);
|
||||||
|
@ -4012,14 +3735,7 @@ mod tests {
|
||||||
let mut memory = [0u8; END_OFFSET];
|
let mut memory = [0u8; END_OFFSET];
|
||||||
let config = Config::default();
|
let config = Config::default();
|
||||||
let mut memory_mapping = MemoryMapping::new(
|
let mut memory_mapping = MemoryMapping::new(
|
||||||
vec![MemoryRegion {
|
vec![MemoryRegion::new_writable(&mut memory, VM_BASE_ADDRESS)],
|
||||||
host_addr: memory.as_mut_ptr() as u64,
|
|
||||||
vm_addr: VM_BASE_ADDRESS,
|
|
||||||
len: END_OFFSET as u64,
|
|
||||||
vm_addr_end: VM_BASE_ADDRESS.saturating_add(END_OFFSET as u64),
|
|
||||||
vm_gap_shift: 63,
|
|
||||||
is_writable: true,
|
|
||||||
}],
|
|
||||||
&config,
|
&config,
|
||||||
)
|
)
|
||||||
.unwrap();
|
.unwrap();
|
||||||
|
@ -4334,4 +4050,24 @@ mod tests {
|
||||||
fn test_check_type_assumptions() {
|
fn test_check_type_assumptions() {
|
||||||
check_type_assumptions();
|
check_type_assumptions();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn bytes_of<T>(val: &T) -> &[u8] {
|
||||||
|
let size = mem::size_of::<T>();
|
||||||
|
unsafe { slice::from_raw_parts(std::slice::from_ref(val).as_ptr().cast(), size) }
|
||||||
|
}
|
||||||
|
|
||||||
|
fn bytes_of_mut<T>(val: &mut T) -> &mut [u8] {
|
||||||
|
let size = mem::size_of::<T>();
|
||||||
|
unsafe { slice::from_raw_parts_mut(slice::from_mut(val).as_mut_ptr().cast(), size) }
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn bytes_of_slice<T>(val: &[T]) -> &[u8] {
|
||||||
|
let size = val.len().wrapping_mul(mem::size_of::<T>());
|
||||||
|
unsafe { slice::from_raw_parts(val.as_ptr().cast(), size) }
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn bytes_of_slice_mut<T>(val: &mut [T]) -> &mut [u8] {
|
||||||
|
let size = val.len().wrapping_mul(mem::size_of::<T>());
|
||||||
|
unsafe { slice::from_raw_parts_mut(val.as_mut_ptr().cast(), size) }
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -6325,9 +6325,9 @@ dependencies = [
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "solana_rbpf"
|
name = "solana_rbpf"
|
||||||
version = "0.2.39"
|
version = "0.2.40"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "199a2e8ced67a32b47695256509d5a62b39cb7340f52f0a5d8de99f4dc2aeb43"
|
checksum = "1a5735b8c9defc3723162321a61ef738d34168401eeef213f62a32809739b0f5"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"byteorder 1.4.3",
|
"byteorder 1.4.3",
|
||||||
"combine",
|
"combine",
|
||||||
|
|
|
@ -24,7 +24,7 @@ num-traits = "0.2"
|
||||||
rand = "0.7"
|
rand = "0.7"
|
||||||
serde = "1.0.112"
|
serde = "1.0.112"
|
||||||
serde_json = "1.0.56"
|
serde_json = "1.0.56"
|
||||||
solana_rbpf = "=0.2.39"
|
solana_rbpf = "=0.2.40"
|
||||||
solana-account-decoder = { path = "../../account-decoder", version = "=1.16.0" }
|
solana-account-decoder = { path = "../../account-decoder", version = "=1.16.0" }
|
||||||
solana-address-lookup-table-program = { path = "../../programs/address-lookup-table", version = "=1.16.0" }
|
solana-address-lookup-table-program = { path = "../../programs/address-lookup-table", version = "=1.16.0" }
|
||||||
solana-bpf-loader-program = { path = "../bpf_loader", version = "=1.16.0" }
|
solana-bpf-loader-program = { path = "../bpf_loader", version = "=1.16.0" }
|
||||||
|
|
|
@ -1,6 +1,8 @@
|
||||||
#![feature(test)]
|
#![feature(test)]
|
||||||
#![cfg(feature = "sbf_c")]
|
#![cfg(feature = "sbf_c")]
|
||||||
|
|
||||||
|
use {solana_rbpf::memory_region::MemoryState, std::slice};
|
||||||
|
|
||||||
extern crate test;
|
extern crate test;
|
||||||
#[macro_use]
|
#[macro_use]
|
||||||
extern crate solana_bpf_loader_program;
|
extern crate solana_bpf_loader_program;
|
||||||
|
@ -8,7 +10,7 @@ extern crate solana_bpf_loader_program;
|
||||||
use {
|
use {
|
||||||
byteorder::{ByteOrder, LittleEndian, WriteBytesExt},
|
byteorder::{ByteOrder, LittleEndian, WriteBytesExt},
|
||||||
solana_bpf_loader_program::{
|
solana_bpf_loader_program::{
|
||||||
create_vm, serialization::serialize_parameters, syscalls::create_loader,
|
create_ebpf_vm, create_vm, serialization::serialize_parameters, syscalls::create_loader,
|
||||||
},
|
},
|
||||||
solana_measure::measure::Measure,
|
solana_measure::measure::Measure,
|
||||||
solana_program_runtime::{
|
solana_program_runtime::{
|
||||||
|
@ -88,13 +90,16 @@ fn bench_program_alu(bencher: &mut Bencher) {
|
||||||
.unwrap();
|
.unwrap();
|
||||||
|
|
||||||
verified_executable.jit_compile().unwrap();
|
verified_executable.jit_compile().unwrap();
|
||||||
let mut vm = create_vm(
|
create_vm!(
|
||||||
|
vm,
|
||||||
&verified_executable,
|
&verified_executable,
|
||||||
|
stack,
|
||||||
|
heap,
|
||||||
vec![MemoryRegion::new_writable(&mut inner_iter, MM_INPUT_START)],
|
vec![MemoryRegion::new_writable(&mut inner_iter, MM_INPUT_START)],
|
||||||
vec![],
|
vec![],
|
||||||
invoke_context,
|
invoke_context
|
||||||
)
|
);
|
||||||
.unwrap();
|
let mut vm = vm.unwrap();
|
||||||
|
|
||||||
println!("Interpreted:");
|
println!("Interpreted:");
|
||||||
vm.env
|
vm.env
|
||||||
|
@ -188,17 +193,6 @@ fn bench_create_vm(bencher: &mut Bencher) {
|
||||||
const BUDGET: u64 = 200_000;
|
const BUDGET: u64 = 200_000;
|
||||||
invoke_context.mock_set_remaining(BUDGET);
|
invoke_context.mock_set_remaining(BUDGET);
|
||||||
|
|
||||||
// Serialize account data
|
|
||||||
let (_serialized, regions, account_lengths) = serialize_parameters(
|
|
||||||
invoke_context.transaction_context,
|
|
||||||
invoke_context
|
|
||||||
.transaction_context
|
|
||||||
.get_current_instruction_context()
|
|
||||||
.unwrap(),
|
|
||||||
true, // should_cap_ix_accounts
|
|
||||||
)
|
|
||||||
.unwrap();
|
|
||||||
|
|
||||||
let loader = create_loader(
|
let loader = create_loader(
|
||||||
&invoke_context.feature_set,
|
&invoke_context.feature_set,
|
||||||
&ComputeBudget::default(),
|
&ComputeBudget::default(),
|
||||||
|
@ -213,14 +207,28 @@ fn bench_create_vm(bencher: &mut Bencher) {
|
||||||
VerifiedExecutable::<RequisiteVerifier, InvokeContext>::from_executable(executable)
|
VerifiedExecutable::<RequisiteVerifier, InvokeContext>::from_executable(executable)
|
||||||
.unwrap();
|
.unwrap();
|
||||||
|
|
||||||
bencher.iter(|| {
|
// Serialize account data
|
||||||
let _ = create_vm(
|
let (_serialized, regions, account_lengths) = serialize_parameters(
|
||||||
&verified_executable,
|
invoke_context.transaction_context,
|
||||||
regions.clone(),
|
invoke_context
|
||||||
account_lengths.clone(),
|
.transaction_context
|
||||||
invoke_context,
|
.get_current_instruction_context()
|
||||||
|
.unwrap(),
|
||||||
|
true, // should_cap_ix_accounts
|
||||||
)
|
)
|
||||||
.unwrap();
|
.unwrap();
|
||||||
|
|
||||||
|
bencher.iter(|| {
|
||||||
|
create_vm!(
|
||||||
|
vm,
|
||||||
|
&verified_executable,
|
||||||
|
stack,
|
||||||
|
heap,
|
||||||
|
clone_regions(®ions),
|
||||||
|
account_lengths.clone(),
|
||||||
|
invoke_context
|
||||||
|
);
|
||||||
|
let _ = vm.unwrap();
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
@ -258,13 +266,16 @@ fn bench_instruction_count_tuner(_bencher: &mut Bencher) {
|
||||||
VerifiedExecutable::<RequisiteVerifier, InvokeContext>::from_executable(executable)
|
VerifiedExecutable::<RequisiteVerifier, InvokeContext>::from_executable(executable)
|
||||||
.unwrap();
|
.unwrap();
|
||||||
|
|
||||||
let mut vm = create_vm(
|
create_vm!(
|
||||||
|
vm,
|
||||||
&verified_executable,
|
&verified_executable,
|
||||||
|
stack,
|
||||||
|
heap,
|
||||||
regions,
|
regions,
|
||||||
account_lengths,
|
account_lengths,
|
||||||
invoke_context,
|
invoke_context
|
||||||
)
|
);
|
||||||
.unwrap();
|
let mut vm = vm.unwrap();
|
||||||
|
|
||||||
let mut measure = Measure::start("tune");
|
let mut measure = Measure::start("tune");
|
||||||
let (instructions, _result) = vm.execute_program(true);
|
let (instructions, _result) = vm.execute_program(true);
|
||||||
|
@ -283,3 +294,29 @@ fn bench_instruction_count_tuner(_bencher: &mut Bencher) {
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn clone_regions(regions: &[MemoryRegion]) -> Vec<MemoryRegion> {
|
||||||
|
unsafe {
|
||||||
|
regions
|
||||||
|
.iter()
|
||||||
|
.map(|region| match region.state.get() {
|
||||||
|
MemoryState::Readable => MemoryRegion::new_readonly(
|
||||||
|
slice::from_raw_parts(region.host_addr.get() as *const _, region.len as usize),
|
||||||
|
region.vm_addr,
|
||||||
|
),
|
||||||
|
MemoryState::Writable => MemoryRegion::new_writable(
|
||||||
|
slice::from_raw_parts_mut(
|
||||||
|
region.host_addr.get() as *mut _,
|
||||||
|
region.len as usize,
|
||||||
|
),
|
||||||
|
region.vm_addr,
|
||||||
|
),
|
||||||
|
MemoryState::Cow(id) => MemoryRegion::new_cow(
|
||||||
|
slice::from_raw_parts(region.host_addr.get() as *const _, region.len as usize),
|
||||||
|
region.vm_addr,
|
||||||
|
id,
|
||||||
|
),
|
||||||
|
})
|
||||||
|
.collect()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -54,7 +54,7 @@ use {
|
||||||
};
|
};
|
||||||
use {
|
use {
|
||||||
solana_bpf_loader_program::{
|
solana_bpf_loader_program::{
|
||||||
create_vm,
|
create_ebpf_vm, create_vm,
|
||||||
serialization::{deserialize_parameters, serialize_parameters},
|
serialization::{deserialize_parameters, serialize_parameters},
|
||||||
syscalls::create_loader,
|
syscalls::create_loader,
|
||||||
},
|
},
|
||||||
|
@ -134,13 +134,16 @@ fn run_program(name: &str) -> u64 {
|
||||||
.unwrap();
|
.unwrap();
|
||||||
|
|
||||||
{
|
{
|
||||||
let mut vm = create_vm(
|
create_vm!(
|
||||||
|
vm,
|
||||||
&verified_executable,
|
&verified_executable,
|
||||||
|
stack,
|
||||||
|
heap,
|
||||||
regions,
|
regions,
|
||||||
account_lengths.clone(),
|
account_lengths.clone(),
|
||||||
invoke_context,
|
invoke_context
|
||||||
)
|
);
|
||||||
.unwrap();
|
let mut vm = vm.unwrap();
|
||||||
let (compute_units_consumed, result) = vm.execute_program(i == 0);
|
let (compute_units_consumed, result) = vm.execute_program(i == 0);
|
||||||
assert_eq!(SUCCESS, result.unwrap());
|
assert_eq!(SUCCESS, result.unwrap());
|
||||||
if i == 1 {
|
if i == 1 {
|
||||||
|
|
|
@ -3,7 +3,7 @@ use {
|
||||||
serde::{Deserialize, Serialize},
|
serde::{Deserialize, Serialize},
|
||||||
serde_json::Result,
|
serde_json::Result,
|
||||||
solana_bpf_loader_program::{
|
solana_bpf_loader_program::{
|
||||||
create_vm, serialization::serialize_parameters, syscalls::create_loader,
|
create_ebpf_vm, create_vm, serialization::serialize_parameters, syscalls::create_loader,
|
||||||
},
|
},
|
||||||
solana_program_runtime::{
|
solana_program_runtime::{
|
||||||
compute_budget::ComputeBudget,
|
compute_budget::ComputeBudget,
|
||||||
|
@ -292,14 +292,16 @@ before execting it in the virtual machine.",
|
||||||
}
|
}
|
||||||
_ => {}
|
_ => {}
|
||||||
}
|
}
|
||||||
|
create_vm!(
|
||||||
let mut vm = create_vm(
|
vm,
|
||||||
&verified_executable,
|
&verified_executable,
|
||||||
|
stack,
|
||||||
|
heap,
|
||||||
regions,
|
regions,
|
||||||
account_lengths,
|
account_lengths,
|
||||||
&mut invoke_context,
|
&mut invoke_context
|
||||||
)
|
);
|
||||||
.unwrap();
|
let mut vm = vm.unwrap();
|
||||||
let start_time = Instant::now();
|
let start_time = Instant::now();
|
||||||
if matches.value_of("use").unwrap() == "debugger" {
|
if matches.value_of("use").unwrap() == "debugger" {
|
||||||
vm.debug_port = Some(matches.value_of("port").unwrap().parse::<u16>().unwrap());
|
vm.debug_port = Some(matches.value_of("port").unwrap().parse::<u16>().unwrap());
|
||||||
|
|
Loading…
Reference in New Issue