use wasm_encoder::{Function, Instruction, ValType};
use super::super::value::*;
use super::RuntimeFuncIndices;
pub(super) fn emit_vec_from_list(rt: &RuntimeFuncIndices) -> Function {
let mut f = Function::new(vec![
(5, ValType::I32), ]);
f.instruction(&Instruction::I32Const(0));
f.instruction(&Instruction::LocalSet(2));
f.instruction(&Instruction::LocalGet(0));
f.instruction(&Instruction::LocalSet(3));
f.instruction(&Instruction::Block(wasm_encoder::BlockType::Empty));
f.instruction(&Instruction::Loop(wasm_encoder::BlockType::Empty));
f.instruction(&Instruction::LocalGet(3));
f.instruction(&Instruction::I32Eqz);
f.instruction(&Instruction::BrIf(1));
f.instruction(&Instruction::LocalGet(2));
f.instruction(&Instruction::I32Const(1));
f.instruction(&Instruction::I32Add);
f.instruction(&Instruction::LocalSet(2));
f.instruction(&Instruction::LocalGet(3));
f.instruction(&Instruction::I32Const(1));
f.instruction(&Instruction::Call(rt.obj_field_i32));
f.instruction(&Instruction::LocalSet(3));
f.instruction(&Instruction::Br(0));
f.instruction(&Instruction::End);
f.instruction(&Instruction::End);
f.instruction(&Instruction::I32Const(8));
f.instruction(&Instruction::LocalGet(2));
f.instruction(&Instruction::I32Const(8));
f.instruction(&Instruction::I32Mul);
f.instruction(&Instruction::I32Add);
f.instruction(&Instruction::Call(rt.alloc));
f.instruction(&Instruction::LocalSet(4));
f.instruction(&Instruction::LocalGet(4));
f.instruction(&Instruction::I64Const(
(OBJ_VECTOR << HDR_KIND_SHIFT) as i64,
));
f.instruction(&Instruction::LocalGet(1));
f.instruction(&Instruction::I64ExtendI32U);
f.instruction(&Instruction::I64Const(HDR_META_SHIFT as i64));
f.instruction(&Instruction::I64Shl);
f.instruction(&Instruction::I64Or);
f.instruction(&Instruction::LocalGet(2));
f.instruction(&Instruction::I64ExtendI32U);
f.instruction(&Instruction::I64Or);
f.instruction(&Instruction::I64Store(wasm_encoder::MemArg {
offset: 0,
align: 3,
memory_index: 0,
}));
f.instruction(&Instruction::I32Const(0));
f.instruction(&Instruction::LocalSet(5)); f.instruction(&Instruction::LocalGet(0));
f.instruction(&Instruction::LocalSet(6)); f.instruction(&Instruction::Block(wasm_encoder::BlockType::Empty));
f.instruction(&Instruction::Loop(wasm_encoder::BlockType::Empty));
f.instruction(&Instruction::LocalGet(6));
f.instruction(&Instruction::I32Eqz);
f.instruction(&Instruction::BrIf(1));
f.instruction(&Instruction::LocalGet(4));
f.instruction(&Instruction::LocalGet(5));
f.instruction(&Instruction::I32Const(8));
f.instruction(&Instruction::I32Mul);
f.instruction(&Instruction::I32Add);
f.instruction(&Instruction::LocalGet(6));
f.instruction(&Instruction::I32Const(0));
f.instruction(&Instruction::Call(rt.obj_field)); f.instruction(&Instruction::I64Store(wasm_encoder::MemArg {
offset: 8,
align: 3,
memory_index: 0,
}));
f.instruction(&Instruction::LocalGet(5));
f.instruction(&Instruction::I32Const(1));
f.instruction(&Instruction::I32Add);
f.instruction(&Instruction::LocalSet(5));
f.instruction(&Instruction::LocalGet(6));
f.instruction(&Instruction::I32Const(1));
f.instruction(&Instruction::Call(rt.obj_field_i32));
f.instruction(&Instruction::LocalSet(6));
f.instruction(&Instruction::Br(0));
f.instruction(&Instruction::End);
f.instruction(&Instruction::End);
f.instruction(&Instruction::LocalGet(4));
f.instruction(&Instruction::End);
f
}
pub(super) fn emit_vec_get(rt: &RuntimeFuncIndices) -> Function {
let mut f = Function::new(vec![
(1, ValType::I32), (1, ValType::I32), (1, ValType::I32), ]);
f.instruction(&Instruction::LocalGet(0));
f.instruction(&Instruction::I64Load(wasm_encoder::MemArg {
offset: 0,
align: 3,
memory_index: 0,
}));
f.instruction(&Instruction::I64Const(0xFFFFFFFF));
f.instruction(&Instruction::I64And);
f.instruction(&Instruction::I32WrapI64);
f.instruction(&Instruction::LocalSet(2));
f.instruction(&Instruction::LocalGet(0));
f.instruction(&Instruction::Call(rt.obj_meta));
f.instruction(&Instruction::LocalSet(4));
f.instruction(&Instruction::LocalGet(1));
f.instruction(&Instruction::I32WrapI64);
f.instruction(&Instruction::LocalSet(3));
f.instruction(&Instruction::LocalGet(3));
f.instruction(&Instruction::I32Const(0));
f.instruction(&Instruction::I32LtS);
f.instruction(&Instruction::LocalGet(3));
f.instruction(&Instruction::LocalGet(2));
f.instruction(&Instruction::I32GeS);
f.instruction(&Instruction::I32Or);
f.instruction(&Instruction::If(wasm_encoder::BlockType::Result(
ValType::I32,
)));
f.instruction(&Instruction::I32Const(NONE_SENTINEL));
f.instruction(&Instruction::Else);
f.instruction(&Instruction::I32Const(WRAP_SOME as i32));
f.instruction(&Instruction::LocalGet(0));
f.instruction(&Instruction::LocalGet(3));
f.instruction(&Instruction::I32Const(8));
f.instruction(&Instruction::I32Mul);
f.instruction(&Instruction::I32Add);
f.instruction(&Instruction::I64Load(wasm_encoder::MemArg {
offset: 8,
align: 3,
memory_index: 0,
}));
f.instruction(&Instruction::LocalGet(4));
f.instruction(&Instruction::I32Const(1));
f.instruction(&Instruction::I32And);
f.instruction(&Instruction::Call(rt.wrap)); f.instruction(&Instruction::End);
f.instruction(&Instruction::End);
f
}
#[allow(dead_code)]
fn _emit_vec_get_or_default() -> Function {
let mut f = Function::new(vec![
(1, ValType::I32), (1, ValType::I32), ]);
f.instruction(&Instruction::LocalGet(0));
f.instruction(&Instruction::I64Load(wasm_encoder::MemArg {
offset: 0,
align: 3,
memory_index: 0,
}));
f.instruction(&Instruction::I64Const(0xFFFFFFFF));
f.instruction(&Instruction::I64And);
f.instruction(&Instruction::I32WrapI64);
f.instruction(&Instruction::LocalSet(3));
f.instruction(&Instruction::LocalGet(1));
f.instruction(&Instruction::I32WrapI64);
f.instruction(&Instruction::LocalSet(4));
f.instruction(&Instruction::LocalGet(4));
f.instruction(&Instruction::I32Const(0));
f.instruction(&Instruction::I32LtS);
f.instruction(&Instruction::LocalGet(4));
f.instruction(&Instruction::LocalGet(3));
f.instruction(&Instruction::I32GeS);
f.instruction(&Instruction::I32Or);
f.instruction(&Instruction::If(wasm_encoder::BlockType::Result(
ValType::I64,
)));
f.instruction(&Instruction::LocalGet(2)); f.instruction(&Instruction::Else);
f.instruction(&Instruction::LocalGet(0));
f.instruction(&Instruction::LocalGet(4));
f.instruction(&Instruction::I32Const(8));
f.instruction(&Instruction::I32Mul);
f.instruction(&Instruction::I32Add);
f.instruction(&Instruction::I64Load(wasm_encoder::MemArg {
offset: 8,
align: 3,
memory_index: 0,
}));
f.instruction(&Instruction::End);
f.instruction(&Instruction::End);
f
}
pub(super) fn emit_vec_len() -> Function {
let mut f = Function::new(vec![]);
f.instruction(&Instruction::LocalGet(0));
f.instruction(&Instruction::I64Load(wasm_encoder::MemArg {
offset: 0,
align: 3,
memory_index: 0,
}));
f.instruction(&Instruction::I64Const(0xFFFFFFFF));
f.instruction(&Instruction::I64And);
f.instruction(&Instruction::End);
f
}
pub(super) fn emit_vec_set(rt: &RuntimeFuncIndices) -> Function {
let mut f = Function::new(vec![
(1, ValType::I32), (1, ValType::I32), (1, ValType::I32), (1, ValType::I32), ]);
f.instruction(&Instruction::LocalGet(0));
f.instruction(&Instruction::I64Load(wasm_encoder::MemArg {
offset: 0,
align: 3,
memory_index: 0,
}));
f.instruction(&Instruction::I64Const(0xFFFFFFFF));
f.instruction(&Instruction::I64And);
f.instruction(&Instruction::I32WrapI64);
f.instruction(&Instruction::LocalSet(3));
f.instruction(&Instruction::LocalGet(1));
f.instruction(&Instruction::I32WrapI64);
f.instruction(&Instruction::LocalSet(5));
f.instruction(&Instruction::LocalGet(5));
f.instruction(&Instruction::I32Const(0));
f.instruction(&Instruction::I32LtS);
f.instruction(&Instruction::LocalGet(5));
f.instruction(&Instruction::LocalGet(3));
f.instruction(&Instruction::I32GeS);
f.instruction(&Instruction::I32Or);
f.instruction(&Instruction::If(wasm_encoder::BlockType::Result(
ValType::I32,
)));
f.instruction(&Instruction::I32Const(NONE_SENTINEL));
f.instruction(&Instruction::Else);
f.instruction(&Instruction::I32Const(8));
f.instruction(&Instruction::LocalGet(3));
f.instruction(&Instruction::I32Const(8));
f.instruction(&Instruction::I32Mul);
f.instruction(&Instruction::I32Add);
f.instruction(&Instruction::LocalSet(6)); f.instruction(&Instruction::LocalGet(6));
f.instruction(&Instruction::Call(rt.alloc));
f.instruction(&Instruction::LocalSet(4));
f.instruction(&Instruction::LocalGet(4));
f.instruction(&Instruction::LocalGet(0));
f.instruction(&Instruction::LocalGet(6));
f.instruction(&Instruction::MemoryCopy {
src_mem: 0,
dst_mem: 0,
});
f.instruction(&Instruction::LocalGet(4));
f.instruction(&Instruction::LocalGet(5));
f.instruction(&Instruction::I32Const(8));
f.instruction(&Instruction::I32Mul);
f.instruction(&Instruction::I32Add);
f.instruction(&Instruction::LocalGet(2));
f.instruction(&Instruction::I64Store(wasm_encoder::MemArg {
offset: 8,
align: 3,
memory_index: 0,
}));
f.instruction(&Instruction::I32Const(WRAP_SOME as i32));
f.instruction(&Instruction::LocalGet(4));
f.instruction(&Instruction::I64ExtendI32U);
f.instruction(&Instruction::I32Const(1));
f.instruction(&Instruction::Call(rt.wrap));
f.instruction(&Instruction::End); f.instruction(&Instruction::End);
f
}
pub(super) fn emit_vec_new(rt: &RuntimeFuncIndices) -> Function {
let mut f = Function::new(vec![
(3, ValType::I32), ]);
f.instruction(&Instruction::LocalGet(0));
f.instruction(&Instruction::I32WrapI64);
f.instruction(&Instruction::LocalSet(3));
f.instruction(&Instruction::I32Const(8));
f.instruction(&Instruction::LocalGet(3));
f.instruction(&Instruction::I32Const(8));
f.instruction(&Instruction::I32Mul);
f.instruction(&Instruction::I32Add);
f.instruction(&Instruction::Call(rt.alloc));
f.instruction(&Instruction::LocalSet(4));
f.instruction(&Instruction::LocalGet(4));
f.instruction(&Instruction::I64Const(
(OBJ_VECTOR << HDR_KIND_SHIFT) as i64,
));
f.instruction(&Instruction::LocalGet(2));
f.instruction(&Instruction::I64ExtendI32U);
f.instruction(&Instruction::I64Const(HDR_META_SHIFT as i64));
f.instruction(&Instruction::I64Shl);
f.instruction(&Instruction::I64Or);
f.instruction(&Instruction::LocalGet(3));
f.instruction(&Instruction::I64ExtendI32U);
f.instruction(&Instruction::I64Or);
f.instruction(&Instruction::I64Store(wasm_encoder::MemArg {
offset: 0,
align: 3,
memory_index: 0,
}));
f.instruction(&Instruction::I32Const(0));
f.instruction(&Instruction::LocalSet(5));
f.instruction(&Instruction::Block(wasm_encoder::BlockType::Empty));
f.instruction(&Instruction::Loop(wasm_encoder::BlockType::Empty));
f.instruction(&Instruction::LocalGet(5));
f.instruction(&Instruction::LocalGet(3));
f.instruction(&Instruction::I32GeU);
f.instruction(&Instruction::BrIf(1));
f.instruction(&Instruction::LocalGet(4));
f.instruction(&Instruction::LocalGet(5));
f.instruction(&Instruction::I32Const(8));
f.instruction(&Instruction::I32Mul);
f.instruction(&Instruction::I32Add);
f.instruction(&Instruction::LocalGet(1));
f.instruction(&Instruction::I64Store(wasm_encoder::MemArg {
offset: 8,
align: 3,
memory_index: 0,
}));
f.instruction(&Instruction::LocalGet(5));
f.instruction(&Instruction::I32Const(1));
f.instruction(&Instruction::I32Add);
f.instruction(&Instruction::LocalSet(5));
f.instruction(&Instruction::Br(0));
f.instruction(&Instruction::End);
f.instruction(&Instruction::End);
f.instruction(&Instruction::LocalGet(4));
f.instruction(&Instruction::End);
f
}
pub(super) fn emit_vec_to_list(rt: &RuntimeFuncIndices) -> Function {
let mut f = Function::new(vec![
(1, ValType::I32), (1, ValType::I32), (1, ValType::I32), (1, ValType::I32), ]);
f.instruction(&Instruction::LocalGet(0));
f.instruction(&Instruction::I64Load(wasm_encoder::MemArg {
offset: 0,
align: 3,
memory_index: 0,
}));
f.instruction(&Instruction::I64Const(0xFFFFFFFF));
f.instruction(&Instruction::I64And);
f.instruction(&Instruction::I32WrapI64);
f.instruction(&Instruction::LocalSet(1));
f.instruction(&Instruction::LocalGet(0));
f.instruction(&Instruction::Call(rt.obj_meta));
f.instruction(&Instruction::LocalSet(4));
f.instruction(&Instruction::I32Const(0));
f.instruction(&Instruction::LocalSet(3)); f.instruction(&Instruction::LocalGet(1));
f.instruction(&Instruction::LocalSet(2));
f.instruction(&Instruction::Block(wasm_encoder::BlockType::Empty));
f.instruction(&Instruction::Loop(wasm_encoder::BlockType::Empty));
f.instruction(&Instruction::LocalGet(2));
f.instruction(&Instruction::I32Eqz);
f.instruction(&Instruction::BrIf(1));
f.instruction(&Instruction::LocalGet(2));
f.instruction(&Instruction::I32Const(1));
f.instruction(&Instruction::I32Sub);
f.instruction(&Instruction::LocalSet(2));
f.instruction(&Instruction::LocalGet(0));
f.instruction(&Instruction::LocalGet(2));
f.instruction(&Instruction::I32Const(8));
f.instruction(&Instruction::I32Mul);
f.instruction(&Instruction::I32Add);
f.instruction(&Instruction::I64Load(wasm_encoder::MemArg {
offset: 8,
align: 3,
memory_index: 0,
}));
f.instruction(&Instruction::LocalGet(3));
f.instruction(&Instruction::LocalGet(4));
f.instruction(&Instruction::I32Const(1));
f.instruction(&Instruction::I32And);
f.instruction(&Instruction::Call(rt.list_cons));
f.instruction(&Instruction::LocalSet(3));
f.instruction(&Instruction::Br(0));
f.instruction(&Instruction::End);
f.instruction(&Instruction::End);
f.instruction(&Instruction::LocalGet(3));
f.instruction(&Instruction::End);
f
}