mirror of
https://github.com/roc-lang/roc.git
synced 2025-09-26 21:39:07 +00:00
676 lines
12 KiB
Rust
676 lines
12 KiB
Rust
#![cfg(test)]
|
|
|
|
use bumpalo::Bump;
|
|
use roc_wasm_interp::{ExecutionState, Value};
|
|
use roc_wasm_module::{opcodes::OpCode, SerialBuffer, ValueType, WasmModule};
|
|
|
|
const DEFAULT_MEMORY_PAGES: u32 = 1;
|
|
const DEFAULT_PROGRAM_COUNTER: usize = 0;
|
|
|
|
// #[test]
|
|
// fn test_block() {}
|
|
|
|
// #[test]
|
|
// fn test_loop() {}
|
|
|
|
// #[test]
|
|
// fn test_if() {}
|
|
|
|
// #[test]
|
|
// fn test_else() {}
|
|
|
|
// #[test]
|
|
// fn test_end() {}
|
|
|
|
// #[test]
|
|
// fn test_br() {}
|
|
|
|
// #[test]
|
|
// fn test_brif() {}
|
|
|
|
// #[test]
|
|
// fn test_brtable() {}
|
|
|
|
// #[test]
|
|
// fn test_return() {}
|
|
|
|
// #[test]
|
|
// fn test_call() {}
|
|
|
|
// #[test]
|
|
// fn test_callindirect() {}
|
|
|
|
// #[test]
|
|
// fn test_drop() {}
|
|
|
|
// #[test]
|
|
// fn test_select() {}
|
|
|
|
#[test]
|
|
fn test_set_get_local() {
|
|
let arena = Bump::new();
|
|
let mut state = ExecutionState::new(&arena, DEFAULT_MEMORY_PAGES, DEFAULT_PROGRAM_COUNTER, []);
|
|
let mut module = WasmModule::new(&arena);
|
|
|
|
let local_decls = [
|
|
(1, ValueType::F32),
|
|
(1, ValueType::F64),
|
|
(1, ValueType::I32),
|
|
(1, ValueType::I64),
|
|
];
|
|
state.call_stack.push_frame(0x1234, &local_decls);
|
|
|
|
module.code.bytes.push(OpCode::I32CONST as u8);
|
|
module.code.bytes.encode_i32(12345);
|
|
module.code.bytes.push(OpCode::SETLOCAL as u8);
|
|
module.code.bytes.encode_u32(2);
|
|
|
|
module.code.bytes.push(OpCode::GETLOCAL as u8);
|
|
module.code.bytes.encode_u32(2);
|
|
|
|
state.execute_next_instruction(&module);
|
|
state.execute_next_instruction(&module);
|
|
state.execute_next_instruction(&module);
|
|
assert_eq!(state.value_stack.len(), 1);
|
|
assert_eq!(state.value_stack.pop(), Value::I32(12345));
|
|
}
|
|
|
|
#[test]
|
|
fn test_tee_get_local() {
|
|
let arena = Bump::new();
|
|
let mut state = ExecutionState::new(&arena, DEFAULT_MEMORY_PAGES, DEFAULT_PROGRAM_COUNTER, []);
|
|
let mut module = WasmModule::new(&arena);
|
|
|
|
let local_decls = [
|
|
(1, ValueType::F32),
|
|
(1, ValueType::F64),
|
|
(1, ValueType::I32),
|
|
(1, ValueType::I64),
|
|
];
|
|
state.call_stack.push_frame(0x1234, &local_decls);
|
|
|
|
module.code.bytes.push(OpCode::I32CONST as u8);
|
|
module.code.bytes.encode_i32(12345);
|
|
module.code.bytes.push(OpCode::TEELOCAL as u8);
|
|
module.code.bytes.encode_u32(2);
|
|
|
|
module.code.bytes.push(OpCode::GETLOCAL as u8);
|
|
module.code.bytes.encode_u32(2);
|
|
|
|
state.execute_next_instruction(&module);
|
|
state.execute_next_instruction(&module);
|
|
state.execute_next_instruction(&module);
|
|
assert_eq!(state.value_stack.len(), 2);
|
|
assert_eq!(state.value_stack.pop(), Value::I32(12345));
|
|
assert_eq!(state.value_stack.pop(), Value::I32(12345));
|
|
}
|
|
|
|
#[test]
|
|
fn test_global() {
|
|
let arena = Bump::new();
|
|
let mut state = ExecutionState::new(
|
|
&arena,
|
|
DEFAULT_MEMORY_PAGES,
|
|
DEFAULT_PROGRAM_COUNTER,
|
|
[Value::F64(1.11), Value::I32(222), Value::F64(3.33)],
|
|
);
|
|
let mut module = WasmModule::new(&arena);
|
|
|
|
module.code.bytes.push(OpCode::GETGLOBAL as u8);
|
|
module.code.bytes.encode_u32(1);
|
|
module.code.bytes.push(OpCode::I32CONST as u8);
|
|
module.code.bytes.encode_i32(555);
|
|
module.code.bytes.push(OpCode::SETGLOBAL as u8);
|
|
module.code.bytes.encode_u32(1);
|
|
module.code.bytes.push(OpCode::GETGLOBAL as u8);
|
|
module.code.bytes.encode_u32(1);
|
|
|
|
state.execute_next_instruction(&module);
|
|
state.execute_next_instruction(&module);
|
|
state.execute_next_instruction(&module);
|
|
state.execute_next_instruction(&module);
|
|
assert_eq!(state.value_stack.len(), 2);
|
|
assert_eq!(state.value_stack.pop(), Value::I32(555));
|
|
assert_eq!(state.value_stack.pop(), Value::I32(222));
|
|
}
|
|
|
|
// #[test]
|
|
// fn test_i32load() {}
|
|
|
|
// #[test]
|
|
// fn test_i64load() {}
|
|
|
|
// #[test]
|
|
// fn test_f32load() {}
|
|
|
|
// #[test]
|
|
// fn test_f64load() {}
|
|
|
|
// #[test]
|
|
// fn test_i32load8s() {}
|
|
|
|
// #[test]
|
|
// fn test_i32load8u() {}
|
|
|
|
// #[test]
|
|
// fn test_i32load16s() {}
|
|
|
|
// #[test]
|
|
// fn test_i32load16u() {}
|
|
|
|
// #[test]
|
|
// fn test_i64load8s() {}
|
|
|
|
// #[test]
|
|
// fn test_i64load8u() {}
|
|
|
|
// #[test]
|
|
// fn test_i64load16s() {}
|
|
|
|
// #[test]
|
|
// fn test_i64load16u() {}
|
|
|
|
// #[test]
|
|
// fn test_i64load32s() {}
|
|
|
|
// #[test]
|
|
// fn test_i64load32u() {}
|
|
|
|
// #[test]
|
|
// fn test_i32store() {}
|
|
|
|
// #[test]
|
|
// fn test_i64store() {}
|
|
|
|
// #[test]
|
|
// fn test_f32store() {}
|
|
|
|
// #[test]
|
|
// fn test_f64store() {}
|
|
|
|
// #[test]
|
|
// fn test_i32store8() {}
|
|
|
|
// #[test]
|
|
// fn test_i32store16() {}
|
|
|
|
// #[test]
|
|
// fn test_i64store8() {}
|
|
|
|
// #[test]
|
|
// fn test_i64store16() {}
|
|
|
|
// #[test]
|
|
// fn test_i64store32() {}
|
|
|
|
// #[test]
|
|
// fn test_currentmemory() {}
|
|
|
|
// #[test]
|
|
// fn test_growmemory() {}
|
|
|
|
#[test]
|
|
fn test_i32const() {
|
|
let arena = Bump::new();
|
|
let mut state = ExecutionState::new(&arena, DEFAULT_MEMORY_PAGES, DEFAULT_PROGRAM_COUNTER, []);
|
|
let mut module = WasmModule::new(&arena);
|
|
|
|
module.code.bytes.push(OpCode::I32CONST as u8);
|
|
module.code.bytes.encode_i32(12345);
|
|
|
|
state.execute_next_instruction(&module);
|
|
assert_eq!(state.value_stack.pop(), Value::I32(12345))
|
|
}
|
|
|
|
#[test]
|
|
fn test_i64const() {
|
|
let arena = Bump::new();
|
|
let mut state = ExecutionState::new(&arena, DEFAULT_MEMORY_PAGES, DEFAULT_PROGRAM_COUNTER, []);
|
|
let mut module = WasmModule::new(&arena);
|
|
|
|
module.code.bytes.push(OpCode::I64CONST as u8);
|
|
module.code.bytes.encode_i64(1234567890);
|
|
|
|
state.execute_next_instruction(&module);
|
|
assert_eq!(state.value_stack.pop(), Value::I64(1234567890))
|
|
}
|
|
|
|
#[test]
|
|
fn test_f32const() {
|
|
let arena = Bump::new();
|
|
let mut state = ExecutionState::new(&arena, DEFAULT_MEMORY_PAGES, DEFAULT_PROGRAM_COUNTER, []);
|
|
let mut module = WasmModule::new(&arena);
|
|
|
|
module.code.bytes.push(OpCode::F32CONST as u8);
|
|
module.code.bytes.encode_f32(123.45);
|
|
|
|
state.execute_next_instruction(&module);
|
|
assert_eq!(state.value_stack.pop(), Value::F32(123.45))
|
|
}
|
|
|
|
#[test]
|
|
fn test_f64const() {
|
|
let arena = Bump::new();
|
|
let mut state = ExecutionState::new(&arena, DEFAULT_MEMORY_PAGES, DEFAULT_PROGRAM_COUNTER, []);
|
|
let mut module = WasmModule::new(&arena);
|
|
|
|
module.code.bytes.push(OpCode::F64CONST as u8);
|
|
module.code.bytes.encode_f64(12345.67890);
|
|
|
|
state.execute_next_instruction(&module);
|
|
assert_eq!(state.value_stack.pop(), Value::F64(12345.67890))
|
|
}
|
|
|
|
// #[test]
|
|
// fn test_i32eqz() {}
|
|
|
|
// #[test]
|
|
// fn test_i32eq() {}
|
|
|
|
// #[test]
|
|
// fn test_i32ne() {}
|
|
|
|
// #[test]
|
|
// fn test_i32lts() {}
|
|
|
|
// #[test]
|
|
// fn test_i32ltu() {}
|
|
|
|
// #[test]
|
|
// fn test_i32gts() {}
|
|
|
|
// #[test]
|
|
// fn test_i32gtu() {}
|
|
|
|
// #[test]
|
|
// fn test_i32les() {}
|
|
|
|
// #[test]
|
|
// fn test_i32leu() {}
|
|
|
|
// #[test]
|
|
// fn test_i32ges() {}
|
|
|
|
// #[test]
|
|
// fn test_i32geu() {}
|
|
|
|
// #[test]
|
|
// fn test_i64eqz() {}
|
|
|
|
// #[test]
|
|
// fn test_i64eq() {}
|
|
|
|
// #[test]
|
|
// fn test_i64ne() {}
|
|
|
|
// #[test]
|
|
// fn test_i64lts() {}
|
|
|
|
// #[test]
|
|
// fn test_i64ltu() {}
|
|
|
|
// #[test]
|
|
// fn test_i64gts() {}
|
|
|
|
// #[test]
|
|
// fn test_i64gtu() {}
|
|
|
|
// #[test]
|
|
// fn test_i64les() {}
|
|
|
|
// #[test]
|
|
// fn test_i64leu() {}
|
|
|
|
// #[test]
|
|
// fn test_i64ges() {}
|
|
|
|
// #[test]
|
|
// fn test_i64geu() {}
|
|
|
|
// #[test]
|
|
// fn test_f32eq() {}
|
|
|
|
// #[test]
|
|
// fn test_f32ne() {}
|
|
|
|
// #[test]
|
|
// fn test_f32lt() {}
|
|
|
|
// #[test]
|
|
// fn test_f32gt() {}
|
|
|
|
// #[test]
|
|
// fn test_f32le() {}
|
|
|
|
// #[test]
|
|
// fn test_f32ge() {}
|
|
|
|
// #[test]
|
|
// fn test_f64eq() {}
|
|
|
|
// #[test]
|
|
// fn test_f64ne() {}
|
|
|
|
// #[test]
|
|
// fn test_f64lt() {}
|
|
|
|
// #[test]
|
|
// fn test_f64gt() {}
|
|
|
|
// #[test]
|
|
// fn test_f64le() {}
|
|
|
|
// #[test]
|
|
// fn test_f64ge() {}
|
|
|
|
// #[test]
|
|
// fn test_i32clz() {}
|
|
|
|
// #[test]
|
|
// fn test_i32ctz() {}
|
|
|
|
// #[test]
|
|
// fn test_i32popcnt() {}
|
|
|
|
#[test]
|
|
fn test_i32add() {
|
|
let arena = Bump::new();
|
|
let mut state = ExecutionState::new(&arena, DEFAULT_MEMORY_PAGES, DEFAULT_PROGRAM_COUNTER, []);
|
|
let mut module = WasmModule::new(&arena);
|
|
|
|
module.code.bytes.push(OpCode::I32CONST as u8);
|
|
module.code.bytes.encode_i32(123);
|
|
module.code.bytes.push(OpCode::I32CONST as u8);
|
|
module.code.bytes.encode_i32(321);
|
|
module.code.bytes.push(OpCode::I32ADD as u8);
|
|
|
|
state.execute_next_instruction(&module);
|
|
state.execute_next_instruction(&module);
|
|
state.execute_next_instruction(&module);
|
|
assert_eq!(state.value_stack.pop(), Value::I32(444))
|
|
}
|
|
|
|
#[test]
|
|
fn test_i32sub() {
|
|
let arena = Bump::new();
|
|
let mut state = ExecutionState::new(&arena, DEFAULT_MEMORY_PAGES, DEFAULT_PROGRAM_COUNTER, []);
|
|
let mut module = WasmModule::new(&arena);
|
|
|
|
module.code.bytes.push(OpCode::I32CONST as u8);
|
|
module.code.bytes.encode_i32(123);
|
|
module.code.bytes.push(OpCode::I32CONST as u8);
|
|
module.code.bytes.encode_i32(321);
|
|
module.code.bytes.push(OpCode::I32SUB as u8);
|
|
|
|
state.execute_next_instruction(&module);
|
|
state.execute_next_instruction(&module);
|
|
state.execute_next_instruction(&module);
|
|
assert_eq!(state.value_stack.pop(), Value::I32(-198))
|
|
}
|
|
|
|
#[test]
|
|
fn test_i32mul() {
|
|
let arena = Bump::new();
|
|
let mut state = ExecutionState::new(&arena, DEFAULT_MEMORY_PAGES, DEFAULT_PROGRAM_COUNTER, []);
|
|
let mut module = WasmModule::new(&arena);
|
|
|
|
module.code.bytes.push(OpCode::I32CONST as u8);
|
|
module.code.bytes.encode_i32(123);
|
|
module.code.bytes.push(OpCode::I32CONST as u8);
|
|
module.code.bytes.encode_i32(321);
|
|
module.code.bytes.push(OpCode::I32MUL as u8);
|
|
|
|
state.execute_next_instruction(&module);
|
|
state.execute_next_instruction(&module);
|
|
state.execute_next_instruction(&module);
|
|
assert_eq!(state.value_stack.pop(), Value::I32(39483))
|
|
}
|
|
|
|
// #[test]
|
|
// fn test_i32divs() {}
|
|
|
|
// #[test]
|
|
// fn test_i32divu() {}
|
|
|
|
// #[test]
|
|
// fn test_i32rems() {}
|
|
|
|
// #[test]
|
|
// fn test_i32remu() {}
|
|
|
|
// #[test]
|
|
// fn test_i32and() {}
|
|
|
|
// #[test]
|
|
// fn test_i32or() {}
|
|
|
|
// #[test]
|
|
// fn test_i32xor() {}
|
|
|
|
// #[test]
|
|
// fn test_i32shl() {}
|
|
|
|
// #[test]
|
|
// fn test_i32shrs() {}
|
|
|
|
// #[test]
|
|
// fn test_i32shru() {}
|
|
|
|
// #[test]
|
|
// fn test_i32rotl() {}
|
|
|
|
// #[test]
|
|
// fn test_i32rotr() {}
|
|
|
|
// #[test]
|
|
// fn test_i64clz() {}
|
|
|
|
// #[test]
|
|
// fn test_i64ctz() {}
|
|
|
|
// #[test]
|
|
// fn test_i64popcnt() {}
|
|
|
|
// #[test]
|
|
// fn test_i64add() {}
|
|
|
|
// #[test]
|
|
// fn test_i64sub() {}
|
|
|
|
// #[test]
|
|
// fn test_i64mul() {}
|
|
|
|
// #[test]
|
|
// fn test_i64divs() {}
|
|
|
|
// #[test]
|
|
// fn test_i64divu() {}
|
|
|
|
// #[test]
|
|
// fn test_i64rems() {}
|
|
|
|
// #[test]
|
|
// fn test_i64remu() {}
|
|
|
|
// #[test]
|
|
// fn test_i64and() {}
|
|
|
|
// #[test]
|
|
// fn test_i64or() {}
|
|
|
|
// #[test]
|
|
// fn test_i64xor() {}
|
|
|
|
// #[test]
|
|
// fn test_i64shl() {}
|
|
|
|
// #[test]
|
|
// fn test_i64shrs() {}
|
|
|
|
// #[test]
|
|
// fn test_i64shru() {}
|
|
|
|
// #[test]
|
|
// fn test_i64rotl() {}
|
|
|
|
// #[test]
|
|
// fn test_i64rotr() {}
|
|
|
|
// #[test]
|
|
// fn test_f32abs() {}
|
|
|
|
// #[test]
|
|
// fn test_f32neg() {}
|
|
|
|
// #[test]
|
|
// fn test_f32ceil() {}
|
|
|
|
// #[test]
|
|
// fn test_f32floor() {}
|
|
|
|
// #[test]
|
|
// fn test_f32trunc() {}
|
|
|
|
// #[test]
|
|
// fn test_f32nearest() {}
|
|
|
|
// #[test]
|
|
// fn test_f32sqrt() {}
|
|
|
|
// #[test]
|
|
// fn test_f32add() {}
|
|
|
|
// #[test]
|
|
// fn test_f32sub() {}
|
|
|
|
// #[test]
|
|
// fn test_f32mul() {}
|
|
|
|
// #[test]
|
|
// fn test_f32div() {}
|
|
|
|
// #[test]
|
|
// fn test_f32min() {}
|
|
|
|
// #[test]
|
|
// fn test_f32max() {}
|
|
|
|
// #[test]
|
|
// fn test_f32copysign() {}
|
|
|
|
// #[test]
|
|
// fn test_f64abs() {}
|
|
|
|
// #[test]
|
|
// fn test_f64neg() {}
|
|
|
|
// #[test]
|
|
// fn test_f64ceil() {}
|
|
|
|
// #[test]
|
|
// fn test_f64floor() {}
|
|
|
|
// #[test]
|
|
// fn test_f64trunc() {}
|
|
|
|
// #[test]
|
|
// fn test_f64nearest() {}
|
|
|
|
// #[test]
|
|
// fn test_f64sqrt() {}
|
|
|
|
// #[test]
|
|
// fn test_f64add() {}
|
|
|
|
// #[test]
|
|
// fn test_f64sub() {}
|
|
|
|
// #[test]
|
|
// fn test_f64mul() {}
|
|
|
|
// #[test]
|
|
// fn test_f64div() {}
|
|
|
|
// #[test]
|
|
// fn test_f64min() {}
|
|
|
|
// #[test]
|
|
// fn test_f64max() {}
|
|
|
|
// #[test]
|
|
// fn test_f64copysign() {}
|
|
|
|
// #[test]
|
|
// fn test_i32wrapi64() {}
|
|
|
|
// #[test]
|
|
// fn test_i32truncsf32() {}
|
|
|
|
// #[test]
|
|
// fn test_i32truncuf32() {}
|
|
|
|
// #[test]
|
|
// fn test_i32truncsf64() {}
|
|
|
|
// #[test]
|
|
// fn test_i32truncuf64() {}
|
|
|
|
// #[test]
|
|
// fn test_i64extendsi32() {}
|
|
|
|
// #[test]
|
|
// fn test_i64extendui32() {}
|
|
|
|
// #[test]
|
|
// fn test_i64truncsf32() {}
|
|
|
|
// #[test]
|
|
// fn test_i64truncuf32() {}
|
|
|
|
// #[test]
|
|
// fn test_i64truncsf64() {}
|
|
|
|
// #[test]
|
|
// fn test_i64truncuf64() {}
|
|
|
|
// #[test]
|
|
// fn test_f32convertsi32() {}
|
|
|
|
// #[test]
|
|
// fn test_f32convertui32() {}
|
|
|
|
// #[test]
|
|
// fn test_f32convertsi64() {}
|
|
|
|
// #[test]
|
|
// fn test_f32convertui64() {}
|
|
|
|
// #[test]
|
|
// fn test_f32demotef64() {}
|
|
|
|
// #[test]
|
|
// fn test_f64convertsi32() {}
|
|
|
|
// #[test]
|
|
// fn test_f64convertui32() {}
|
|
|
|
// #[test]
|
|
// fn test_f64convertsi64() {}
|
|
|
|
// #[test]
|
|
// fn test_f64convertui64() {}
|
|
|
|
// #[test]
|
|
// fn test_f64promotef32() {}
|
|
|
|
// #[test]
|
|
// fn test_i32reinterpretf32() {}
|
|
|
|
// #[test]
|
|
// fn test_i64reinterpretf64() {}
|
|
|
|
// #[test]
|
|
// fn test_f32reinterpreti32() {}
|
|
|
|
// #[test]
|
|
// fn test_f64reinterpreti64() {}
|