/* * nanowasm, a tiny WebAssembly/Wasm interpreter * Copyright (C) 2023-2024 Xavier Del Campo Romero * * This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at https://mozilla.org/MPL/2.0/. */ #include #include #include #include #include #include #include #include #include #include static int set_i32(struct nw_interp *const i, void *const data, const varuint32 idx) { int32_t *const a = data, v; if (interp_stack_pop(i, &v, sizeof v)) { LOG("%s: interp_stack_pop failed\n", __func__); return -1; } a[idx] = v; return 0; } static int set_i64(struct nw_interp *const i, void *const data, const varuint32 idx) { int64_t *const a = data, v; if (interp_stack_pop(i, &v, sizeof v)) { LOG("%s: interp_stack_pop failed\n", __func__); return -1; } a[idx] = v; return 0; } static int set_f32(struct nw_interp *const i, void *const data, const varuint32 idx) { float *const a = data, v; if (interp_stack_pop(i, &v, sizeof v)) { LOG("%s: interp_stack_pop failed\n", __func__); return -1; } a[idx] = v; return 0; } static int set_f64(struct nw_interp *const i, void *const data, const varuint32 idx) { double *const a = data, v; if (interp_stack_pop(i, &v, sizeof v)) { LOG("%s: interp_stack_pop failed\n", __func__); return -1; } a[idx] = v; return 0; } static int set_local(const varuint32 local_index, struct nw_interp *const i) { varuint32 idx; struct nw_locals *l; if (!locals_get(local_index, i, &l, &idx)) { LOG("%s: get_pos failed\n", __func__); return -1; } static int (*const set[])(struct nw_interp *, void *, varuint32) = { [VALUE_TYPE_I32] = set_i32, [VALUE_TYPE_I64] = set_i64, [VALUE_TYPE_F32] = set_f32, [VALUE_TYPE_F64] = set_f64, }; if (set[l->type](i, l + 1, idx)) { LOG("%s: set type %d failed\n", __func__, l->type); return -1; } return 0; } static int op(struct nw_interp *const i, FILE *const f) { varuint32 local_index; if (varuint32_read(f, &local_index)) { LOG("%s: varuint32_read failed\n", __func__); return 1; } else if (i && set_local(local_index, i)) { LOG("%s: set_local failed\n", __func__); return -1; } return 0; } int op_set_local(struct nw_interp *const i) { return op(i, i->f); } int check_set_local(FILE *const f) { return op(NULL, f); }