use super::super::host; /// NOTE: These syscalls only support wasm_32 for now because they take u32 offset use libc::{c_int, getgrnam as libc_getgrnam, getpwnam as libc_getpwnam, c_long, sysconf}; use std::ffi::CStr; use std::mem; use std::os::raw::c_char; use super::utils::{copy_cstr_into_wasm, copy_terminated_array_of_cstrs}; use crate::webassembly::Instance; #[no_mangle] /// emscripten: _getenv // (name: *const char) -> *const c_char; pub extern "C" fn _getenv(name: c_int, instance: &mut Instance) -> u32 { debug!("emscripten::_getenv {}", name); let name = unsafe { let name_addr = instance.memory_offset_addr(0, name as usize) as *const c_char; CStr::from_ptr(name_addr).to_str().unwrap() }; match host::get_env(name, instance) { Ok(env_value) => { // Append null byte let env_value = env_value + "\0"; let env_value_ptr = env_value.as_ptr() as *mut c_char; let res = unsafe { copy_cstr_into_wasm(instance, env_value_ptr) }; // Test let c_str = instance.memory_offset_addr(0, res as _) as *mut i8; use std::ffi::CStr; debug!("#### cstr = {:?}", unsafe { CStr::from_ptr(c_str) }); debug!("res = {}", res); res } Err(_) => 0, } } pub extern "C" fn _getpwnam(name_ptr: c_int, instance: &mut Instance) -> c_int { debug!("emscripten::_getpwnam {}", name_ptr); #[repr(C)] struct GuestPasswd { pw_name: u32, pw_passwd: u32, pw_uid: u32, pw_gid: u32, pw_gecos: u32, pw_dir: u32, pw_shell: u32, } let name = unsafe { let memory_name_ptr = instance.memory_offset_addr(0, name_ptr as usize) as *const c_char; CStr::from_ptr(memory_name_ptr) }; unsafe { let passwd = &*libc_getpwnam(name.as_ptr()); let passwd_struct_offset = (instance.emscripten_data.as_ref().unwrap().malloc)( mem::size_of::() as _, instance, ); let passwd_struct_ptr = instance.memory_offset_addr(0, passwd_struct_offset as _) as *mut GuestPasswd; (*passwd_struct_ptr).pw_name = copy_cstr_into_wasm(instance, passwd.pw_name); (*passwd_struct_ptr).pw_passwd = copy_cstr_into_wasm(instance, passwd.pw_passwd); (*passwd_struct_ptr).pw_gecos = copy_cstr_into_wasm(instance, passwd.pw_gecos); (*passwd_struct_ptr).pw_dir = copy_cstr_into_wasm(instance, passwd.pw_dir); (*passwd_struct_ptr).pw_shell = copy_cstr_into_wasm(instance, passwd.pw_shell); (*passwd_struct_ptr).pw_uid = passwd.pw_uid; (*passwd_struct_ptr).pw_gid = passwd.pw_gid; passwd_struct_offset as c_int } } pub extern "C" fn _getgrnam(name_ptr: c_int, instance: &mut Instance) -> c_int { debug!("emscripten::_getgrnam {}", name_ptr); #[repr(C)] struct GuestGroup { gr_name: u32, gr_passwd: u32, gr_gid: u32, gr_mem: u32, } let name = unsafe { let memory_name_ptr = instance.memory_offset_addr(0, name_ptr as usize) as *const c_char; CStr::from_ptr(memory_name_ptr) }; unsafe { let group = &*libc_getgrnam(name.as_ptr()); let group_struct_offset = (instance.emscripten_data.as_ref().unwrap().malloc)( mem::size_of::() as _, instance, ); let group_struct_ptr = instance.memory_offset_addr(0, group_struct_offset as _) as *mut GuestGroup; (*group_struct_ptr).gr_name = copy_cstr_into_wasm(instance, group.gr_name); (*group_struct_ptr).gr_passwd = copy_cstr_into_wasm(instance, group.gr_passwd); (*group_struct_ptr).gr_gid = group.gr_gid; (*group_struct_ptr).gr_mem = copy_terminated_array_of_cstrs(instance, group.gr_mem); group_struct_offset as c_int } } pub extern "C" fn _getpagesize() -> u32 { debug!("emscripten::_getpagesize"); 16384 } pub extern "C" fn ___build_environment(environ: c_int) { debug!("emscripten::___build_environment {}", environ); } pub extern "C" fn _sysconf(name: c_int, _instance: &mut Instance) -> c_long { debug!("emscripten::_sysconf {}", name); // TODO: Implement like emscripten expects regarding memory/page size unsafe { sysconf(name) } }