wasmer/src/apis/emscripten/mod.rs

442 lines
13 KiB
Rust
Raw Normal View History

2018-11-24 14:55:21 +00:00
/// NOTE: TODO: These emscripten api implementation only support wasm32 for now because they assume offsets are u32
use crate::webassembly::{ImportObject, ImportValue, LinearMemory};
use byteorder::{ByteOrder, LittleEndian};
use std::mem;
2018-11-20 19:11:58 +00:00
// EMSCRIPTEN APIS
2018-11-21 23:10:03 +00:00
mod env;
2018-11-22 04:59:23 +00:00
mod io;
2018-11-20 19:11:58 +00:00
mod memory;
mod process;
2018-11-21 23:10:03 +00:00
mod syscalls;
2018-11-24 14:55:21 +00:00
mod lock;
2018-11-22 04:59:23 +00:00
mod utils;
2018-11-23 05:13:01 +00:00
mod varargs;
2018-11-24 14:55:21 +00:00
mod errno;
mod storage;
mod nullfunc;
2018-11-26 20:02:29 +00:00
mod time;
2018-11-20 19:11:58 +00:00
pub use self::utils::is_emscripten_module;
2018-11-24 14:55:21 +00:00
pub use self::storage::{align_memory, static_alloc};
// TODO: Magic number - how is this calculated?
const TOTAL_STACK: u32 = 5242880;
// TODO: Magic number stolen from the generated JS - how is this calculated?
const DYNAMICTOP_PTR_DIFF: u32 = 1088;
const STATIC_BUMP: u32 = 215536; // TODO: make this variable
fn stacktop(static_bump: u32) -> u32 {
align_memory(dynamictop_ptr(static_bump) + 4)
}
fn stack_max(static_bump: u32) -> u32 {
stacktop(static_bump) + TOTAL_STACK
}
fn dynamic_base(static_bump: u32) -> u32 {
align_memory(stack_max(static_bump))
}
fn dynamictop_ptr(static_bump: u32) -> u32 {
static_bump + DYNAMICTOP_PTR_DIFF
}
// fn static_alloc(size: usize, static_top: &mut size) -> usize {
// let ret = *static_top;
// *static_top = (*static_top + size + 15) & (-16 as usize);
// ret
// }
pub fn emscripten_set_up_memory(memory: &mut LinearMemory) {
let dynamictop_ptr = dynamictop_ptr(STATIC_BUMP) as usize;
let mem = &mut memory[dynamictop_ptr..dynamictop_ptr+mem::size_of::<u32>()];
LittleEndian::write_u32(mem, dynamic_base(STATIC_BUMP));
}
macro_rules! mock_external {
($import:ident, $name:ident) => {{
2018-11-26 21:06:27 +00:00
extern fn _mocked_fn() -> i32 {
debug!("emscripten::{} <mock>", stringify!($name));
2018-11-26 21:42:44 +00:00
0
}
$import.set(
"env",
stringify!($name),
ImportValue::Func(_mocked_fn as _),
);
}};
}
pub fn generate_emscripten_env<'a, 'b>() -> ImportObject<&'a str, &'b str> {
let mut import_object = ImportObject::new();
2018-11-24 14:55:21 +00:00
// Global
import_object.set(
"env",
"global1",
ImportValue::Global(24), // TODO
);
import_object.set(
"env",
"global2",
ImportValue::Global(50), // TODO
);
import_object.set(
"env",
"global3",
ImportValue::Global(67), // TODO
);
import_object.set(
"env",
"STACKTOP",
ImportValue::Global(stacktop(STATIC_BUMP) as _),
);
import_object.set(
"env",
"STACK_MAX",
ImportValue::Global(stack_max(STATIC_BUMP) as _),
);
import_object.set(
"env",
"DYNAMICTOP_PTR",
ImportValue::Global(dynamictop_ptr(STATIC_BUMP) as _),
);
import_object.set(
"env",
"tableBase",
ImportValue::Global(0),
);
2018-11-24 14:55:21 +00:00
// Print functions
2018-11-20 19:11:58 +00:00
import_object.set("env", "printf", ImportValue::Func(io::printf as *const u8));
2018-11-22 04:59:23 +00:00
import_object.set(
"env",
"putchar",
ImportValue::Func(io::putchar as *const u8),
);
2018-11-24 14:55:21 +00:00
// Lock
import_object.set(
"env",
"___lock",
ImportValue::Func(lock::___lock as *const u8),
);
import_object.set(
"env",
"___unlock",
ImportValue::Func(lock::___unlock as *const u8),
);
// Env
2018-11-22 04:59:23 +00:00
import_object.set(
"env",
"_getenv",
ImportValue::Func(env::_getenv as *const u8),
);
2018-11-26 21:15:49 +00:00
import_object.set(
"env",
"_getpwnam",
ImportValue::Func(env::_getpwnam as _),
);
import_object.set(
"env",
"_getgrnam",
ImportValue::Func(env::_getgrnam as _),
);
2018-11-24 14:55:21 +00:00
// Errno
import_object.set(
"env",
"___setErrNo",
ImportValue::Func(errno::___seterrno as *const u8),
);
// Syscalls
import_object.set(
"env",
"___syscall1",
ImportValue::Func(syscalls::___syscall1 as *const u8),
);
2018-11-22 04:59:23 +00:00
import_object.set(
"env",
"___syscall3",
ImportValue::Func(syscalls::___syscall3 as *const u8),
);
import_object.set(
"env",
"___syscall4",
ImportValue::Func(syscalls::___syscall4 as *const u8),
);
import_object.set(
"env",
"___syscall5",
ImportValue::Func(syscalls::___syscall5 as *const u8),
);
2018-11-24 14:55:21 +00:00
import_object.set(
"env",
"___syscall6",
ImportValue::Func(syscalls::___syscall6 as *const u8),
);
2018-11-22 04:59:23 +00:00
import_object.set(
"env",
"___syscall54",
ImportValue::Func(syscalls::___syscall54 as *const u8),
);
import_object.set(
"env",
2018-11-24 14:55:21 +00:00
"___syscall140",
ImportValue::Func(syscalls::___syscall140 as *const u8),
2018-11-22 04:59:23 +00:00
);
2018-11-24 14:55:21 +00:00
import_object.set(
"env",
"___syscall145",
ImportValue::Func(syscalls::___syscall145 as *const u8),
);
import_object.set(
"env",
"___syscall146",
ImportValue::Func(syscalls::___syscall146 as *const u8),
);
import_object.set(
"env",
"___syscall221",
ImportValue::Func(syscalls::___syscall221 as *const u8),
);
import_object.set(
"env",
"___syscall20",
ImportValue::Func(syscalls::___syscall20 as *const u8),
);
import_object.set(
"env",
"___syscall64",
ImportValue::Func(syscalls::___syscall64 as *const u8),
);
import_object.set(
"env",
"___syscall122",
ImportValue::Func(syscalls::___syscall122 as *const u8),
);
2018-11-24 14:55:21 +00:00
// Process
2018-11-22 04:59:23 +00:00
import_object.set(
"env",
"abort",
ImportValue::Func(process::em_abort as *const u8),
);
import_object.set(
"env",
"_abort",
ImportValue::Func(process::_abort as *const u8),
);
2018-11-24 14:55:21 +00:00
import_object.set(
"env",
"abortStackOverflow",
ImportValue::Func(process::abort_stack_overflow as *const u8),
);
// Memory
2018-11-22 04:59:23 +00:00
import_object.set(
"env",
"abortOnCannotGrowMemory",
2018-11-24 14:55:21 +00:00
ImportValue::Func(memory::abort_on_cannot_grow_memory as *const u8),
2018-11-22 04:59:23 +00:00
);
import_object.set(
"env",
"_emscripten_memcpy_big",
ImportValue::Func(memory::_emscripten_memcpy_big as *const u8),
);
import_object.set(
"env",
"enlargeMemory",
ImportValue::Func(memory::enlarge_memory as *const u8),
);
import_object.set(
"env",
"getTotalMemory",
ImportValue::Func(memory::get_total_memory as *const u8),
);
2018-11-24 14:55:21 +00:00
// NullFuncs
import_object.set(
"env",
"nullFunc_ii",
ImportValue::Func(nullfunc::nullfunc_ii as *const u8),
);
import_object.set(
"env",
2018-11-24 17:31:38 +00:00
"nullFunc_iii",
ImportValue::Func(nullfunc::nullfunc_iii as *const u8),
2018-11-24 14:55:21 +00:00
);
import_object.set(
"env",
"nullFunc_iiii",
ImportValue::Func(nullfunc::nullfunc_iiii as *const u8),
);
import_object.set(
"env",
"nullFunc_iiiii",
ImportValue::Func(nullfunc::nullfunc_iiiii as *const u8),
);
import_object.set(
"env",
"nullFunc_iiiiii",
ImportValue::Func(nullfunc::nullfunc_iiiiii as *const u8),
);
import_object.set(
"env",
"nullFunc_vi",
ImportValue::Func(nullfunc::nullfunc_vi as *const u8),
);
import_object.set(
"env",
"nullFunc_vii",
ImportValue::Func(nullfunc::nullfunc_vii as *const u8),
);
import_object.set(
"env",
"nullFunc_viii",
ImportValue::Func(nullfunc::nullfunc_viii as *const u8),
);
import_object.set(
"env",
"nullFunc_viiii",
ImportValue::Func(nullfunc::nullfunc_viiii as *const u8),
);
2018-11-26 21:15:49 +00:00
// Time
2018-11-26 06:17:33 +00:00
import_object.set(
"env",
2018-11-26 21:15:49 +00:00
"_gettimeofday",
ImportValue::Func(time::_gettimeofday as _),
2018-11-26 06:17:33 +00:00
);
import_object.set(
"env",
2018-11-26 21:15:49 +00:00
"_clock_gettime",
ImportValue::Func(time::_clock_gettime as _),
);
2018-11-26 20:02:29 +00:00
import_object.set(
"env",
2018-11-26 21:15:49 +00:00
"_localtime",
ImportValue::Func(time::_localtime as _),
2018-11-26 20:02:29 +00:00
);
import_object.set(
"env",
2018-11-26 21:15:49 +00:00
"_time",
ImportValue::Func(time::_time as _),
);
2018-11-26 20:42:47 +00:00
import_object.set(
"env",
"_localtime_r",
ImportValue::Func(env::_localtime_r as _),
);
import_object.set(
"env",
"_getpagesize",
ImportValue::Func(env::_getpagesize as _),
);
mock_external!(import_object, _waitpid);
mock_external!(import_object, _utimes);
mock_external!(import_object, _usleep);
2018-11-26 21:15:49 +00:00
// mock_external!(import_object, _time);
mock_external!(import_object, _sysconf);
mock_external!(import_object, _strftime);
mock_external!(import_object, _sigsuspend);
mock_external!(import_object, _sigprocmask);
mock_external!(import_object, _sigemptyset);
mock_external!(import_object, _sigaddset);
mock_external!(import_object, _sigaction);
mock_external!(import_object, _setitimer);
mock_external!(import_object, _setgroups);
mock_external!(import_object, _setgrent);
mock_external!(import_object, _sem_wait);
mock_external!(import_object, _sem_post);
mock_external!(import_object, _sem_init);
mock_external!(import_object, _sched_yield);
mock_external!(import_object, _raise);
mock_external!(import_object, _mktime);
2018-11-26 20:42:47 +00:00
// mock_external!(import_object, _localtime_r);
2018-11-26 21:51:24 +00:00
// mock_external!(import_object, _localtime);
mock_external!(import_object, _llvm_stacksave);
mock_external!(import_object, _llvm_stackrestore);
mock_external!(import_object, _kill);
mock_external!(import_object, _gmtime_r);
2018-11-26 20:02:29 +00:00
// mock_external!(import_object, _gettimeofday);
2018-11-26 20:42:47 +00:00
// mock_external!(import_object, _getpagesize);
mock_external!(import_object, _getgrent);
mock_external!(import_object, _getaddrinfo);
mock_external!(import_object, _fork);
mock_external!(import_object, _exit);
mock_external!(import_object, _execve);
mock_external!(import_object, _endgrent);
// mock_external!(import_object, _clock_gettime);
mock_external!(import_object, ___syscall97);
mock_external!(import_object, ___syscall91);
mock_external!(import_object, ___syscall85);
mock_external!(import_object, ___syscall75);
mock_external!(import_object, ___syscall66);
// mock_external!(import_object, ___syscall64);
mock_external!(import_object, ___syscall63);
mock_external!(import_object, ___syscall60);
mock_external!(import_object, ___syscall54);
mock_external!(import_object, ___syscall39);
mock_external!(import_object, ___syscall38);
mock_external!(import_object, ___syscall340);
mock_external!(import_object, ___syscall334);
mock_external!(import_object, ___syscall300);
mock_external!(import_object, ___syscall295);
mock_external!(import_object, ___syscall272);
mock_external!(import_object, ___syscall268);
mock_external!(import_object, ___syscall221);
mock_external!(import_object, ___syscall220);
mock_external!(import_object, ___syscall212);
mock_external!(import_object, ___syscall201);
mock_external!(import_object, ___syscall199);
mock_external!(import_object, ___syscall197);
mock_external!(import_object, ___syscall196);
mock_external!(import_object, ___syscall195);
mock_external!(import_object, ___syscall194);
mock_external!(import_object, ___syscall191);
mock_external!(import_object, ___syscall181);
mock_external!(import_object, ___syscall180);
mock_external!(import_object, ___syscall168);
// mock_external!(import_object, ___syscall146);
// mock_external!(import_object, ___syscall145);
mock_external!(import_object, ___syscall142);
mock_external!(import_object, ___syscall140);
// mock_external!(import_object, ___syscall122);
mock_external!(import_object, ___syscall102);
// mock_external!(import_object, ___syscall20);
mock_external!(import_object, ___syscall15);
mock_external!(import_object, ___syscall10);
2018-11-26 06:17:33 +00:00
import_object
}
#[cfg(test)]
mod tests {
use super::generate_emscripten_env;
use crate::webassembly::{instantiate, Export, Instance};
#[test]
fn test_putchar() {
let wasm_bytes = include_wast2wasm_bytes!("tests/putchar.wast");
let import_object = generate_emscripten_env();
let result_object = instantiate(wasm_bytes, import_object).expect("Not compiled properly");
let func_index = match result_object.module.info.exports.get("main") {
Some(&Export::Function(index)) => index,
_ => panic!("Function not found"),
};
let main: fn(&Instance) = get_instance_function!(result_object.instance, func_index);
main(&result_object.instance);
}
#[test]
fn test_print() {
let wasm_bytes = include_wast2wasm_bytes!("tests/printf.wast");
let import_object = generate_emscripten_env();
let result_object = instantiate(wasm_bytes, import_object).expect("Not compiled properly");
let func_index = match result_object.module.info.exports.get("main") {
Some(&Export::Function(index)) => index,
_ => panic!("Function not found"),
};
let main: fn(&Instance) = get_instance_function!(result_object.instance, func_index);
main(&result_object.instance);
}
}