* 概要 [#n5707163] - OCaml特有の話 -- 直接OCamlの関数を.soなり.dllなりでエクスポートすることはできない。 Cの関数でスタブを書く必要がある。 -- OCaml側で, モジュール初期化時に '''Callback.register 文字列 関数''' とすれば 関数を 名前 '''"文字列"''' でエクスポートできる -- C側では*caml_named_value(文字列)として関数の「名前」を取得する。 -- caml_callbackで関数を呼ぶ - FFI一般の話 -- OCamlランタイムを初期化する必要がある。 -- OCamlの値とCの値の相互変換が必要である。下の例ではVal_int(C→OCaml)とInt_val(OCaml→C)を用いている。 OCamlのint型は31ビット幅であることに注意。 * Linuxの場合 [#gc0eda21] ** ライブラリ側ファイル [#s85a88f9] - foo.ml let succ x = x+1 let _ = Callback.register "succ" succ - foostub.h int succ_caml(int x); - foostub.c #include <caml/mlvalues.h> #include <caml/callback.h> #include "foostub.h" int succ_caml(int x) { return Int_val(caml_callback(*caml_named_value("succ"), Val_int(x))); } *** コンパイル [#g467c1d9] - ocamlcにオプション -output-obj -o lib'''name'''.so とすれば共有ライブラリが得られる. gcc -I/usr/local/lib/ocaml -c foostub.c ocamlc -output-obj -o libfoo.so foo.ml foostub.o ** プログラム本体 [#s0de3411] - usefoo.c #include "foostub.h" #include <stdio.h> #include <caml/callback.h> int main(int argc, char** argv) { caml_startup(argv); printf("%d\n", succ_caml(10)); return 0; } *** コンパイル [#d73d3ca5] - 普通の共有ライブラリとしてリンクして実行。 gcc -I/usr/local/lib/ocaml -o usefoo libfoo.so usefoo.c *** 実行 [#jcb410e8] $ export LD_LIBRARY_PATH=`pwd`:$LD_LIBRARY_PATH $ ./usefoo 11