Skip to content
This repository has been archived by the owner on Mar 24, 2022. It is now read-only.

Pinned heap test #273

Closed
wants to merge 8 commits into from
Closed
Show file tree
Hide file tree
Changes from 7 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
29 changes: 23 additions & 6 deletions benchmarks/lucet-benchmarks/src/context.rs
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,8 @@ fn context_init(c: &mut Criterion) {

c.bench_function("context_init", move |b| {
b.iter(|| {
ContextHandle::create_and_init(&mut *stack, f as usize, &[]).unwrap();
ContextHandle::create_and_init(&mut *stack, f as usize, &[], std::ptr::null_mut())
.unwrap();
})
});
}
Expand All @@ -22,7 +23,13 @@ fn context_swap_return(c: &mut Criterion) {
b.iter_batched(
|| {
let mut stack = vec![0u64; 1024].into_boxed_slice();
let child = ContextHandle::create_and_init(&mut *stack, f as usize, &[]).unwrap();
let child = ContextHandle::create_and_init(
&mut *stack,
f as usize,
&[],
std::ptr::null_mut(),
)
.unwrap();
(stack, child)
},
|(stack, mut child)| unsafe {
Expand All @@ -44,8 +51,13 @@ fn context_init_swap_return(c: &mut Criterion) {
|| vec![0u64; 1024].into_boxed_slice(),
|mut stack| {
let mut parent = ContextHandle::new();
let mut child =
ContextHandle::create_and_init(&mut *stack, f as usize, &[]).unwrap();
let mut child = ContextHandle::create_and_init(
&mut *stack,
f as usize,
&[],
std::ptr::null_mut(),
)
.unwrap();
unsafe { Context::swap(&mut parent, &mut child) };
stack
},
Expand Down Expand Up @@ -332,8 +344,13 @@ fn context_init_swap_return_many_args(c: &mut Criterion) {
|| vec![0u64; 1024].into_boxed_slice(),
|mut stack| {
let mut parent = ContextHandle::new();
let mut child =
ContextHandle::create_and_init(&mut *stack, f as usize, &args).unwrap();
let mut child = ContextHandle::create_and_init(
&mut *stack,
f as usize,
&args,
std::ptr::null_mut(),
)
.unwrap();
unsafe { Context::swap(&mut parent, &mut child) };
stack
},
Expand Down
2 changes: 2 additions & 0 deletions lucet-module/src/module_data.rs
Original file line number Diff line number Diff line change
Expand Up @@ -59,6 +59,7 @@ pub struct ModuleFeatures {
pub lzcnt: bool,
pub popcnt: bool,
pub instruction_count: bool,
pub pinned_heap: bool,
_hidden: (),
}

Expand All @@ -75,6 +76,7 @@ impl ModuleFeatures {
lzcnt: false,
popcnt: false,
instruction_count: false,
pinned_heap: false,
_hidden: (),
}
}
Expand Down
2 changes: 2 additions & 0 deletions lucet-runtime/lucet-runtime-internals/src/alloc/tests.rs
Original file line number Diff line number Diff line change
Expand Up @@ -610,6 +610,7 @@ macro_rules! alloc_tests {
inst.alloc_mut().stack_u64_mut(),
heap_touching_child as usize,
&[Val::CPtr(heap_ptr)],
heap_ptr,
)
.expect("context init succeeds");
Context::swap(&mut parent, &mut child);
Expand Down Expand Up @@ -659,6 +660,7 @@ macro_rules! alloc_tests {
inst.alloc_mut().stack_u64_mut(),
stack_pattern_child as usize,
&[Val::CPtr(heap_ptr)],
heap_ptr,
)
.expect("context init succeeds");
Context::swap(&mut parent, &mut child);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -118,6 +118,7 @@ _lucet_context_swap:
mov %r14, (6*8)(%rdi)
mov %r15, (7*8)(%rdi)
mov %rsi, (8*8)(%rdi)
mov %r8, (9*8)(%rdi)

movdqu %xmm0, (10*8 + 0*16)(%rdi)
movdqu %xmm1, (10*8 + 1*16)(%rdi)
Expand All @@ -137,6 +138,7 @@ _lucet_context_swap:
mov (5*8)(%rsi), %r13
mov (6*8)(%rsi), %r14
mov (7*8)(%rsi), %r15
mov (9*8)(%rsi), %r8

movdqu (10*8 + 0*16)(%rsi), %xmm0
movdqu (10*8 + 1*16)(%rsi), %xmm1
Expand Down
14 changes: 13 additions & 1 deletion lucet-runtime/lucet-runtime-internals/src/context/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -35,6 +35,7 @@ pub(crate) struct GpRegs {
r14: u64,
r15: u64,
pub(crate) rsi: u64,
r8: u64,
}

impl GpRegs {
Expand All @@ -49,6 +50,7 @@ impl GpRegs {
r14: 0,
r15: 0,
rsi: 0,
r8: 0,
}
}
}
Expand Down Expand Up @@ -206,9 +208,10 @@ impl ContextHandle {
stack: &mut [u64],
fptr: usize,
args: &[Val],
heap: *mut core::ffi::c_void,
) -> Result<ContextHandle, Error> {
let mut child = ContextHandle::new();
Context::init(stack, &mut child, fptr, args)?;
Context::init(stack, &mut child, fptr, args, heap)?;
Ok(child)
}
}
Expand Down Expand Up @@ -301,6 +304,7 @@ impl Context {
/// &mut child,
/// entrypoint as usize,
/// &[Val::U64(120), Val::F32(3.14)],
/// std::ptr::null_mut(),
/// );
/// assert!(res.is_ok());
/// ```
Expand All @@ -324,6 +328,7 @@ impl Context {
/// &mut child,
/// entrypoint as usize,
/// &[Val::U64(120), Val::F32(3.14)],
/// std::ptr::null_mut(),
/// );
/// assert!(res.is_ok());
/// ```
Expand Down Expand Up @@ -365,6 +370,7 @@ impl Context {
child: &mut Context,
fptr: usize,
args: &[Val],
heap: *mut core::ffi::c_void,
) -> Result<(), Error> {
Context::init_with_callback(
stack,
Expand All @@ -373,6 +379,7 @@ impl Context {
ptr::null_mut(),
fptr,
args,
heap,
)
}

Expand All @@ -391,6 +398,7 @@ impl Context {
backstop_data: *mut Instance,
fptr: usize,
args: &[Val],
heap: *mut core::ffi::c_void,
) -> Result<(), Error> {
if !stack_is_aligned(stack) {
return Err(Error::UnalignedStack);
Expand Down Expand Up @@ -471,6 +479,9 @@ impl Context {

child.gpr.rbp = child as *const Context as u64;

// testing out heap pinning
child.gpr.r15 = heap as u64;

// Read the mask to be restored if we ever need to jump out of a signal handler. If this
// isn't possible, die.
signal::pthread_sigmask(
Expand Down Expand Up @@ -552,6 +563,7 @@ impl Context {
/// &mut child,
/// entrypoint as usize,
/// &[],
/// std::ptr::null_mut(),
/// ).unwrap();
///
/// unsafe { Context::swap(&mut parent, &mut child); }
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -55,6 +55,7 @@ macro_rules! init_and_swap {
&mut *$stack,
$fn as usize,
&[$( $args ),*],
std::ptr::null_mut(),
).unwrap()));

child_regs = child;
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -31,7 +31,12 @@ fn init_rejects_unaligned() {
let mut stack_unaligned = unsafe { slice::from_raw_parts_mut(ptr, len) };

// now we have the unaligned stack, let's make sure it blows up right
let res = ContextHandle::create_and_init(&mut stack_unaligned, dummy as usize, &[]);
let res = ContextHandle::create_and_init(
&mut stack_unaligned,
dummy as usize,
&[],
std::ptr::null_mut(),
);

if let Err(Error::UnalignedStack) = res {
assert!(true);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -51,6 +51,7 @@ macro_rules! init_and_swap {
&mut *$stack,
$fn as usize,
&[$( $args ),*],
std::ptr::null_mut(),
).unwrap();
CHILD = Some(child);

Expand Down
4 changes: 3 additions & 1 deletion lucet-runtime/lucet-runtime-internals/src/instance.rs
Original file line number Diff line number Diff line change
Expand Up @@ -843,7 +843,8 @@ impl Instance {

self.entrypoint = Some(func.ptr);

let mut args_with_vmctx = vec![Val::from(self.alloc.slot().heap)];
let heap = self.alloc.slot().heap;
let mut args_with_vmctx = vec![Val::from(heap)];
args_with_vmctx.extend_from_slice(args);

let self_ptr = self as *mut _;
Expand All @@ -854,6 +855,7 @@ impl Instance {
self_ptr,
func.ptr.as_usize(),
&args_with_vmctx,
heap,
)?;

// Set up the guest to set itself as terminable, then continue to
Expand Down
4 changes: 3 additions & 1 deletion lucet-runtime/lucet-runtime-tests/src/stack.rs
Original file line number Diff line number Diff line change
Expand Up @@ -157,7 +157,9 @@ macro_rules! stack_tests {
expect_stack_overflow(
// Same note as `expect_ok_locals64_481`
stack_testcase(64 - 4).expect("generate stack_testcase 64"),
481,
// TODO: pick some high enough number that this test overflows still
// cranelift changes made 480 the wrong magic number
591,
true,
);
}
Expand Down
1 change: 1 addition & 0 deletions lucet-spectest/src/script.rs
Original file line number Diff line number Diff line change
Expand Up @@ -74,6 +74,7 @@ impl ScriptEnv {
true,
&None,
false,
false,
)
.map_err(program_error)?;

Expand Down
5 changes: 5 additions & 0 deletions lucet-wasi-sdk/tests/lucetc.rs
Original file line number Diff line number Diff line change
Expand Up @@ -55,6 +55,7 @@ mod lucetc_tests {
false,
&Some(v),
false,
false,
)
.expect("compile empty");
let mdata = c.module_data().unwrap();
Expand Down Expand Up @@ -101,6 +102,7 @@ mod lucetc_tests {
false,
&Some(v),
false,
false,
)
.expect("compile c");
let mdata = c.module_data().unwrap();
Expand Down Expand Up @@ -133,6 +135,7 @@ mod lucetc_tests {
false,
&Some(v),
false,
false,
)
.expect("compile d");
let mdata = c.module_data().unwrap();
Expand Down Expand Up @@ -161,6 +164,7 @@ mod lucetc_tests {
false,
&Some(v),
false,
false,
)
.expect("compile c & d");
let mdata = c.module_data().unwrap();
Expand Down Expand Up @@ -213,6 +217,7 @@ mod lucetc_tests {
false,
&Some(v),
false,
false,
)
.expect("compile empty");
let mdata = c.module_data().unwrap();
Expand Down
6 changes: 6 additions & 0 deletions lucetc/lucetc/options.rs
Original file line number Diff line number Diff line change
Expand Up @@ -453,6 +453,12 @@ SSE3 but not AVX:
.takes_value(false)
.help("Instrument the produced binary to count the number of wasm operations the translated program executes")
)
.arg(
Arg::with_name("pinned_heap")
.long("--pinned-heap-reg")
.takes_value(false)
.help("This feature is not stable - it may be removed in the future! Pin a register to use as this module's heap base. Typically improves performance.")
)
.arg(
Arg::with_name("error_style")
.long("error-style")
Expand Down
18 changes: 17 additions & 1 deletion lucetc/src/compiler.rs
Original file line number Diff line number Diff line change
Expand Up @@ -56,6 +56,7 @@ pub struct Compiler<'a> {
count_instructions: bool,
module_translation_state: ModuleTranslationState,
canonicalize_nans: bool,
pinned_heap: bool,
}

impl<'a> Compiler<'a> {
Expand All @@ -69,8 +70,15 @@ impl<'a> Compiler<'a> {
count_instructions: bool,
validator: &Option<Validator>,
canonicalize_nans: bool,
pinned_heap: bool,
) -> Result<Self, Error> {
let isa = Self::target_isa(target.clone(), opt_level, &cpu_features, canonicalize_nans)?;
let isa = Self::target_isa(
target.clone(),
opt_level,
&cpu_features,
canonicalize_nans,
pinned_heap,
)?;

let frontend_config = isa.frontend_config();
let mut module_info = ModuleInfo::new(frontend_config.clone());
Expand Down Expand Up @@ -127,12 +135,14 @@ impl<'a> Compiler<'a> {
module_translation_state,
target,
canonicalize_nans,
pinned_heap,
})
}

pub fn module_features(&self) -> ModuleFeatures {
let mut mf: ModuleFeatures = (&self.cpu_features).into();
mf.instruction_count = self.count_instructions;
mf.pinned_heap = self.pinned_heap;
mf
}

Expand Down Expand Up @@ -239,6 +249,7 @@ impl<'a> Compiler<'a> {
self.opt_level,
&self.cpu_features,
self.canonicalize_nans,
self.pinned_heap,
)?,
))
}
Expand All @@ -248,6 +259,7 @@ impl<'a> Compiler<'a> {
opt_level: OptLevel,
cpu_features: &CpuFeatures,
canonicalize_nans: bool,
pinned_heap: bool,
) -> Result<Box<dyn TargetIsa>, Error> {
let mut flags_builder = settings::builder();
let isa_builder = cpu_features.isa_builder(target)?;
Expand All @@ -257,6 +269,10 @@ impl<'a> Compiler<'a> {
if canonicalize_nans {
flags_builder.enable("enable_nan_canonicalization").unwrap();
}
if pinned_heap {
flags_builder.enable("enable_pinned_reg").unwrap();
flags_builder.enable("use_pinned_reg_as_heap_base").unwrap();
}
Ok(isa_builder.finish(settings::Flags::new(flags_builder)))
}
}
Expand Down
Loading