Webassembly从wasm调用JavaScript方法,即在c代码中调用

我正在使用webassembly,到目前为止,我能够管理emscripten将我的测试c项目编译为wasm文件
em提供了2个文件,即

mainTest.js mainTest.wasm

当我在我的html页面加载mainTest.js然后我得到一个名为的Java脚本对象
“模块”.

我确实找到了如何从javascript调用c / wasm方法,例如:

var myTestInteger = Module._callMyTestMethod();

并从中读取字符串
Module.wasmMemory.buffer
,但我不明白如何从C代码调用javascript.

即我希望能够像这样做:

#ifdef __cplusplus
extern "C" {
#endif
extern void testExternJSMethod();

int main() 
{
    cout << " Hello From my Test1 !" << endl;

    testExternJSMethod();
    return 0;


}
int EMSCRIPTEN_KEEPALIVE callMyTestMethod(){
    return 26;
}
#ifdef __cplusplus
}
#endif

我正在加载另一个名为utils.js的js文件中的my js方法testExternMethod

function testExternMethod() {
  console.log("Hello from testExternMethod!" + )
}

在这里我想从c调用JavaScript testExternJSMethod.

当我在Firefox中运行页面时,在调试器控制台中获得“-1”.

那么我在这种情况下缺少什么?不幸的是,Mozilla文档只给出了那些S表达式而不是C表示的例子.

我在例子中缺少什么?即在c中我用extern关键字定义了方法,即

extern void testExternJSMethod();

但我觉得这不是我必须做的全部.

我相信我应该以某种方式将该java脚本方法以某种方式链接到模块,但我不知道如何.
Module.asm给了我出口.哪种方法调用应该给我进口?因为我相信这个_testExternJSMethod()应​​该在某些导入方法中,我无法弄清楚如何实现它.

解决方法:

我不完全确定您的用例,但是您缺少能够使用函数testExternalJSMethod的重要步骤.你有两个选择:

选项1 – 图书馆

1 – 用c / c定义你的功能.

extern void testExternalJSMethod();

2 – 创建一个名为myLibrary.js的文件

3 – 需要使用以下代码将JS函数添加到库文件中的LibraryManager:

function makeAlert(text) {
    alert(text);
}

if (typeof mergeInto !== 'undefined') mergeInto(LibraryManager.library, {
    testExternalJSMethod: function() {
        makeAlert("Hello world");
    }
});

4 – 如果testExternalJSMethod依赖于其自身范围之外的任何内容(例如,上面的makeAlert),请确保在html页面中包含该脚本

<script async type="text/javascript" src="myLibrary.js"></script>

5 – 在emcc命令中添加选项–js-library,并在myLibrary.js的相对路径后立即添加

emcc ... --js-library myLibrary.js

选项2 – 传递指针

1 – 在c / c中定义你的javascript函数类型

typedef void testExternalJSMethod()

2 – 无论你想在哪里使用这个函数,接受一个int参数,它将是函数指针,并将指针强制转换为你的函数

void passFnPointer(int ptr) {
    ((testExternalJSMethod*)ptr)();
}

3 – 使用emscripten的addFunction()并存储其返回值(指针在c / c中)

var fnPtr = Module.addFunction(function () {
    alert("You called testExternalJSMethod");
});

4 – 使用步骤3中存储的指针值传递给我们的函数passFnPointer

var passFnPointer = Module.cwrap('passFnPointer', 'undefined', ['number']);
passFnPointer(fnPtr);

5 – 向emcc命令添加选项-s RESERVED_FUNCTION_POINTERS

emcc ... -s RESERVED_FUNCTION_POINTERS=10

结论

在完成其中一个选项中的所有步骤之后,您将能够通过c / c调用testExternalJSMethod()而无需使用emscripten_run_script()或EM_ASM()

上一篇:shell脚本一键部署cobbler服务器_20210715


下一篇:Java SE之向上转型与向下转型