mirror of
https://github.com/fluencelabs/assemblyscript
synced 2025-06-14 07:21:30 +00:00
Initial loader that unifies utils and demangle
This commit is contained in:
163
lib/loader/index.js
Normal file
163
lib/loader/index.js
Normal file
@ -0,0 +1,163 @@
|
||||
"use strict";
|
||||
|
||||
/** Instantiates an AssemblyScript module using the specified imports. */
|
||||
function instantiate(module, imports) {
|
||||
|
||||
// Set up the imports object
|
||||
if (!imports) imports = {};
|
||||
if (!imports.env) imports.env = {};
|
||||
if (!imports.env.abort) imports.env.abort = function(mesg, file, line, colm) {
|
||||
mesg = mem ? getString(mesg) : "";
|
||||
file = mem ? getString(file) : "<instantiate>";
|
||||
throw Error("abort: " + mesg + " at " + file + ":" + line + ":" + colm);
|
||||
}
|
||||
|
||||
// Instantiate the module and obtain its (flat) exports
|
||||
var instance = new WebAssembly.Instance(module, imports);
|
||||
var exports = instance.exports;
|
||||
|
||||
// Provide views for all sorts of basic values
|
||||
var mem, I8, U8, I16, U16, I32, U32, F32, F64;
|
||||
|
||||
/** Updates memory views if memory has grown meanwhile. */
|
||||
function checkMem() {
|
||||
// see: https://github.com/WebAssembly/design/issues/1210
|
||||
if (mem !== exports.memory.buffer) {
|
||||
mem = exports.memory.buffer;
|
||||
I8 = new Int8Array(mem);
|
||||
U8 = new Uint8Array(mem);
|
||||
I16 = new Int16Array(mem);
|
||||
U16 = new Uint16Array(mem);
|
||||
I32 = new Int32Array(mem);
|
||||
U32 = new Uint32Array(mem);
|
||||
F32 = new Float32Array(mem);
|
||||
F64 = new Float64Array(mem);
|
||||
}
|
||||
}
|
||||
checkMem();
|
||||
|
||||
/** Allocates a new string in the module's memory and returns its pointer. */
|
||||
function newString(str) {
|
||||
var dataLength = str.length;
|
||||
var ptr = exports.allocate_memory(4 + (dataLength << 1));
|
||||
var dataOffset = (4 + ptr) >>> 1;
|
||||
checkMem();
|
||||
U32[ptr >>> 2] = dataLength;
|
||||
for (let i = 0; i < dataLength; ++i) U16[dataOffset + i] = str.charCodeAt(i);
|
||||
return ptr;
|
||||
}
|
||||
|
||||
/** Gets a string from the module's memory by its pointer. */
|
||||
function getString(ptr) {
|
||||
checkMem();
|
||||
var dataLength = U32[ptr >>> 2];
|
||||
var dataOffset = (ptr + 4) >>> 1;
|
||||
var dataRemain = dataLength;
|
||||
var parts = [];
|
||||
const chunkSize = 1024;
|
||||
while (dataRemain > chunkSize) {
|
||||
let last = U16[dataOffset + chunkSize - 1];
|
||||
let size = last >= 0xD800 && last < 0xDC00 ? chunkSize - 1 : chunkSize;
|
||||
let part = U16.subarray(dataOffset, dataOffset += size);
|
||||
parts.push(String.fromCharCode.apply(String, part));
|
||||
dataRemain -= size;
|
||||
}
|
||||
return parts.join("") + String.fromCharCode.apply(String, U16.subarray(dataOffset, dataOffset + dataRemain));
|
||||
}
|
||||
|
||||
// Demangle exports and provide the usual utility on the prototype
|
||||
return demangle(exports, {
|
||||
get I8() { checkMem(); return I8; },
|
||||
get U8() { checkMem(); return U8; },
|
||||
get I16() { checkMem(); return I16; },
|
||||
get U16() { checkMem(); return U16; },
|
||||
get I32() { checkMem(); return I32; },
|
||||
get U32() { checkMem(); return U32; },
|
||||
get F32() { checkMem(); return F32; },
|
||||
get F64() { checkMem(); return F64; },
|
||||
newString,
|
||||
getString
|
||||
});
|
||||
}
|
||||
|
||||
exports.instantiate = instantiate;
|
||||
|
||||
/** Instantiates an AssemblyScript module from a buffer using the specified imports. */
|
||||
function instantiateBuffer(buffer, imports) {
|
||||
var module = new WebAssembly.Module(buffer);
|
||||
return instantiate(module, imports);
|
||||
}
|
||||
|
||||
exports.instantiateBuffer = instantiateBuffer;
|
||||
|
||||
/** Instantiates an AssemblyScript module from a response using the specified imports. */
|
||||
async function instantiateStreaming(response, imports) {
|
||||
var module = await WebAssembly.compileStreaming(response);
|
||||
return instantiate(module, imports);
|
||||
}
|
||||
|
||||
exports.instantiateStreaming = instantiateStreaming;
|
||||
|
||||
/** Demangles an AssemblyScript module's exports to a friendly object structure. */
|
||||
function demangle(exports, baseModule) {
|
||||
var module = baseModule ? Object.create(baseModule) : {};
|
||||
function hasOwnProperty(elem, prop) {
|
||||
return Object.prototype.hasOwnProperty.call(elem, prop);
|
||||
}
|
||||
for (let internalName in exports) {
|
||||
if (!hasOwnProperty(exports, internalName)) continue;
|
||||
let elem = exports[internalName];
|
||||
let parts = internalName.split(".");
|
||||
let curr = module;
|
||||
while (parts.length > 1) {
|
||||
let part = parts.shift();
|
||||
if (!hasOwnProperty(curr, part)) curr[part] = {};
|
||||
curr = curr[part];
|
||||
}
|
||||
let name = parts[0];
|
||||
let hash = name.indexOf("#");
|
||||
if (hash >= 0) {
|
||||
let className = name.substring(0, hash);
|
||||
let classElem = curr[className];
|
||||
if (typeof classElem === "undefined" || !classElem.prototype) {
|
||||
let ctor = function(...args) {
|
||||
return ctor.wrap(ctor.prototype.constructor(...args));
|
||||
};
|
||||
ctor.prototype = {};
|
||||
ctor.wrap = function(thisValue) {
|
||||
return Object.create(ctor.prototype, { "this": { value: thisValue, writable: false } });
|
||||
};
|
||||
if (classElem) Object.getOwnPropertyNames(classElem).forEach(name =>
|
||||
Object.defineProperty(ctor, name, Object.getOwnPropertyDescriptor(classElem, name))
|
||||
);
|
||||
curr[className] = ctor;
|
||||
}
|
||||
name = name.substring(hash + 1);
|
||||
curr = curr[className].prototype;
|
||||
if (/^(get|set):/.test(name)) {
|
||||
if (!hasOwnProperty(curr, name = name.substring(4))) {
|
||||
let getter = exports[internalName.replace("set:", "get:")];
|
||||
let setter = exports[internalName.replace("get:", "set:")];
|
||||
Object.defineProperty(curr, name, {
|
||||
get: function() { return getter(this.this); },
|
||||
set: function(value) { setter(this.this, value); },
|
||||
enumerable: true
|
||||
});
|
||||
}
|
||||
} else curr[name] = function(...args) { return elem(this.this, ...args); };
|
||||
} else {
|
||||
if (/^(get|set):/.test(name)) {
|
||||
if (!hasOwnProperty(curr, name = name.substring(4))) {
|
||||
Object.defineProperty(curr, name, {
|
||||
get: exports[internalName.replace("set:", "get:")],
|
||||
set: exports[internalName.replace("get:", "set:")],
|
||||
enumerable: true
|
||||
});
|
||||
}
|
||||
} else curr[name] = elem;
|
||||
}
|
||||
}
|
||||
return module;
|
||||
}
|
||||
|
||||
exports.demangle = demangle;
|
Reference in New Issue
Block a user