我正在构建一个 Webassembly 运行时,目前正在实现 WASI API。根据这份文档,我想知道 ABI 的外观如何:https ://github.com/WebAssembly/WASI/blob/main/phases/snapshot/docs.md
为了进行测试,我使用 emscripten 将这个 C 应用程序编译为一个独立的 WASM 模块。
#include <stdio.h>
int main() {
printf("Hello, World!\n");
return 0;
}
使用 wasm-objdump 检查后,我可以看到以下函数导入:
Import[2]:
- func[0] sig=2 <__wasi_proc_exit> <- wasi_snapshot_preview1.proc_exit
- func[1] sig=11 <__wasi_fd_write> <- wasi_snapshot_preview1.fd_write
带有类型签名:
- type[2] (i32) -> nil
- type[11] (i32, i32, i32, i32) -> i32
根据规范,该函数具有映射到 POSIX 系统调用fd_write
的签名。 fd_write(fd: fd, iovs: ciovec_array) -> Result<size, errno>
ssize_t writev(int fd, const struct iovec *iov, int iovcnt);
但是 WASM 文件中的第四个参数是什么?它接收一些指向内存地址的指针。所以我想我必须写到Result<size, errno>
那个地址,但是如果我这样做并返回 0(表示成功),fd_write
就会一遍又一遍地调用(大概是因为 printf 函数假定没有写入任何内容)。如果我返回写入的字节,程序会正确终止,但是第四个参数是什么?另外,我怎样才能返回Result
不适合 i32 的更复杂的 s?