我正在研究如何将 OCaml TCP/IP 堆栈集成到我的 C++ 项目中。由于这个答案,我已经知道如何从 OCaml 调用 C 并从 C 调用 OCaml:OCaml 作为 C 库,hello world 示例
OCaml 将由 C++ 控制,而不是相反。因此,对于 TCP/IP 堆栈,我必须能够发送和接收数据包。我可以通过C++轻松地将数据发送到TCP/IP堆栈,但是如何接收它呢?我需要将一个 C 函数(回调)作为参数传递给 OCaml,以便它在数据到达时传递数据。是否可以?
小智 5
为此,您需要两个 C 函数。第一个(wrap_fun如下)是从 C 代码调用的。它接受一个 C 回调并返回一个 OCaml 值,然后您可以将该值传递给 OCaml 代码。第二个(call_wrapped如下)是从 OCaml 代码中调用的。它采用第一个函数创建的 OCaml 值并调用存储在其中的回调。
您没有指定有关回调签名的任何信息,因此下面的代码适用于value(value).
#include <caml/alloc.h>
#include <caml/memory.h>
#include <caml/mlvalues.h>
typedef value (*cb)(value);
value wrap_fun(cb f) {
value v = caml_alloc_small(1, Abstract_tag);
Field(v, 0) = (value)f;
return v;
}
value call_wrapped(value f, value x) {
CAMLparam2(f, x);
cb g = (cb)Field(f, 0);
value z = g(x);
CAMLreturn(z);
}
Run Code Online (Sandbox Code Playgroud)
在 OCaml 方面,它看起来如下:
external call_wrapped : ('a, 'b) wrapped_fun -> 'a -> 'b = "call_wrapped"
let foo f x =
let y = x + 1 in
let z = call_wrapped f y in
z ^ "a"
Run Code Online (Sandbox Code Playgroud)
和中CAMLparam2的CAMLreturn宏call_wrapped只是用于教学目的。它们可以安全地删除,因为该函数只是实际函数的包装器。