EXPORTED_FUNCTIONS 在 emcc 编译期间被忽略

问题描述

我正在尝试学习如何将 CPP 编译成 WASM,但我似乎无法让它正常工作。

我有一个简单的 CPP 代码,它实现了对给定整数求平方的平方函数

#ifdef __EMSCRIPTEN__
#include <emscripten.h>
#endif

int EMSCRIPTEN_KEEPALIVE squarer(int num) {return num * num;}

我可以将它编译到 WASM 中,但是当我尝试在 JS 中加载它时,它说没有导出函数。 查看 WAT 版本(通过从 wabt 运行 wasm2wat)很明显问题在于未导出平方函数

(module
  (type $t0 (func (param i32) (result i32)))
  (import "env" "__linear_memory" (memory $env.__linear_memory 0))
  (import "env" "__indirect_function_table" (table $env.__indirect_function_table 0 funcref))
  (func $_Z7squareri (type $t0) (param $p0 i32) (result i32)
    (i32.mul
      (local.get $p0)
      (local.get $p0))))

因此,我尝试使用 EXPORTED_FUNCTIONS 标志再次编译(请参阅以下行),但收到以下警告并且平方函数仍未导出。

$ emcc squarer.cc -o squarer.wasm -c -O1 -s EXPORTED_FUNCTIONS="_Z7squareri","squarer"
emcc: warning: linker setting ignored during compilation: 'EXPORTED_FUNCTIONS' [-Wunused-command-line-argument]

当我编辑 WAT 以包含导出(通过将 func $_Z7squareri 更改为 func (export "squarer"))并将其编译回 WASM 时,JS 成功加载了该函数,因此似乎缺少的导出确实是问题。

我在两台 Linux 机器上对其进行了测试(一台运行 ubuntu 20.04,另一台运行 18.04,两者都按照 MDN website 中的说明从他们的 github 下载了最新版本的 emsdk)。 我还尝试使用所有其他带有“EXPORT”的 emcc 标志,但它们都收到了相同的警告。

这是我第一次构建 WASM 模块,我从互联网上学到了一切,所以我可能错过了一些关键步骤。我做错了吗?

我使用的是过时的工具吗?

我应该使用更好/更新的 CPP 到 WASM 编译器吗?

谢谢! 意泰

解决方法

-c 命令行选项会生成一个 wasm 目标文件,该文件不是设计为直接执行的。然后您需要使用 emcc(不带 -c)链接该目标文件以获得最终的二进制文件。

只需从上面给出的示例命令行中删除 -c,您就可以在单个命令中进行编译和链接。

如果您使用 EMSCRIPTEN_KEEPALIVE,则命令行上不需要 EXPORTED_FUNCTIONS,因为它们具有相同的用途。