将 Nim Anon 函数导出到 C++

Mne*_*nth 3 c++ ffi nim-lang

我正在尝试从 C++ 调用 Nim 代码。具体来说,就是一个接受匿名函数的函数。

我在 Nim 中有以下代码:

proc test*(a: proc()) {.exportc.} = a()
Run Code Online (Sandbox Code Playgroud)

我编译成一个静态库。然后我将它链接到我的 C++ 可执行文件并尝试使用

extern "C" test(void a(void);
Run Code Online (Sandbox Code Playgroud)

并调用它

void anon() { printf("hello"); }
...
test(anon)
Run Code Online (Sandbox Code Playgroud)

一切都编译得很好,但是当我运行程序时,它崩溃了。

zah*_*zah 5

默认情况下,Nim 将匿名 proc 类型编译为闭包,由一对 C 函数指针和一个指向包含闭包捕获的所有局部变量的结构的空指针表示。在生成的代码中它看起来像这样:

typedef struct {
N_NIMCALL_PTR(void, ClP_0) (void* ClE_0);
void* ClE_0;
} tyProc_XXXXXX;
Run Code Online (Sandbox Code Playgroud)

因此,要解决该问题,您必须修改 C 代码中测试函数的 extern "C" 定义以接受兼容的结构类型。或者,您可以通过将cdecl编译指示添加到 proc 类型,让 Nim 将 proc 参数编译为常规 C 函数:

proc test*(a: proc() {.cdecl.}) {.exportc.} = a()
Run Code Online (Sandbox Code Playgroud)

有关 Nim 支持的调用约定的完整列表,请查看Nim 手册中有关 proc 类型的部分。