wasmer/src/apis/emscripten/utils.rs

139 lines
4.6 KiB
Rust
Raw Normal View History

2018-12-30 10:23:16 +00:00
use crate::runtime::{Instance, Module};
//use crate::webassembly::Instance;
2018-12-15 06:46:11 +00:00
use byteorder::{ByteOrder, LittleEndian};
2018-11-27 04:29:26 +00:00
use libc::stat;
2018-12-09 07:02:26 +00:00
use std::ffi::CStr;
2018-12-15 06:46:11 +00:00
use std::mem::size_of;
2018-11-26 21:15:49 +00:00
use std::os::raw::c_char;
2018-11-27 04:28:13 +00:00
use std::slice;
2018-12-30 10:23:16 +00:00
use std::sync::Arc;
/// We check if a provided module is an Emscripten generated one
pub fn is_emscripten_module(module: &Module) -> bool {
2018-12-30 10:23:16 +00:00
for (_, import_name) in &module.imported_functions {
if import_name.name == "_emscripten_memcpy_big" && import_name.module == "env" {
return true;
}
}
2018-12-08 06:27:36 +00:00
false
}
2018-12-11 17:06:20 +00:00
pub unsafe fn write_to_buf(string: *const c_char, buf: u32, max: u32, instance: &Instance) -> u32 {
let buf_addr = instance.memory_offset_addr(0, buf as _) as *mut c_char;
2018-12-14 01:58:08 +00:00
for i in 0..max {
*buf_addr.add(i as _) = *string.add(i as _);
2018-12-11 17:06:20 +00:00
}
buf
}
2018-12-13 23:09:07 +00:00
/// This function expects nullbyte to be appended.
2018-12-06 20:21:58 +00:00
pub unsafe fn copy_cstr_into_wasm(instance: &mut Instance, cstr: *const c_char) -> u32 {
2018-12-07 03:08:17 +00:00
let s = CStr::from_ptr(cstr).to_str().unwrap();
let cstr_len = s.len();
2018-12-15 06:46:11 +00:00
let space_offset =
2018-12-30 10:23:16 +00:00
(instance.emscripten_data().as_ref().unwrap().malloc)((cstr_len as i32) + 1, instance);
2018-12-07 03:08:17 +00:00
let raw_memory = instance.memory_offset_addr(0, space_offset as _) as *mut u8;
let slice = slice::from_raw_parts_mut(raw_memory, cstr_len);
for (byte, loc) in s.bytes().zip(slice.iter_mut()) {
*loc = byte;
}
2018-12-07 13:50:35 +00:00
// TODO: Appending null byte won't work, because there is CStr::from_ptr(cstr)
// at the top that crashes when there is no null byte
2018-12-07 03:08:17 +00:00
*raw_memory.add(cstr_len) = 0;
space_offset
}
2018-12-15 06:46:11 +00:00
pub unsafe fn allocate_on_stack<'a, T: Copy>(
count: u32,
instance: &'a Instance,
) -> (u32, &'a mut [T]) {
2018-12-30 10:23:16 +00:00
let offset = (instance.emscripten_data().as_ref().unwrap().stack_alloc)(
2018-12-15 06:46:11 +00:00
count * (size_of::<T>() as u32),
instance,
);
2018-12-07 03:40:44 +00:00
let addr = instance.memory_offset_addr(0, offset as _) as *mut T;
let slice = slice::from_raw_parts_mut(addr, count as usize);
(offset, slice)
}
pub unsafe fn allocate_cstr_on_stack<'a>(s: &str, instance: &'a Instance) -> (u32, &'a [u8]) {
let (offset, slice) = allocate_on_stack((s.len() + 1) as u32, instance);
use std::iter;
for (byte, loc) in s.bytes().chain(iter::once(0)).zip(slice.iter_mut()) {
*loc = byte;
}
(offset, slice)
}
2018-11-27 04:29:26 +00:00
pub unsafe fn copy_terminated_array_of_cstrs(
_instance: &mut Instance,
cstrs: *mut *mut c_char,
) -> u32 {
2018-11-26 21:15:49 +00:00
let total_num = {
let mut ptr = cstrs;
let mut counter = 0;
while !(*ptr).is_null() {
counter += 1;
ptr = ptr.add(1);
}
counter
};
2018-11-27 04:29:26 +00:00
debug!(
"emscripten::copy_terminated_array_of_cstrs::total_num: {}",
total_num
);
2018-11-26 21:15:49 +00:00
0
}
pub unsafe fn copy_stat_into_wasm(instance: &mut Instance, buf: u32, stat: &stat) {
let buf_ptr = instance.memory_offset_addr(0, buf as _) as *mut u8;
2018-11-27 04:28:13 +00:00
let buf = slice::from_raw_parts_mut(buf_ptr, 76);
2018-11-27 04:29:26 +00:00
LittleEndian::write_u32(&mut buf[..], stat.st_dev as _);
LittleEndian::write_u32(&mut buf[4..], 0);
LittleEndian::write_u32(&mut buf[8..], stat.st_ino as _);
LittleEndian::write_u32(&mut buf[12..], stat.st_mode as _);
LittleEndian::write_u32(&mut buf[16..], stat.st_nlink as _);
LittleEndian::write_u32(&mut buf[20..], stat.st_uid);
LittleEndian::write_u32(&mut buf[24..], stat.st_gid);
LittleEndian::write_u32(&mut buf[28..], stat.st_rdev as _);
LittleEndian::write_u32(&mut buf[32..], 0);
LittleEndian::write_u32(&mut buf[36..], stat.st_size as _);
LittleEndian::write_u32(&mut buf[40..], 4096);
LittleEndian::write_u32(&mut buf[44..], stat.st_blocks as _);
LittleEndian::write_u32(&mut buf[48..], stat.st_atime as _);
LittleEndian::write_u32(&mut buf[52..], 0);
LittleEndian::write_u32(&mut buf[56..], stat.st_mtime as _);
LittleEndian::write_u32(&mut buf[60..], 0);
LittleEndian::write_u32(&mut buf[64..], stat.st_ctime as _);
LittleEndian::write_u32(&mut buf[68..], 0);
LittleEndian::write_u32(&mut buf[72..], stat.st_ino as _);
}
#[cfg(test)]
mod tests {
2018-11-22 04:59:23 +00:00
use super::is_emscripten_module;
use crate::webassembly::compile;
#[test]
fn should_detect_emscripten_files() {
let wasm_bytes = include_wast2wasm_bytes!("tests/is_emscripten_true.wast");
2018-12-30 10:23:16 +00:00
let module = compile(&wasm_bytes).expect("Not compiled properly");
assert!(is_emscripten_module(&module));
}
#[test]
fn should_detect_non_emscripten_files() {
let wasm_bytes = include_wast2wasm_bytes!("tests/is_emscripten_false.wast");
2018-12-30 10:23:16 +00:00
let module = compile(&wasm_bytes).expect("Not compiled properly");
assert!(!is_emscripten_module(&module));
}
2018-11-22 04:59:23 +00:00
}