Calc size ahead of time to alloc once (#12154)
This commit is contained in:
parent
c575f45e24
commit
fd47d38e59
|
@ -1,19 +1,31 @@
|
||||||
#![feature(test)]
|
#![feature(test)]
|
||||||
#![cfg(feature = "bpf_c")]
|
#![cfg(feature = "bpf_c")]
|
||||||
|
|
||||||
extern crate test;
|
extern crate test;
|
||||||
|
#[macro_use]
|
||||||
|
extern crate solana_bpf_loader_program;
|
||||||
|
|
||||||
use byteorder::{ByteOrder, LittleEndian, WriteBytesExt};
|
use byteorder::{ByteOrder, LittleEndian, WriteBytesExt};
|
||||||
|
use solana_bpf_loader_program::serialization::{deserialize_parameters, serialize_parameters};
|
||||||
use solana_rbpf::EbpfVm;
|
use solana_rbpf::EbpfVm;
|
||||||
|
use solana_runtime::{
|
||||||
|
bank::Bank,
|
||||||
|
bank_client::BankClient,
|
||||||
|
genesis_utils::{create_genesis_config, GenesisConfigInfo},
|
||||||
|
loader_utils::load_program,
|
||||||
|
};
|
||||||
use solana_sdk::{
|
use solana_sdk::{
|
||||||
account::Account,
|
account::{create_keyed_readonly_accounts, Account, KeyedAccount},
|
||||||
bpf_loader,
|
bpf_loader, bpf_loader_deprecated,
|
||||||
|
client::SyncClient,
|
||||||
|
entrypoint::SUCCESS,
|
||||||
entrypoint_native::{ComputeBudget, ComputeMeter, InvokeContext, Logger, ProcessInstruction},
|
entrypoint_native::{ComputeBudget, ComputeMeter, InvokeContext, Logger, ProcessInstruction},
|
||||||
instruction::{CompiledInstruction, InstructionError},
|
instruction::{AccountMeta, CompiledInstruction, Instruction, InstructionError},
|
||||||
message::Message,
|
message::Message,
|
||||||
pubkey::Pubkey,
|
pubkey::Pubkey,
|
||||||
|
signature::{Keypair, Signer},
|
||||||
};
|
};
|
||||||
use std::{cell::RefCell, rc::Rc};
|
use std::{cell::RefCell, env, fs::File, io::Read, mem, path::PathBuf, rc::Rc, sync::Arc};
|
||||||
use std::{env, fs::File, io::Read, mem, path::PathBuf};
|
|
||||||
use test::Bencher;
|
use test::Bencher;
|
||||||
|
|
||||||
/// BPF program file extension
|
/// BPF program file extension
|
||||||
|
@ -34,31 +46,33 @@ fn empty_check(_prog: &[u8]) -> Result<(), solana_bpf_loader_program::BPFError>
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
fn load_elf() -> Result<Vec<u8>, std::io::Error> {
|
fn load_elf(name: &str) -> Result<Vec<u8>, std::io::Error> {
|
||||||
let path = create_bpf_path("bench_alu");
|
let path = create_bpf_path(name);
|
||||||
let mut file = File::open(&path).expect(&format!("Unable to open {:?}", path));
|
let mut file = File::open(&path).expect(&format!("Unable to open {:?}", path));
|
||||||
let mut elf = Vec::new();
|
let mut elf = Vec::new();
|
||||||
file.read_to_end(&mut elf).unwrap();
|
file.read_to_end(&mut elf).unwrap();
|
||||||
Ok(elf)
|
Ok(elf)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn load_bpf_program(
|
||||||
|
bank_client: &BankClient,
|
||||||
|
loader_id: &Pubkey,
|
||||||
|
payer_keypair: &Keypair,
|
||||||
|
name: &str,
|
||||||
|
) -> Pubkey {
|
||||||
|
let path = create_bpf_path(name);
|
||||||
|
let mut file = File::open(path).unwrap();
|
||||||
|
let mut elf = Vec::new();
|
||||||
|
file.read_to_end(&mut elf).unwrap();
|
||||||
|
load_program(bank_client, payer_keypair, loader_id, elf)
|
||||||
|
}
|
||||||
|
|
||||||
const ARMSTRONG_LIMIT: u64 = 500;
|
const ARMSTRONG_LIMIT: u64 = 500;
|
||||||
const ARMSTRONG_EXPECTED: u64 = 5;
|
const ARMSTRONG_EXPECTED: u64 = 5;
|
||||||
|
|
||||||
#[bench]
|
|
||||||
fn bench_program_load_elf(bencher: &mut Bencher) {
|
|
||||||
let elf = load_elf().unwrap();
|
|
||||||
let mut vm = EbpfVm::<solana_bpf_loader_program::BPFError>::new(None).unwrap();
|
|
||||||
vm.set_verifier(empty_check).unwrap();
|
|
||||||
|
|
||||||
bencher.iter(|| {
|
|
||||||
vm.set_elf(&elf).unwrap();
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
#[bench]
|
#[bench]
|
||||||
fn bench_program_verify(bencher: &mut Bencher) {
|
fn bench_program_verify(bencher: &mut Bencher) {
|
||||||
let elf = load_elf().unwrap();
|
let elf = load_elf("bench_alu").unwrap();
|
||||||
let mut vm = EbpfVm::<solana_bpf_loader_program::BPFError>::new(None).unwrap();
|
let mut vm = EbpfVm::<solana_bpf_loader_program::BPFError>::new(None).unwrap();
|
||||||
vm.set_verifier(empty_check).unwrap();
|
vm.set_verifier(empty_check).unwrap();
|
||||||
vm.set_elf(&elf).unwrap();
|
vm.set_elf(&elf).unwrap();
|
||||||
|
@ -80,14 +94,14 @@ fn bench_program_alu(bencher: &mut Bencher) {
|
||||||
inner_iter.write_u64::<LittleEndian>(0).unwrap();
|
inner_iter.write_u64::<LittleEndian>(0).unwrap();
|
||||||
let mut invoke_context = MockInvokeContext::default();
|
let mut invoke_context = MockInvokeContext::default();
|
||||||
|
|
||||||
let elf = load_elf().unwrap();
|
let elf = load_elf("bench_alu").unwrap();
|
||||||
let (mut vm, _) =
|
let (mut vm, _) =
|
||||||
solana_bpf_loader_program::create_vm(&bpf_loader::id(), &elf, &[], &mut invoke_context)
|
solana_bpf_loader_program::create_vm(&bpf_loader::id(), &elf, &[], &mut invoke_context)
|
||||||
.unwrap();
|
.unwrap();
|
||||||
|
|
||||||
println!("Interpreted:");
|
println!("Interpreted:");
|
||||||
assert_eq!(
|
assert_eq!(
|
||||||
0, /*success*/
|
SUCCESS,
|
||||||
vm.execute_program(&mut inner_iter, &[], &[]).unwrap()
|
vm.execute_program(&mut inner_iter, &[], &[]).unwrap()
|
||||||
);
|
);
|
||||||
assert_eq!(ARMSTRONG_LIMIT, LittleEndian::read_u64(&inner_iter));
|
assert_eq!(ARMSTRONG_LIMIT, LittleEndian::read_u64(&inner_iter));
|
||||||
|
@ -135,6 +149,99 @@ fn bench_program_alu(bencher: &mut Bencher) {
|
||||||
// println!("{{ \"type\": \"bench\", \"name\": \"bench_program_alu_jit_to_native_mips\", \"median\": {:?}, \"deviation\": 0 }}", mips);
|
// println!("{{ \"type\": \"bench\", \"name\": \"bench_program_alu_jit_to_native_mips\", \"median\": {:?}, \"deviation\": 0 }}", mips);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[bench]
|
||||||
|
fn bench_program_execute_noop(bencher: &mut Bencher) {
|
||||||
|
// solana_logger::setup(); // TODO remove
|
||||||
|
|
||||||
|
let GenesisConfigInfo {
|
||||||
|
genesis_config,
|
||||||
|
mint_keypair,
|
||||||
|
..
|
||||||
|
} = create_genesis_config(50);
|
||||||
|
let mut bank = Bank::new(&genesis_config);
|
||||||
|
let (name, id, entrypoint) = solana_bpf_loader_program!();
|
||||||
|
bank.add_builtin_loader(&name, id, entrypoint);
|
||||||
|
let bank = Arc::new(bank);
|
||||||
|
let bank_client = BankClient::new_shared(&bank);
|
||||||
|
|
||||||
|
let invoke_program_id =
|
||||||
|
load_bpf_program(&bank_client, &bpf_loader::id(), &mint_keypair, "noop");
|
||||||
|
|
||||||
|
let mint_pubkey = mint_keypair.pubkey();
|
||||||
|
let account_metas = vec![AccountMeta::new(mint_pubkey, true)];
|
||||||
|
|
||||||
|
let instruction = Instruction::new(invoke_program_id, &[u8::MAX, 0, 0, 0], account_metas);
|
||||||
|
let message = Message::new(&[instruction], Some(&mint_pubkey));
|
||||||
|
|
||||||
|
bank_client
|
||||||
|
.send_and_confirm_message(&[&mint_keypair], message.clone())
|
||||||
|
.unwrap();
|
||||||
|
|
||||||
|
println!("start bench");
|
||||||
|
bencher.iter(|| {
|
||||||
|
bank.clear_signatures();
|
||||||
|
bank_client
|
||||||
|
.send_and_confirm_message(&[&mint_keypair], message.clone())
|
||||||
|
.unwrap();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
fn create_serialization_create_params() -> (Vec<u8>, Vec<(Pubkey, RefCell<Account>)>) {
|
||||||
|
let accounts = vec![
|
||||||
|
(
|
||||||
|
Pubkey::new_rand(),
|
||||||
|
RefCell::new(Account::new(0, 100, &Pubkey::new_rand())),
|
||||||
|
),
|
||||||
|
(
|
||||||
|
Pubkey::new_rand(),
|
||||||
|
RefCell::new(Account::new(0, 100, &Pubkey::new_rand())),
|
||||||
|
),
|
||||||
|
(
|
||||||
|
Pubkey::new_rand(),
|
||||||
|
RefCell::new(Account::new(0, 250, &Pubkey::new_rand())),
|
||||||
|
),
|
||||||
|
(
|
||||||
|
Pubkey::new_rand(),
|
||||||
|
RefCell::new(Account::new(0, 1000, &Pubkey::new_rand())),
|
||||||
|
),
|
||||||
|
];
|
||||||
|
(vec![0xee; 100], accounts)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[bench]
|
||||||
|
fn bench_serialization_aligned(bencher: &mut Bencher) {
|
||||||
|
let (data, accounts) = create_serialization_create_params();
|
||||||
|
let keyed_accounts = create_keyed_readonly_accounts(&accounts);
|
||||||
|
|
||||||
|
bencher.iter(|| {
|
||||||
|
let buffer = serialize_parameters(
|
||||||
|
&bpf_loader_deprecated::id(),
|
||||||
|
&Pubkey::new_rand(),
|
||||||
|
&keyed_accounts,
|
||||||
|
&data,
|
||||||
|
)
|
||||||
|
.unwrap();
|
||||||
|
deserialize_parameters(&bpf_loader_deprecated::id(), &keyed_accounts, &buffer).unwrap();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
#[bench]
|
||||||
|
fn bench_serialization_unaligned(bencher: &mut Bencher) {
|
||||||
|
let (data, accounts) = create_serialization_create_params();
|
||||||
|
let keyed_accounts = create_keyed_readonly_accounts(&accounts);
|
||||||
|
|
||||||
|
bencher.iter(|| {
|
||||||
|
let buffer = serialize_parameters(
|
||||||
|
&bpf_loader_deprecated::id(),
|
||||||
|
&Pubkey::new_rand(),
|
||||||
|
&keyed_accounts,
|
||||||
|
&data,
|
||||||
|
)
|
||||||
|
.unwrap();
|
||||||
|
deserialize_parameters(&bpf_loader_deprecated::id(), &keyed_accounts, &buffer).unwrap();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
#[derive(Debug, Default)]
|
#[derive(Debug, Default)]
|
||||||
pub struct MockInvokeContext {
|
pub struct MockInvokeContext {
|
||||||
key: Pubkey,
|
key: Pubkey,
|
||||||
|
|
|
@ -50,7 +50,28 @@ pub fn serialize_parameters_unaligned(
|
||||||
) -> Result<Vec<u8>, InstructionError> {
|
) -> Result<Vec<u8>, InstructionError> {
|
||||||
assert_eq!(32, size_of::<Pubkey>());
|
assert_eq!(32, size_of::<Pubkey>());
|
||||||
|
|
||||||
let mut v: Vec<u8> = Vec::new();
|
// Calculate size in order to alloc once
|
||||||
|
let mut size = size_of::<u64>();
|
||||||
|
for (i, keyed_account) in keyed_accounts.iter().enumerate() {
|
||||||
|
let (is_dup, _) = is_dup(&keyed_accounts[..i], keyed_account);
|
||||||
|
size += 1; // dup, signer, writable, executable
|
||||||
|
if !is_dup {
|
||||||
|
let data_len = keyed_account.data_len()?;
|
||||||
|
size += size_of::<Pubkey>() // key
|
||||||
|
+ size_of::<Pubkey>() // owner
|
||||||
|
+ size_of::<u64>() // lamports
|
||||||
|
+ size_of::<u64>() // data len
|
||||||
|
+ data_len
|
||||||
|
+ MAX_PERMITTED_DATA_INCREASE
|
||||||
|
+ (data_len as *const u8).align_offset(align_of::<u128>())
|
||||||
|
+ size_of::<u64>(); // rent epoch;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
size += size_of::<u64>() // data len
|
||||||
|
+ instruction_data.len()
|
||||||
|
+ size_of::<Pubkey>(); // program id;
|
||||||
|
let mut v: Vec<u8> = Vec::with_capacity(size);
|
||||||
|
|
||||||
v.write_u64::<LittleEndian>(keyed_accounts.len() as u64)
|
v.write_u64::<LittleEndian>(keyed_accounts.len() as u64)
|
||||||
.unwrap();
|
.unwrap();
|
||||||
for (i, keyed_account) in keyed_accounts.iter().enumerate() {
|
for (i, keyed_account) in keyed_accounts.iter().enumerate() {
|
||||||
|
@ -120,10 +141,31 @@ pub fn serialize_parameters_aligned(
|
||||||
) -> Result<Vec<u8>, InstructionError> {
|
) -> Result<Vec<u8>, InstructionError> {
|
||||||
assert_eq!(32, size_of::<Pubkey>());
|
assert_eq!(32, size_of::<Pubkey>());
|
||||||
|
|
||||||
let mut v: Vec<u8> = Vec::new();
|
// Calculate size in order to alloc once
|
||||||
|
let mut size = size_of::<u64>();
|
||||||
|
for (i, keyed_account) in keyed_accounts.iter().enumerate() {
|
||||||
|
let (is_dup, _) = is_dup(&keyed_accounts[..i], keyed_account);
|
||||||
|
size += 8; // dup, signer, writable, executable
|
||||||
|
if !is_dup {
|
||||||
|
let data_len = keyed_account.data_len()?;
|
||||||
|
size += size_of::<Pubkey>() // key
|
||||||
|
+ size_of::<Pubkey>() // owner
|
||||||
|
+ size_of::<u64>() // lamports
|
||||||
|
+ size_of::<u64>() // data len
|
||||||
|
+ data_len
|
||||||
|
+ MAX_PERMITTED_DATA_INCREASE
|
||||||
|
+ (data_len as *const u8).align_offset(align_of::<u128>())
|
||||||
|
+ size_of::<u64>(); // rent epoch;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
size += size_of::<u64>() // data len
|
||||||
|
+ instruction_data.len()
|
||||||
|
+ size_of::<Pubkey>(); // program id;
|
||||||
|
let mut v: Vec<u8> = Vec::with_capacity(size);
|
||||||
|
|
||||||
|
// Serialize into the buffer
|
||||||
v.write_u64::<LittleEndian>(keyed_accounts.len() as u64)
|
v.write_u64::<LittleEndian>(keyed_accounts.len() as u64)
|
||||||
.unwrap();
|
.unwrap();
|
||||||
|
|
||||||
if v.as_ptr().align_offset(align_of::<u128>()) != 0 {
|
if v.as_ptr().align_offset(align_of::<u128>()) != 0 {
|
||||||
panic!();
|
panic!();
|
||||||
}
|
}
|
||||||
|
@ -178,10 +220,10 @@ pub fn deserialize_parameters_aligned(
|
||||||
} else {
|
} else {
|
||||||
let mut account = keyed_account.try_account_ref_mut()?;
|
let mut account = keyed_account.try_account_ref_mut()?;
|
||||||
start += size_of::<u8>() // is_signer
|
start += size_of::<u8>() // is_signer
|
||||||
+ size_of::<u8>() // is_writable
|
+ size_of::<u8>() // is_writable
|
||||||
+ size_of::<u8>() // executable
|
+ size_of::<u8>() // executable
|
||||||
+ 4 // padding to 128-bit aligned
|
+ 4 // padding to 128-bit aligned
|
||||||
+ size_of::<Pubkey>(); // key
|
+ size_of::<Pubkey>(); // key
|
||||||
account.owner = Pubkey::new(&buffer[start..start + size_of::<Pubkey>()]);
|
account.owner = Pubkey::new(&buffer[start..start + size_of::<Pubkey>()]);
|
||||||
start += size_of::<Pubkey>(); // owner
|
start += size_of::<Pubkey>(); // owner
|
||||||
account.lamports = LittleEndian::read_u64(&buffer[start..]);
|
account.lamports = LittleEndian::read_u64(&buffer[start..]);
|
||||||
|
|
Loading…
Reference in New Issue