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

时间:2019-09-22 00:43:40

标签: javascript llvm webassembly

我正在尝试使用Emscripten作为学习机会,将完整的C ++程序编译为Wasm。以下命令会生成完整的有效且有效的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);
  

导入函数env:_assert必须是可调用的(评估“新的WebAssembly.Instance(模块,导入)”)

检查生成的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。

我的问题是:有没有一种方法可以用二进制中直接删除或实现的那些依赖项进行编译?如果没有,是否有理想的轻量级库可以为这些功能提供实现?

1 个答案:

答案 0 :(得分:2)

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

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

为回答更广泛的问题,使用Emscripten构建的wasm文件对Emscripten JS库代码相当厌倦。对于构建独立的wasm文件有一些支持,并且随着时间的推移,该支持会得到改善,但是在一般情况下,不可能完全使用沿着wasm的文件。