aboutsummaryrefslogtreecommitdiff
path: root/src/op/variable_access/set_local.c
diff options
context:
space:
mode:
authorXavier Del Campo Romero <xavi.dcr@tutanota.com>2023-11-26 22:43:30 +0100
committerXavier Del Campo Romero <xavi.dcr@tutanota.com>2024-04-21 01:51:24 +0200
commitf25b015e5b668028c34974bbb22faa4105c26690 (patch)
tree28f2b08c17b3585d06694ad74004d0617eadb785 /src/op/variable_access/set_local.c
downloadnanowasm-sync-f25b015e5b668028c34974bbb22faa4105c26690.tar.gz
First commit
Diffstat (limited to 'src/op/variable_access/set_local.c')
-rw-r--r--src/op/variable_access/set_local.c124
1 files changed, 124 insertions, 0 deletions
diff --git a/src/op/variable_access/set_local.c b/src/op/variable_access/set_local.c
new file mode 100644
index 0000000..8cf6440
--- /dev/null
+++ b/src/op/variable_access/set_local.c
@@ -0,0 +1,124 @@
+/*
+ * 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 <nw/log.h>
+#include <nw/ops.h>
+#include <nanowasm/nw.h>
+#include <nw/types.h>
+#include <stdbool.h>
+#include <stddef.h>
+#include <stdio.h>
+#include <stdint.h>
+#include <string.h>
+
+static int set_i32(struct nw_interp *const i, void *const data,
+ const varuint32 idx)
+{
+ const void *const stackptr = (const char *)i->cfg.stack.buf + i->stack_i;
+ const int32_t *const src = (const int32_t *)stackptr - 1;
+ int32_t *const dst = (int32_t *)data + idx;
+
+ *dst = *src;
+ return 0;
+}
+
+static int set_i64(struct nw_interp *const i, void *const data,
+ const varuint32 idx)
+{
+ const void *const stackptr = (const char *)i->cfg.stack.buf + i->stack_i;
+ const int64_t *const src = (const int64_t *)stackptr - 1;
+ int64_t *const dst = (int64_t *)data + idx;
+
+ *dst = *src;
+ return 0;
+}
+
+static int set_f32(struct nw_interp *const i, void *const data,
+ const varuint32 idx)
+{
+ const void *const stackptr = (const char *)i->cfg.stack.buf + i->stack_i;
+ const float *const src = (const float *)stackptr - 1;
+ float *const dst = (float *)data + idx;
+
+ *dst = *src;
+ return 0;
+}
+
+static int set_f64(struct nw_interp *const i, void *const data,
+ const varuint32 idx)
+{
+ const void *const stackptr = (const char *)i->cfg.stack.buf + i->stack_i;
+ const double *const src = (const double *)stackptr - 1;
+ double *const dst = (double *)data + idx;
+
+ *dst = *src;
+ return 0;
+}
+
+static int set_local(const varuint32 local_index, struct nw_interp *const i)
+{
+ struct nw_frame *fr;
+ varuint32 idx;
+
+ for (idx = local_index, fr = i->fp; fr; fr = fr->next)
+ if (idx < fr->n_locals)
+ break;
+
+ if (!fr)
+ {
+ LOG("%s: cannot access local variable %lu\n", __func__,
+ (unsigned long)local_index);
+ i->exception = "local variable index out of bounds";
+ 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[fr->local_type](i, fr + 1, idx))
+ {
+ LOG("%s: set type %d failed\n", __func__, fr->local_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);
+}