4

我打算使用 NIF 为我打算在 Erlang 中编写代码的应用程序操作二进制文件。下面给出了 NIF 的 cpp 文件和 erl 文件的要点链接。

[Erl 要点链接] https://gist.github.com/abhijitiitr/3a5bc97184d6dd32f97b

[C++ 要点链接] https://gist.github.com/abhijitiitr/24d2b780f2cdacebfb07

基本上我正在尝试做一个简单的测试。在 NIF 调用之间共享二进制文件,并通过连续的 NIF 调用成功地操作它们。

如果您通过以下方式测试 erlang REPL 中的代码

c(binary_test).
Ref=binary_test:open(<<1>>).
binary_test:increment(Ref,<<3>>).

二进制文件存储了 NIF 调用之间的更改。第三个命令的 REPL 输出是

1
 3
  60
    60
      <<"?">>

<<1>>在初始化阶段通过了。为什么改成<<60>>? 我无法弄清楚这里发生了什么。有人可以指出错误吗?

C++编译指令

clang++ -std=c++11 -stdlib=libc++ -undefined dynamic_lookup -O3 -dynamiclib binary_test.cpp -o binary_test.so -I /usr/local/Cellar/erlang/17.0/lib/erlang/erts-6.0/include/ 

在我的 Mac 上。

我还想询问有关在 NIF 中操作共享资源的并发进程。这是可能的还是有一个规则是必须在单个 Erlang 进程中访问 NIF。

4

2 回答 2

5

您遇到问题是因为您非法访问内存。在您的BinaryStore构造函数中,您试图从传递给的参数列表中保存二进制文件binary_test:open/1,但这不起作用,因为一旦 NIF 调用完成,这些参数就会被释放。您需要保存参数的副本以供以后使用。为此,首先向您的BinaryStore班级添加一个新成员:

    ErlNifEnv* term_env;

接下来,修改您的构造函数以分配term_env,然后使用它来复制传入的术语:

    BinaryStore(ERL_NIF_TERM binary)
    {
        term_env = enif_alloc_env();
        binary_term = enif_make_copy(term_env, binary);
    }

binary_term这在环境中分配,term_env然后将传入的术语复制到其中。你还需要一个析构函数来释放term_env

    ~BinaryStore()
    {
        enif_free_env(term_env);
    }

最后,您需要通过term_env而不是在函数中env进行检查时:binary_termincrement_binary

    nifpp::get_throws(term_env, binary_term, ibin);

完成这些修改后,我通过运行代码得到以下结果:

1> Ref=binary_test:open(<<1>>).
Reading symbols for shared libraries . done
<<>>
2> binary_test:increment(Ref,<<3>>).
1
 3
  1
   1
    <<4>>

(顺便说一句,你应该使用"\r\n"换行符而不是仅仅"\n"在从 Erlang 模拟器内部打印时,这样换行符总是返回到最左边的列。)

你还有一个问题,就是你泄漏了分配给new_bin2.

我对学习 NIF 细节的建议是避免像nifpp一开始那样使用包,这样你就可以学习NIF API以及有关内存所有权、资源分配和释放以及参数转换的所有细节。一旦你理解了它们,使用类似的包nifpp就会变得更容易和更有成效。

于 2014-10-22T13:31:27.260 回答
2

ERL_NIF_TERMs 必须与 a 相关联ErlNifEnv,并且传递给您的 nif 函数的 env 仅在该函数调用期间有效。当您将术语存储到BinaryStore对象中,然后在另一个 nif 调用中使用它时,您违反了此规则。您的选择:

  1. 为您的二进制存储创建一个新的 ErlNifEnv,并将 nif 调用中的术语复制到这个新的环境中。

  2. 使用 C++ 数据结构(例如std::vector<unsigned char>)来存储您的二进制数据。我认为这对于您的情况会更简单。

于 2014-10-22T13:26:38.880 回答