use super::env::get_emscripten_data; use super::process::abort_with_message; use libc::{c_int, c_void, memcpy, size_t}; use wasmer_runtime_core::{ units::{Pages, WASM_MAX_PAGES, WASM_MIN_PAGES, WASM_PAGE_SIZE}, vm::Ctx, }; /// emscripten: _emscripten_memcpy_big pub fn _emscripten_memcpy_big(ctx: &mut Ctx, dest: u32, src: u32, len: u32) -> u32 { debug!( "emscripten::_emscripten_memcpy_big {}, {}, {}", dest, src, len ); let dest_addr = emscripten_memory_pointer!(ctx.memory(0), dest) as *mut c_void; let src_addr = emscripten_memory_pointer!(ctx.memory(0), src) as *mut c_void; unsafe { memcpy(dest_addr, src_addr, len as size_t); } dest } /// emscripten: _emscripten_get_heap_size pub fn _emscripten_get_heap_size(ctx: &mut Ctx) -> u32 { trace!("emscripten::_emscripten_get_heap_size"); let result = ctx.memory(0).size().bytes().0 as u32; trace!("=> {}", result); result } // From emscripten implementation fn align_up(mut val: usize, multiple: usize) -> usize { if val % multiple > 0 { val += multiple - val % multiple; } val } /// emscripten: _emscripten_resize_heap /// Note: this function only allows growing the size of heap pub fn _emscripten_resize_heap(ctx: &mut Ctx, requested_size: u32) -> u32 { debug!("emscripten::_emscripten_resize_heap {}", requested_size); let current_memory_pages = ctx.memory(0).size(); let current_memory = current_memory_pages.bytes().0 as u32; // implementation from emscripten let mut new_size = usize::max(current_memory as usize, WASM_MIN_PAGES * WASM_PAGE_SIZE); while new_size < requested_size as usize { if new_size <= 0x2000_0000 { new_size = align_up(new_size * 2, WASM_PAGE_SIZE); } else { new_size = usize::min( align_up((3 * new_size + 0x8000_0000) / 4, WASM_PAGE_SIZE), WASM_PAGE_SIZE * WASM_MAX_PAGES, ); } } let amount_to_grow = (new_size - current_memory as usize) / WASM_PAGE_SIZE; if let Ok(_pages_allocated) = ctx.memory(0).grow(Pages(amount_to_grow as u32)) { debug!("{} pages allocated", _pages_allocated.0); 1 } else { 0 } } /// emscripten: sbrk pub fn sbrk(ctx: &mut Ctx, increment: i32) -> i32 { debug!("emscripten::sbrk"); // let old_dynamic_top = 0; // let new_dynamic_top = 0; let globals = get_emscripten_data(ctx).globals; let dynamictop_ptr = (globals.dynamictop_ptr) as usize; let old_dynamic_top = ctx.memory(0).view::()[dynamictop_ptr].get() as i32; let new_dynamic_top: i32 = old_dynamic_top + increment; let total_memory = _emscripten_get_heap_size(ctx) as i32; debug!( " => PTR {}, old: {}, new: {}, increment: {}, total: {}", dynamictop_ptr, old_dynamic_top, new_dynamic_top, increment, total_memory ); if increment > 0 && new_dynamic_top < old_dynamic_top || new_dynamic_top < 0 { abort_on_cannot_grow_memory_old(ctx); return -1; } if new_dynamic_top > total_memory { let resized = _emscripten_resize_heap(ctx, new_dynamic_top as u32); if resized == 0 { return -1; } } ctx.memory(0).view::()[dynamictop_ptr].set(new_dynamic_top as u32); return old_dynamic_top as _; } /// emscripten: getTotalMemory pub fn get_total_memory(_ctx: &mut Ctx) -> u32 { debug!("emscripten::get_total_memory"); // instance.memories[0].current_pages() // TODO: Fix implementation _ctx.memory(0).size().bytes().0 as u32 } /// emscripten: enlargeMemory pub fn enlarge_memory(_ctx: &mut Ctx) -> u32 { debug!("emscripten::enlarge_memory"); // instance.memories[0].grow(100); // TODO: Fix implementation 0 } /// emscripten: abortOnCannotGrowMemory pub fn abort_on_cannot_grow_memory(ctx: &mut Ctx, _requested_size: u32) -> u32 { debug!( "emscripten::abort_on_cannot_grow_memory {}", _requested_size ); abort_with_message(ctx, "Cannot enlarge memory arrays!"); 0 } /// emscripten: abortOnCannotGrowMemory pub fn abort_on_cannot_grow_memory_old(ctx: &mut Ctx) -> u32 { debug!("emscripten::abort_on_cannot_grow_memory"); abort_with_message(ctx, "Cannot enlarge memory arrays!"); 0 } /// emscripten: segfault pub fn segfault(ctx: &mut Ctx) { debug!("emscripten::segfault"); abort_with_message(ctx, "segmentation fault"); } /// emscripten: alignfault pub fn alignfault(ctx: &mut Ctx) { debug!("emscripten::alignfault"); abort_with_message(ctx, "alignment fault"); } /// emscripten: ftfault pub fn ftfault(ctx: &mut Ctx) { debug!("emscripten::ftfault"); abort_with_message(ctx, "Function table mask error"); } /// emscripten: ___map_file pub fn ___map_file(_ctx: &mut Ctx, _one: u32, _two: u32) -> c_int { debug!("emscripten::___map_file"); // NOTE: TODO: Em returns -1 here as well. May need to implement properly -1 }