我想使用由以前的Module实例创建的WebAssembly.Memory对象实例化一个WebAssembly模块(所有这些都使用emscripten的粘合代码),是否可行?
答案 0 :(得分:0)
您可以自己创建一个内存对象,然后在实例化一个或多个WASM模块期间将其注入,要容易一些。我设法通过覆盖instantiateWasm来使用Emscripten。
首先创建一个将由WASM实例共享的内存:
var TOTAL_MEMORY = 16777216;
var WASM_PAGE_SIZE = 65536;
var wasmMemory = new WebAssembly.Memory({ 'initial': TOTAL_MEMORY / WASM_PAGE_SIZE, 'maximum': TOTAL_MEMORY / WASM_PAGE_SIZE });
var buffer = wasmMemory.buffer;
然后实现一个自定义实例化器,将该实例注入到导入对象中:
function wasmInstantiator(wasmBinary) {
return (info, receiveInstance) => {
var importObject = Object.assign({}, info);
importObject['env']['memory'] = wasmMemory;
WebAssembly.instantiateStreaming(fetch(wasmBinary, { credentials: 'same-origin' }), importObject)
.then((output) => { receiveInstance(output['instance']); },
(err) => { console.error('wasm streaming compile failed: ' + err);});
return {};
};
};
现在您可以使用相同的内存实例化两个模块。在这里,我建议使用Modularize
(请参阅documentation和settings.js,这意味着在-s Modularize=1 -s EXPORT_NAME='MY_MODULE_NAME_1'
命令行中添加emcc
。Emscripten将创建一个名为{{ 1}}接受一个MY_MODULE_NAME_1
对象,您可以在其中覆盖某些元素,例如实例化器和内存。
Module
有了这些类似承诺的WASM实例,您现在可以与两个模块进行交互:
var createInstance1 = MY_MODULE_NAME_1( { instantiateWasm: wasmInstantiator('module1.wasm') , TOTAL_MEMORY, buffer } );
var createInstance2 = MY_MODULE_NAME_2( { instantiateWasm: wasmInstantiator('module2.wasm') , TOTAL_MEMORY, buffer } );
但是-这里有一个严重的警告:如果两个模块都试图在同一内存中分配动态内存,则它们会覆盖彼此分配的缓冲区!到目前为止,我还没有找到如何解决此问题的方法:-(。
请参见This repo中的工作示例。