aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--vendor/wasm/js/runtime.js92
1 files changed, 63 insertions, 29 deletions
diff --git a/vendor/wasm/js/runtime.js b/vendor/wasm/js/runtime.js
index f3df85648..5c7f97fae 100644
--- a/vendor/wasm/js/runtime.js
+++ b/vendor/wasm/js/runtime.js
@@ -13,14 +13,18 @@ function stripNewline(str) {
return str.replace(/\n/, ' ')
}
-const INT_SIZE = 4; // NOTE: set to `8` if the target has 64 bit ints (`wasm64p32` for example).
-const STRING_SIZE = 2*INT_SIZE;
-
class WasmMemoryInterface {
constructor() {
this.memory = null;
this.exports = null;
this.listenerMap = {};
+
+ // Size (in bytes) of the integer type, should be 4 on `js_wasm32` and 8 on `js_wasm64p32`
+ this.intSize = 4;
+ }
+
+ setIntSize(size) {
+ this.intSize = size;
}
setMemory(memory) {
@@ -73,21 +77,21 @@ class WasmMemoryInterface {
loadF32(addr) { return this.mem.getFloat32(addr, true); }
loadF64(addr) { return this.mem.getFloat64(addr, true); }
loadInt(addr) {
- if (INT_SIZE == 8) {
+ if (this.intSize == 8) {
return this.loadI64(addr);
- } else if (INT_SIZE == 4) {
+ } else if (this.intSize == 4) {
return this.loadI32(addr);
} else {
- throw new Error('Unhandled `INT_SIZE`, expected `4` or `8`');
+ throw new Error('Unhandled `intSize`, expected `4` or `8`');
}
};
loadUint(addr) {
- if (INT_SIZE == 8) {
+ if (this.intSize == 8) {
return this.loadU64(addr);
- } else if (INT_SIZE == 4) {
+ } else if (this.intSize == 4) {
return this.loadU32(addr);
} else {
- throw new Error('Unhandled `INT_SIZE`, expected `4` or `8`');
+ throw new Error('Unhandled `intSize`, expected `4` or `8`');
}
};
loadPtr(addr) { return this.loadU32(addr); }
@@ -108,31 +112,43 @@ class WasmMemoryInterface {
storeU32(addr, value) { this.mem.setUint32 (addr, value, true); }
storeI32(addr, value) { this.mem.setInt32 (addr, value, true); }
storeU64(addr, value) {
- this.mem.setUint32(addr + 0, value, true);
- this.mem.setUint32(addr + 4, Math.floor(value / 4294967296), true);
+ this.mem.setUint32(addr + 0, Number(value), true);
+
+ let div = 4294967296;
+ if (typeof value == 'bigint') {
+ div = BigInt(div);
+ }
+
+ this.mem.setUint32(addr + 4, Math.floor(Number(value / div)), true);
}
storeI64(addr, value) {
- this.mem.setUint32(addr + 0, value, true);
- this.mem.setInt32 (addr + 4, Math.floor(value / 4294967296), true);
+ this.mem.setUint32(addr + 0, Number(value), true);
+
+ let div = 4294967296;
+ if (typeof value == 'bigint') {
+ div = BigInt(div);
+ }
+
+ this.mem.setInt32(addr + 4, Math.floor(Number(value / div)), true);
}
storeF32(addr, value) { this.mem.setFloat32(addr, value, true); }
storeF64(addr, value) { this.mem.setFloat64(addr, value, true); }
storeInt(addr, value) {
- if (INT_SIZE == 8) {
+ if (this.intSize == 8) {
this.storeI64(addr, value);
- } else if (INT_SIZE == 4) {
+ } else if (this.intSize == 4) {
this.storeI32(addr, value);
} else {
- throw new Error('Unhandled `INT_SIZE`, expected `4` or `8`');
+ throw new Error('Unhandled `intSize`, expected `4` or `8`');
}
}
storeUint(addr, value) {
- if (INT_SIZE == 8) {
+ if (this.intSize == 8) {
this.storeU64(addr, value);
- } else if (INT_SIZE == 4) {
+ } else if (this.intSize == 4) {
this.storeU32(addr, value);
} else {
- throw new Error('Unhandled `INT_SIZE`, expected `4` or `8`');
+ throw new Error('Unhandled `intSize`, expected `4` or `8`');
}
}
@@ -241,10 +257,11 @@ class WebGLInterface {
}
}
getSource(shader, strings_ptr, strings_length) {
+ const stringSize = this.mem.intSize*2;
let source = "";
for (let i = 0; i < strings_length; i++) {
- let ptr = this.mem.loadPtr(strings_ptr + i*STRING_SIZE);
- let len = this.mem.loadPtr(strings_ptr + i*STRING_SIZE + 4);
+ let ptr = this.mem.loadPtr(strings_ptr + i*stringSize);
+ let len = this.mem.loadPtr(strings_ptr + i*stringSize + 4);
let str = this.mem.loadString(ptr, len);
source += str;
}
@@ -1151,10 +1168,11 @@ class WebGLInterface {
},
TransformFeedbackVaryings: (program, varyings_ptr, varyings_len, bufferMode) => {
this.assertWebGL2();
+ const stringSize = this.mem.intSize*2;
let varyings = [];
for (let i = 0; i < varyings_len; i++) {
- let ptr = this.mem.loadPtr(varyings_ptr + i*STRING_SIZE + 0*4);
- let len = this.mem.loadPtr(varyings_ptr + i*STRING_SIZE + 1*4);
+ let ptr = this.mem.loadPtr(varyings_ptr + i*stringSize + 0*4);
+ let len = this.mem.loadPtr(varyings_ptr + i*stringSize + 1*4);
varyings.push(this.mem.loadString(ptr, len));
}
this.ctx.transformFeedbackVaryings(this.programs[program], varyings, bufferMode);
@@ -1393,7 +1411,7 @@ function odinSetupDefaultImports(wasmMemoryInterface, consoleElement) {
},
"odin_dom": {
init_event_raw: (ep) => {
- const W = 4;
+ const W = wasmMemoryInterface.intSize;
let offset = ep;
let off = (amount, alignment) => {
if (alignment === undefined) {
@@ -1407,6 +1425,13 @@ function odinSetupDefaultImports(wasmMemoryInterface, consoleElement) {
return x;
};
+ let align = (alignment) => {
+ const modulo = offset & (alignment-1);
+ if (modulo != 0) {
+ offset += alignment - modulo
+ }
+ };
+
let wmi = wasmMemoryInterface;
let e = event_temp_data.event;
@@ -1427,10 +1452,12 @@ function odinSetupDefaultImports(wasmMemoryInterface, consoleElement) {
wmi.storeU32(off(4), 0);
}
- wmi.storeUint(off(W), event_temp_data.id_ptr);
+ align(W);
+
+ wmi.storeI32(off(W), event_temp_data.id_ptr);
wmi.storeUint(off(W), event_temp_data.id_len);
- wmi.storeUint(off(W), 0); // padding
+ align(8);
wmi.storeF64(off(8), e.timeStamp*1e-3);
wmi.storeU8(off(1), e.eventPhase);
@@ -1442,7 +1469,7 @@ function odinSetupDefaultImports(wasmMemoryInterface, consoleElement) {
wmi.storeU8(off(1), !!e.isComposing);
wmi.storeU8(off(1), !!e.isTrusted);
- let base = off(0, 8);
+ align(8);
if (e instanceof WheelEvent) {
wmi.storeF64(off(8), e.deltaX);
wmi.storeF64(off(8), e.deltaY);
@@ -1689,8 +1716,15 @@ function odinSetupDefaultImports(wasmMemoryInterface, consoleElement) {
};
};
-async function runWasm(wasmPath, consoleElement, extraForeignImports) {
- let wasmMemoryInterface = new WasmMemoryInterface();
+/**
+ * @param {string} wasmPath - Path to the WASM module to run
+ * @param {?HTMLPreElement} consoleElement - Optional console/pre element to append output to, in addition to the console
+ * @param {any} extraForeignImports - Imports, in addition to the default runtime to provide the module
+ * @param {?int} intSize - Size (in bytes) of the integer type, should be 4 on `js_wasm32` and 8 on `js_wasm64p32`
+ */
+async function runWasm(wasmPath, consoleElement, extraForeignImports, intSize = 4) {
+ const wasmMemoryInterface = new WasmMemoryInterface();
+ wasmMemoryInterface.setIntSize(intSize);
let imports = odinSetupDefaultImports(wasmMemoryInterface, consoleElement);
let exports = {};