使用 Emscripten 将C++代码编译为独立的 Wasm 会导致 .wasm 需要某些导入函数



我正在尝试使用Emscripten作为学习机会为Wasm编译一个完整的C++程序。以下命令确实会生成完全有效且有效的 HTML+JS+Wasm:

emcc main.cpp [...libraries] -Oz --std=c++17 -o output.html
-s "EXPORTED_FUNCTIONS=['_solve']"

但是,当我尝试生成独立的 Wasm 二进制文件时,我遇到了以下错误,即:

emcc main.cpp [...libraries] -Oz --std=c++17 -o output.wasm
-s "EXPORTED_FUNCTIONS=['_solve']"

并在 JS 中实例化它:

// `buffer` is pre-defined and contains the wasm binary in a typed array.
const module = new WebAssembly.Module(buffer);
// `imports` is an object of values and functions.
new WebAssembly.Instance(module, imports);

import function env:_assert 必须是可调用的(评估 'new WebAssembly.Instance(module, imports('(

检查生成的 Wasm 二进制文件会发现它需要主机环境提供相当长的函数和变量列表:

(import "env" "_abort" (func (;0;) (type 12)))
(import "env" "___assert_fail" (func (;1;) (type 13)))
(import "env" "abort" (func (;2;) (type 7)))
(import "env" "_llvm_stacksave" (func (;3;) (type 21)))
(import "env" "_llvm_stackrestore" (func (;4;) (type 7)))
(import "env" "_getenv" (func (;5;) (type 2)))
(import "env" "___setErrNo" (func (;6;) (type 7)))
(import "env" "___unlock" (func (;7;) (type 7)))
(import "env" "___syscall140" (func (;8;) (type 1)))
(import "env" "___map_file" (func (;9;) (type 1)))
(import "env" "___lock" (func (;10;) (type 7)))
(import "env" "___cxa_uncaught_exceptions" (func (;11;) (type 21)))
(import "env" "___cxa_throw" (func (;12;) (type 6)))
(import "env" "___cxa_pure_virtual" (func (;13;) (type 12)))
(import "env" "abortOnCannotGrowMemory" (func (;14;) (type 2)))
(import "env" "_strftime_l" (func (;15;) (type 10)))
(import "env" "_pthread_cond_wait" (func (;16;) (type 1)))
(import "env" "_llvm_trap" (func (;17;) (type 12)))
(import "env" "___cxa_allocate_exception" (func (;18;) (type 2)))
(import "env" "_emscripten_resize_heap" (func (;19;) (type 2)))
(import "env" "_emscripten_memcpy_big" (func (;20;) (type 3)))
(import "env" "_emscripten_get_heap_size" (func (;21;) (type 21)))
(import "env" "___wasi_fd_write" (func (;22;) (type 9)))
(import "env" "___syscall91" (func (;23;) (type 1)))
(import "env" "___syscall6" (func (;24;) (type 1)))
(import "env" "__table_base" (global (;0;) i32))
(import "env" "DYNAMICTOP_PTR" (global (;1;) i32))
(import "global" "NaN" (global (;2;) f64))
(import "global" "Infinity" (global (;3;) f64))
(import "env" "memory" (memory (;0;) 256 256))
(import "env" "table" (table (;0;) 725 725 funcref))

我可以指示Emscripten将源代码编译为完整的JS + Wasm捆绑包或纯JS文件,但是:

  • 一个完整的 JS + Wasm 捆绑包:Emscripten 异步加载 WASM 二进制文件,这在我的情况下是有问题的(确切原因超出了这个问题的范围。
  • 一个纯JS文件:文件非常大,不再是Wasm了。

我的问题是:有没有办法在二进制文件中删除或实现这些依赖项进行编译?如果没有,是否有库(理想情况下是轻量级的(可以为这些功能提供实现?

更新:emscripten 现在支持 -s STANDALONE_WASM,它将构建尽可能可移植的 wasm 文件。 对于低级别系统调用,生成的二进制文件将使用 WASI 系统调用。

听起来您可能可以使用-s WASM_ASYNC_COMPILATION=0选项来实现您的目标。

为了回答更广泛的问题,emscripten构建的wasm文件与emscripten JS库代码相当相关。 有一些对构建独立 wasm 文件的支持,并且随着时间的推移,这种支持正在改进,但在一般情况下,完全独立的 wasm 文件是不可能的。

相关内容

  • 没有找到相关文章

最新更新