mirror of
https://github.com/remko/waforth
synced 2024-12-27 09:59:29 +01:00
standalone: Add wabt version
This commit is contained in:
parent
d999f24704
commit
0751ed383a
4 changed files with 316 additions and 0 deletions
5
src/standalone/wabt/.gitignore
vendored
Normal file
5
src/standalone/wabt/.gitignore
vendored
Normal file
|
@ -0,0 +1,5 @@
|
|||
/waforth
|
||||
/waforth_core.h
|
||||
/wasm-micro-runtime
|
||||
/wasmtime-*
|
||||
/*.tgz
|
80
src/standalone/wabt/Makefile
Normal file
80
src/standalone/wabt/Makefile
Normal file
|
@ -0,0 +1,80 @@
|
|||
.DEFAULT_GOAL := all
|
||||
|
||||
VERSION?=$(shell cat ../../../package.json | grep '"version"' | sed -e 's/.*:.*"\(.*\)".*/\1/')
|
||||
|
||||
ifeq ($(OS),Windows_NT)
|
||||
CC=gcc
|
||||
LIBS=
|
||||
else
|
||||
ifeq (, $(shell which gtar))
|
||||
TAR := tar
|
||||
else
|
||||
# bsd-tar corrupts files on GitHub: https://github.com/actions/virtual-environments/issues/2619
|
||||
TAR := gtar
|
||||
endif
|
||||
UNAME_S=$(shell uname -s)
|
||||
UNAME_P=$(shell uname -p)
|
||||
WABT_DIR = /usr/local/wabt
|
||||
ifeq ($(UNAME_S),Darwin)
|
||||
CXXFLAGS = -std=c++17
|
||||
ifeq ($(UNAME_P),arm)
|
||||
PACKAGE_SUFFIX=arm64-macos
|
||||
else
|
||||
PACKAGE_SUFFIX=x86_64-macos
|
||||
endif
|
||||
else
|
||||
LIBS=-lpthread -lm -ldl
|
||||
PACKAGE_SUFFIX=x86_64-linux
|
||||
endif
|
||||
package: waforth
|
||||
$(TAR) czf waforth-v$(VERSION)-$(PACKAGE_SUFFIX).tgz waforth
|
||||
endif
|
||||
WABT_INCLUDE_DIR := $(WABT_DIR)/include
|
||||
WABT_LIB_DIR := $(WABT_DIR)/lib
|
||||
CXXFLAGS:= -I$(WABT_INCLUDE_DIR) $(CXXFLAGS)
|
||||
LIBS:=$(WABT_LIB_DIR)/libwabt.a $(LIBS)
|
||||
|
||||
ifeq ($(DEBUG),1)
|
||||
CXXFLAGS := $(CXXFLAGS) -g
|
||||
else
|
||||
CXXFLAGS := $(CXXFLAGS) -O2
|
||||
endif
|
||||
|
||||
################################################
|
||||
|
||||
BIN2H=../../../scripts/bin2h
|
||||
WAT2WASM=wat2wasm
|
||||
WAT2WASM_FLAGS=--debug-names
|
||||
|
||||
CXXFLAGS:=-DVERSION='"$(VERSION)"' $(CXXFLAGS)
|
||||
OBJECTS=main.o $(RESOURCE_OBJECTS)
|
||||
|
||||
all: waforth
|
||||
|
||||
waforth: $(OBJECTS)
|
||||
$(CXX) $(CXXFLAGS) -o $@ $(OBJECTS) $(LDFLAGS) $(LIBS)
|
||||
|
||||
main.o: waforth_core.h
|
||||
|
||||
waforth_core.wasm: ../../waforth.wat
|
||||
$(WAT2WASM) $(WAT2WASM_FLAGS) -o $@ $<
|
||||
|
||||
waforth_core.h: waforth_core.wasm
|
||||
$(BIN2H) $< $@
|
||||
|
||||
.PHONY: package
|
||||
|
||||
.PHONY: check
|
||||
check:
|
||||
-rm -f test.out
|
||||
echo ": MAIN 1 2 3 + .S ; MAIN" | ./waforth | tee test.out
|
||||
grep "1 5 ok" test.out
|
||||
rm test.out
|
||||
./waforth ../../examples/hello.fs | tee test.out
|
||||
grep "Hello, Forth" test.out
|
||||
rm test.out
|
||||
|
||||
|
||||
.PHONY: clean
|
||||
clean:
|
||||
-rm -f waforth_core.wasm waforth_core.h $(OBJECTS) waforth *.exe *.tgz *.zip test.out
|
2
src/standalone/wabt/README.md
Normal file
2
src/standalone/wabt/README.md
Normal file
|
@ -0,0 +1,2 @@
|
|||
This is an experimental version of the standalone shell that uses the WABT interpreter.
|
||||
When WABT implements the most recent version of the WASM C API, this can go away.
|
229
src/standalone/wabt/main.cpp
Normal file
229
src/standalone/wabt/main.cpp
Normal file
|
@ -0,0 +1,229 @@
|
|||
#if defined(WIN32) || defined(_WIN32) || defined(__WIN32__) || defined(__NT__)
|
||||
#include <windows.h>
|
||||
#else
|
||||
#include <termios.h>
|
||||
#endif
|
||||
|
||||
#include <cstdio>
|
||||
|
||||
#include <wabt/binary-reader.h>
|
||||
#include <wabt/interp/binary-reader-interp.h>
|
||||
#include <wabt/interp/interp-util.h>
|
||||
#include <wabt/interp/interp.h>
|
||||
#include <wabt/result.h>
|
||||
|
||||
#include "waforth_core.h"
|
||||
|
||||
namespace wabti = wabt::interp;
|
||||
|
||||
#ifndef VERSION
|
||||
#define VERSION "dev"
|
||||
#endif
|
||||
|
||||
#define CORE_RUN_EXPORT_INDEX 0
|
||||
#define CORE_TABLE_EXPORT_INDEX 1
|
||||
#define CORE_MEMORY_EXPORT_INDEX 2
|
||||
#define CORE_ERROR_EXPORT_INDEX 9
|
||||
|
||||
#define ERR_UNKNOWN 0x1
|
||||
#define ERR_QUIT 0x2
|
||||
#define ERR_ABORT 0x3
|
||||
#define ERR_EOI 0x4
|
||||
#define ERR_BYE 0x5
|
||||
|
||||
static wabti::Store store;
|
||||
static std::unique_ptr<wabt::FileStream> stderrStream;
|
||||
static wabt::Features features;
|
||||
static wabti::Memory::Ptr memory;
|
||||
static wabti::Table::Ptr table;
|
||||
static wabt::Errors errors;
|
||||
|
||||
FILE *input;
|
||||
|
||||
wabt::Result emit_cb(wabti::Thread &thread, const wabti::Values ¶ms, wabti::Values &results, wabti::Trap::Ptr *trap) {
|
||||
putchar(params[0].Get<wabti::s32>());
|
||||
return wabt::Result::Ok;
|
||||
}
|
||||
|
||||
wabt::Result key_cb(wabti::Thread &thread, const wabti::Values ¶ms, wabti::Values &results, wabti::Trap::Ptr *trap) {
|
||||
#if defined(WIN32) || defined(_WIN32) || defined(__WIN32__) || defined(__NT__)
|
||||
HANDLE h = GetStdHandle(STD_INPUT_HANDLE);
|
||||
if (h == NULL) {
|
||||
return trap_from_string("no console");
|
||||
}
|
||||
DWORD mode;
|
||||
GetConsoleMode(h, &mode);
|
||||
SetConsoleMode(h, mode & ~(ENABLE_LINE_INPUT | ENABLE_ECHO_INPUT));
|
||||
TCHAR ch = 0;
|
||||
DWORD cc;
|
||||
ReadConsole(h, &ch, 1, &cc, NULL);
|
||||
SetConsoleMode(h, mode);
|
||||
#else
|
||||
struct termios old, current;
|
||||
tcgetattr(0, &old);
|
||||
current = old;
|
||||
current.c_lflag &= ~ICANON;
|
||||
current.c_lflag &= ~ECHO;
|
||||
tcsetattr(0, TCSANOW, ¤t);
|
||||
char ch = getchar();
|
||||
tcsetattr(0, TCSANOW, &old);
|
||||
#endif
|
||||
results[0].Set((wabti::u32)ch);
|
||||
return wabt::Result::Ok;
|
||||
}
|
||||
|
||||
wabt::Result read_cb(wabti::Thread &thread, const wabti::Values ¶ms, wabti::Values &results, wabti::Trap::Ptr *trap) {
|
||||
auto addr = (char *)memory->UnsafeData() + params[0].Get<wabti::s32>();
|
||||
auto size = params[1].Get<wabti::s32>();
|
||||
*addr = 0;
|
||||
fgets(addr, size, input);
|
||||
int n = strlen(addr);
|
||||
results[0].Set((wabti::u32)n);
|
||||
return wabt::Result::Ok;
|
||||
}
|
||||
|
||||
wabt::Result load_cb(wabti::Thread &thread, const wabti::Values ¶ms, wabti::Values &results, wabti::Trap::Ptr *trap) {
|
||||
auto addr = params[0].Get<wabti::s32>();
|
||||
auto size = params[1].Get<wabti::s32>();
|
||||
wabti::ModuleDesc desc;
|
||||
CHECK_RESULT(wabti::ReadBinaryInterp("word.wasm", memory->UnsafeData() + addr, size, wabt::ReadBinaryOptions(features, nullptr, true, true, true),
|
||||
&errors, &desc));
|
||||
auto mod = wabti::Module::New(store, desc);
|
||||
wabti::RefVec imports = {table.ref(), memory.ref()};
|
||||
auto modi = wabti::Instance::Instantiate(store, mod.ref(), imports, trap);
|
||||
if (!modi) {
|
||||
printf("error instantiating word module\n");
|
||||
return wabt::Result::Error;
|
||||
;
|
||||
}
|
||||
return wabt::Result::Ok;
|
||||
}
|
||||
|
||||
wabt::Result call_cb(wabti::Thread &thread, const wabti::Values ¶ms, wabti::Values &results, wabti::Trap::Ptr *trap) {
|
||||
printf("`call` is not available in standalone\n");
|
||||
return wabt::Result::Error;
|
||||
}
|
||||
|
||||
wabt::Result run(bool interactive) {
|
||||
stderrStream = wabt::FileStream::CreateStderr();
|
||||
|
||||
// Load core module
|
||||
wabti::ModuleDesc desc;
|
||||
CHECK_RESULT(wabti::ReadBinaryInterp("waforth.wasm", waforth_core, sizeof(waforth_core),
|
||||
wabt::ReadBinaryOptions(features, nullptr, true, true, true), &errors, &desc));
|
||||
auto core = wabti::Module::New(store, desc);
|
||||
|
||||
// Core Exports
|
||||
wabti::Func::Ptr errorFn;
|
||||
wabti::Func::Ptr runFn;
|
||||
|
||||
// Bind core imports
|
||||
wabti::RefVec imports;
|
||||
for (auto &&import : core->desc().imports) {
|
||||
if (import.type.type->kind == wabti::ExternKind::Func && import.type.module == "shell") {
|
||||
auto ft = *wabt::cast<wabti::FuncType>(import.type.type.get());
|
||||
wabti::HostFunc::Callback cb;
|
||||
if (import.type.name == "emit") {
|
||||
cb = emit_cb;
|
||||
} else if (import.type.name == "read") {
|
||||
cb = read_cb;
|
||||
} else if (import.type.name == "key") {
|
||||
cb = key_cb;
|
||||
} else if (import.type.name == "load") {
|
||||
cb = load_cb;
|
||||
} else if (import.type.name == "call") {
|
||||
cb = call_cb;
|
||||
} else {
|
||||
printf("Unknown import: %s\n", import.type.name.c_str());
|
||||
return wabt::Result::Error;
|
||||
}
|
||||
auto func = wabti::HostFunc::New(store, ft, cb);
|
||||
imports.push_back(func.ref());
|
||||
continue;
|
||||
}
|
||||
imports.push_back(wabti::Ref::Null);
|
||||
}
|
||||
|
||||
// Instantiate module
|
||||
wabti::Trap::Ptr trap;
|
||||
auto corei = wabti::Instance::Instantiate(store, core.ref(), imports, &trap);
|
||||
if (!corei) {
|
||||
printf("error instantiating module\n");
|
||||
if (trap) {
|
||||
wabti::WriteTrap(stderrStream.get(), " error ", trap);
|
||||
}
|
||||
return wabt::Result::Error;
|
||||
}
|
||||
|
||||
// Load exports
|
||||
for (auto &&export_ : core->desc().exports) {
|
||||
if (export_.type.type->kind == wabt::ExternalKind::Memory) {
|
||||
memory = store.UnsafeGet<wabti::Memory>(corei->memories()[export_.index]);
|
||||
} else if (export_.type.type->kind == wabt::ExternalKind::Table) {
|
||||
table = store.UnsafeGet<wabti::Table>(corei->tables()[export_.index]);
|
||||
} else if (export_.type.name == "run") {
|
||||
runFn = store.UnsafeGet<wabti::Func>(corei->funcs()[export_.index]);
|
||||
} else if (export_.type.name == "error") {
|
||||
errorFn = store.UnsafeGet<wabti::Func>(corei->funcs()[export_.index]);
|
||||
}
|
||||
}
|
||||
|
||||
// Run
|
||||
wabti::Values runParams = {wabti::Value::Make(interactive ? 0 : 1)};
|
||||
wabti::Values runResults;
|
||||
wabti::Values errorParams;
|
||||
wabti::Values errorResults;
|
||||
for (int stopped = false; !stopped;) {
|
||||
auto runRes = runFn->Call(store, runParams, runResults, &trap, nullptr);
|
||||
CHECK_RESULT(errorFn->Call(store, errorParams, errorResults, &trap, nullptr));
|
||||
switch (errorResults[0].Get<wabti::s32>()) {
|
||||
case ERR_QUIT:
|
||||
case ERR_ABORT:
|
||||
assert(!Succeeded(runRes));
|
||||
break;
|
||||
case ERR_EOI:
|
||||
assert(Succeeded(runRes));
|
||||
stopped = true;
|
||||
break;
|
||||
case ERR_BYE:
|
||||
assert(!Succeeded(runRes));
|
||||
stopped = true;
|
||||
break;
|
||||
case ERR_UNKNOWN:
|
||||
assert(!Succeeded(runRes));
|
||||
if (trap) {
|
||||
wabti::WriteTrap(stderrStream.get(), " error ", trap);
|
||||
} else {
|
||||
printf("unknown error\n");
|
||||
}
|
||||
break;
|
||||
default:
|
||||
printf("unknown error code\n");
|
||||
if (trap) {
|
||||
wabti::WriteTrap(stderrStream.get(), " error ", trap);
|
||||
}
|
||||
assert(false);
|
||||
}
|
||||
}
|
||||
return wabt::Result::Ok;
|
||||
}
|
||||
|
||||
int main(int argc, char *argv[]) {
|
||||
if (argc >= 2) {
|
||||
input = fopen(argv[1], "r");
|
||||
} else {
|
||||
input = stdin;
|
||||
}
|
||||
|
||||
if (input == stdin) {
|
||||
printf("WAForth (" VERSION ")\n");
|
||||
}
|
||||
|
||||
auto result = run(input == stdin);
|
||||
|
||||
if (input != stdin) {
|
||||
fclose(input);
|
||||
}
|
||||
|
||||
return Succeeded(result) ? 0 : 1;
|
||||
}
|
Loading…
Reference in a new issue