Add rust bpf allocator (#4426)

This commit is contained in:
Jack May
2019-05-24 16:21:42 -07:00
committed by GitHub
parent 8611b40074
commit bfa1c025fd
17 changed files with 401 additions and 49 deletions

View File

@ -21,7 +21,7 @@ walkdir = "2"
bincode = "1.1.4"
byteorder = "1.3.1"
elf = "0.0.10"
solana_rbpf = "=0.1.11"
solana_rbpf = "=0.1.12"
solana-bpfloader = { path = "../bpf_loader", version = "0.15.0" }
solana-logger = { path = "../../logger", version = "0.15.0" }
solana-runtime = { path = "../../runtime", version = "0.15.0" }

View File

@ -76,12 +76,12 @@ fn bench_program_alu(bencher: &mut Bencher) {
inner_iter.write_u64::<LittleEndian>(0).unwrap();
let elf = load_elf().unwrap();
let mut vm = solana_bpf_loader::create_vm(&elf).unwrap();
let (mut vm, _) = solana_bpf_loader::create_vm(&elf).unwrap();
println!("Interpreted:");
assert_eq!(
1, /*true*/
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!(
@ -90,7 +90,7 @@ fn bench_program_alu(bencher: &mut Bencher) {
);
bencher.iter(|| {
vm.execute_program(&mut inner_iter).unwrap();
vm.execute_program(&mut inner_iter, &[], &[]).unwrap();
});
let instructions = vm.get_last_instruction_count();
let summary = bencher.bench(|_bencher| {}).unwrap();

View File

@ -67,7 +67,7 @@ fn main() {
.expect("Unable to create BPF install directory")
.success());
let rust_programs = ["iter", "noop"];
let rust_programs = ["alloc", "iter", "noop"];
for program in rust_programs.iter() {
println!(
"cargo:warning=(not a warning) Building Rust-based BPF programs: solana_bpf_rust_{}",

3
programs/bpf/rust/alloc/.gitignore vendored Normal file
View File

@ -0,0 +1,3 @@
/target/
Cargo.lock

View File

@ -0,0 +1,22 @@
# Note: This crate must be built using build.sh
[package]
name = "solana-bpf-rust-alloc"
version = "0.15.0"
description = "Solana BPF alloc program written in Rust"
authors = ["Solana Maintainers <maintainers@solana.com>"]
repository = "https://github.com/solana-labs/solana"
license = "Apache-2.0"
homepage = "https://solana.com/"
edition = "2018"
[dependencies]
solana-sdk-bpf-utils = { path = "../../../../sdk/bpf/rust-utils", version = "0.15.0" }
[workspace]
members = []
[lib]
name = "solana_bpf_rust_alloc"
crate-type = ["cdylib"]

View File

@ -0,0 +1,7 @@
[dependencies.compiler_builtins]
path = "../../../../sdk/bpf/rust-bpf-sysroot/src/compiler-builtins"
[target.bpfel-unknown-unknown.dependencies]
alloc = { path = "../../../../sdk/bpf/rust-bpf-sysroot/src/liballoc" }

View File

@ -0,0 +1,98 @@
//! @brief Example Rust-based BPF program that prints out the parameters passed to it
#![no_std]
#[macro_use]
extern crate alloc;
extern crate solana_sdk_bpf_utils;
use solana_sdk_bpf_utils::log::*;
use core::alloc::Layout;
use core::mem;
#[no_mangle]
pub extern "C" fn entrypoint(_input: *mut u8) -> bool {
unsafe {
// Confirm large allocation fails
let layout = Layout::from_size_align(core::usize::MAX, mem::align_of::<u8>()).unwrap();
let ptr = alloc::alloc::alloc(layout);
if !ptr.is_null() {
sol_log("Error: Alloc of very larger buffer should fail");
panic!();
}
}
unsafe {
// Test modest allocation and deallocation
let layout = Layout::from_size_align(100, mem::align_of::<u8>()).unwrap();
let ptr = alloc::alloc::alloc(layout);
if ptr.is_null() {
sol_log("Error: Alloc of 100 bytes failed");
alloc::alloc::handle_alloc_error(layout);
}
alloc::alloc::dealloc(ptr, layout);
}
unsafe {
// Test allocated memory read and write
let layout = Layout::from_size_align(100, mem::align_of::<u8>()).unwrap();
let ptr = alloc::alloc::alloc(layout);
if ptr.is_null() {
sol_log("Error: Alloc of 100 bytes failed");
alloc::alloc::handle_alloc_error(layout);
}
let iter = 0..100; // This weirdness due to #issue $#4271
for (i, _) in iter.enumerate() {
*ptr.add(i) = i as u8;
}
let iter = 0..100; // This weirdness due to #issue $#4271
for (i, _) in iter.enumerate() {
assert_eq!(*ptr.add(i as usize), i as u8);
}
sol_log_64(0x3, 0, 0, 0, *ptr.add(42) as u64);
assert_eq!(*ptr.add(42), 42);
alloc::alloc::dealloc(ptr, layout);
}
// // TODO not supported for system or bump allocator
// unsafe {
// // Test alloc all bytes and one more (assumes heap size of 2048)
// let layout = Layout::from_size_align(2048, mem::align_of::<u8>()).unwrap();
// let ptr = alloc::alloc::alloc(layout);
// if ptr.is_null() {
// sol_log("Error: Alloc of 2048 bytes failed");
// alloc::alloc::handle_alloc_error(layout);
// }
// let layout = Layout::from_size_align(1, mem::align_of::<u8>()).unwrap();
// let ptr_fail = alloc::alloc::alloc(layout);
// if !ptr_fail.is_null() {
// sol_log("Error: Able to alloc 1 more then max");
// panic!();
// }
// alloc::alloc::dealloc(ptr, layout);
// }
// TODO gremlin causes crash only on CI
// {
// // Test allocated vector
// const ITERS: usize = 100;
// let ones = vec![1_u64; ITERS];
// let mut sum: u64 = 0;
// for (i, v) in ones.iter().enumerate() {
// sol_log_64(i as u64, 0, 0, 0, 0);
// sum += ones[i as usize];
// }
// sol_log_64(0x4, 0, 0, 0, sum);
// assert_eq!(sum, ITERS as u64);
// }
sol_log("Success");
true
}

View File

@ -107,6 +107,7 @@ mod bpf {
solana_logger::setup();
let programs = [
"solana_bpf_rust_alloc",
// Disable due to #4271 "solana_bpf_rust_iter",
"solana_bpf_rust_noop",
];