Merge pull request #155 from syumai/update-wasm-exec-js

update wasm_exec.js contents
This commit is contained in:
syumai 2025-02-23 00:02:59 +09:00 committed by GitHub
commit 276ef2efb5
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
8 changed files with 60 additions and 35 deletions

View File

@ -70,7 +70,7 @@ First, please install the following tools:
* Node.js (and npm) * Node.js (and npm)
* [wrangler](https://developers.cloudflare.com/workers/wrangler/) * [wrangler](https://developers.cloudflare.com/workers/wrangler/)
- You can install it by running `npm install -g wrangler`. - You can install it by running `npm install -g wrangler`.
* Go 1.21.3 or later * Go 1.24.0 or later
* [gonew](https://pkg.go.dev/golang.org/x/tools/cmd/gonew) * [gonew](https://pkg.go.dev/golang.org/x/tools/cmd/gonew)
- You can install it by running `go install golang.org/x/tools/cmd/gonew@latest` - You can install it by running `go install golang.org/x/tools/cmd/gonew@latest`

View File

@ -4,7 +4,7 @@ dev:
.PHONY: build .PHONY: build
build: build:
go run github.com/syumai/workers/cmd/workers-assets-gen@v0.23.1 -mode=go go run github.com/syumai/workers/cmd/workers-assets-gen@v0.28.1 -mode=go
GOOS=js GOARCH=wasm go build -o ./build/app.wasm . GOOS=js GOARCH=wasm go build -o ./build/app.wasm .
.PHONY: deploy .PHONY: deploy

View File

@ -4,7 +4,7 @@ dev:
.PHONY: build .PHONY: build
build: build:
go run github.com/syumai/workers/cmd/workers-assets-gen@v0.23.1 go run github.com/syumai/workers/cmd/workers-assets-gen@v0.28.1
tinygo build -o ./build/app.wasm -target wasm -no-debug ./... tinygo build -o ./build/app.wasm -target wasm -no-debug ./...
.PHONY: deploy .PHONY: deploy

View File

@ -4,7 +4,7 @@ dev:
.PHONY: build .PHONY: build
build: build:
go run github.com/syumai/workers/cmd/workers-assets-gen@v0.23.1 go run github.com/syumai/workers/cmd/workers-assets-gen@v0.28.1
tinygo build -o ./build/app.wasm -target wasm -no-debug ./... tinygo build -o ./build/app.wasm -target wasm -no-debug ./...
.PHONY: deploy .PHONY: deploy

View File

@ -4,7 +4,7 @@ dev:
.PHONY: build .PHONY: build
build: build:
go run github.com/syumai/workers/cmd/workers-assets-gen@v0.23.1 -mode=go go run github.com/syumai/workers/cmd/workers-assets-gen@v0.28.1 -mode=go
GOOS=js GOARCH=wasm go build -o ./build/app.wasm . GOOS=js GOARCH=wasm go build -o ./build/app.wasm .
.PHONY: deploy .PHONY: deploy

View File

@ -4,7 +4,7 @@ dev:
.PHONY: build .PHONY: build
build: build:
go run github.com/syumai/workers/cmd/workers-assets-gen@v0.23.1 go run github.com/syumai/workers/cmd/workers-assets-gen@v0.28.1
tinygo build -o ./build/app.wasm -target wasm -no-debug ./... tinygo build -o ./build/app.wasm -target wasm -no-debug ./...
.PHONY: deploy .PHONY: deploy

View File

@ -14,7 +14,7 @@
if (!globalThis.fs) { if (!globalThis.fs) {
let outputBuf = ""; let outputBuf = "";
globalThis.fs = { globalThis.fs = {
constants: { O_WRONLY: -1, O_RDWR: -1, O_CREAT: -1, O_TRUNC: -1, O_APPEND: -1, O_EXCL: -1 }, // unused constants: { O_WRONLY: -1, O_RDWR: -1, O_CREAT: -1, O_TRUNC: -1, O_APPEND: -1, O_EXCL: -1, O_DIRECTORY: -1 }, // unused
writeSync(fd, buf) { writeSync(fd, buf) {
outputBuf += decoder.decode(buf); outputBuf += decoder.decode(buf);
const nl = outputBuf.lastIndexOf("\n"); const nl = outputBuf.lastIndexOf("\n");
@ -73,6 +73,14 @@
} }
} }
if (!globalThis.path) {
globalThis.path = {
resolve(...pathSegments) {
return pathSegments.join("/");
}
}
}
if (!globalThis.crypto) { if (!globalThis.crypto) {
throw new Error("globalThis.crypto is not available, polyfill required (crypto.getRandomValues only)"); throw new Error("globalThis.crypto is not available, polyfill required (crypto.getRandomValues only)");
} }
@ -208,10 +216,16 @@
return decoder.decode(new DataView(this._inst.exports.mem.buffer, saddr, len)); return decoder.decode(new DataView(this._inst.exports.mem.buffer, saddr, len));
} }
const testCallExport = (a, b) => {
this._inst.exports.testExport0();
return this._inst.exports.testExport(a, b);
}
const timeOrigin = Date.now() - performance.now(); const timeOrigin = Date.now() - performance.now();
this.importObject = { this.importObject = {
_gotest: { _gotest: {
add: (a, b) => a + b, add: (a, b) => a + b,
callExport: testCallExport,
}, },
gojs: { gojs: {
// Go's SP does not change as long as no Go code is running. Some operations (e.g. calls, getters and setters) // Go's SP does not change as long as no Go code is running. Some operations (e.g. calls, getters and setters)

View File

@ -32,7 +32,7 @@
/* /*
if (!global.fs && global.require) { if (!global.fs && global.require) {
global.fs = require("fs"); global.fs = require("node:fs");
} }
*/ */
@ -106,7 +106,7 @@
/* /*
if (!global.crypto) { if (!global.crypto) {
const nodeCrypto = require("crypto"); const nodeCrypto = require("node:crypto");
global.crypto = { global.crypto = {
getRandomValues(b) { getRandomValues(b) {
nodeCrypto.randomFillSync(b); nodeCrypto.randomFillSync(b);
@ -126,11 +126,11 @@
/* /*
if (!global.TextEncoder) { if (!global.TextEncoder) {
global.TextEncoder = require("util").TextEncoder; global.TextEncoder = require("node:util").TextEncoder;
} }
if (!global.TextDecoder) { if (!global.TextDecoder) {
global.TextDecoder = require("util").TextDecoder; global.TextDecoder = require("node:util").TextDecoder;
} }
*/ */
@ -140,6 +140,7 @@
const decoder = new TextDecoder("utf-8"); const decoder = new TextDecoder("utf-8");
let reinterpretBuf = new DataView(new ArrayBuffer(8)); let reinterpretBuf = new DataView(new ArrayBuffer(8));
var logLine = []; var logLine = [];
const wasmExit = {}; // thrown to exit via proc_exit (not an error)
global.Go = class { global.Go = class {
constructor() { constructor() {
@ -278,14 +279,11 @@
fd_close: () => 0, // dummy fd_close: () => 0, // dummy
fd_fdstat_get: () => 0, // dummy fd_fdstat_get: () => 0, // dummy
fd_seek: () => 0, // dummy fd_seek: () => 0, // dummy
"proc_exit": (code) => { proc_exit: (code) => {
if (global.process) { this.exited = true;
// Node.js this.exitCode = code;
process.exit(code); this._resolveExitPromise();
} else { throw wasmExit;
// Can't exit in a browser.
throw 'trying to exit with code ' + code;
}
}, },
random_get: (bufPtr, bufLen) => { random_get: (bufPtr, bufLen) => {
crypto.getRandomValues(loadSlice(bufPtr, bufLen)); crypto.getRandomValues(loadSlice(bufPtr, bufLen));
@ -301,7 +299,14 @@
// func sleepTicks(timeout float64) // func sleepTicks(timeout float64)
"runtime.sleepTicks": (timeout) => { "runtime.sleepTicks": (timeout) => {
// Do not sleep, only reactivate scheduler after the given timeout. // Do not sleep, only reactivate scheduler after the given timeout.
setTimeout(this._inst.exports.go_scheduler, timeout); setTimeout(() => {
if (this.exited) return;
try {
this._inst.exports.go_scheduler();
} catch (e) {
if (e !== wasmExit) throw e;
}
}, timeout);
}, },
// func finalizeRef(v ref) // func finalizeRef(v ref)
@ -481,23 +486,25 @@
this._ids = new Map(); // mapping from JS values to reference ids this._ids = new Map(); // mapping from JS values to reference ids
this._idPool = []; // unused ids that have been garbage collected this._idPool = []; // unused ids that have been garbage collected
this.exited = false; // whether the Go program has exited this.exited = false; // whether the Go program has exited
this.exitCode = 0;
const mem = new DataView(this._inst.exports.memory.buffer) if (this._inst.exports._start) {
let exitPromise = new Promise((resolve, reject) => {
while (true) { this._resolveExitPromise = resolve;
const callbackPromise = new Promise((resolve) => {
this._resolveCallbackPromise = () => {
if (this.exited) {
throw new Error("bad callback: Go program has already exited");
}
setTimeout(resolve, 0); // make sure it is asynchronous
};
}); });
this._inst.exports._start();
if (this.exited) { // Run program, but catch the wasmExit exception that's thrown
break; // to return back here.
try {
this._inst.exports._start();
} catch (e) {
if (e !== wasmExit) throw e;
} }
await callbackPromise;
await exitPromise;
return this.exitCode;
} else {
this._inst.exports._initialize();
} }
} }
@ -505,7 +512,11 @@
if (this.exited) { if (this.exited) {
throw new Error("Go program has already exited"); throw new Error("Go program has already exited");
} }
this._inst.exports.resume(); try {
this._inst.exports.resume();
} catch (e) {
if (e !== wasmExit) throw e;
}
if (this.exited) { if (this.exited) {
this._resolveExitPromise(); this._resolveExitPromise();
} }