带有自定义链接的 OCaml 编译是否应该在 Windows 中工作(通过 MinGW)?
Should OCaml compilation with custom linking work in Windows (via MinGW)?
我想编译一个与 C 代码接口的 OCaml 程序,使用基于 MinGW 的 GCC,并使用单独的编译(GCC 生成 .o
,然后 ocamlopt
生成最终的可执行文件)。
我不清楚 (1) 这是否适用于 Windows,如果适用,(2) 哪些命令行参数是必需的。
我正在使用 Jonathan Protzenko 的 OCaml on Windows installer to install OCaml 4.02.1 along with a Cygwin shell (note that it uses a native windows OCaml compiler, not a Cygwin-based one). I installed gcc using Nuwen's MinGW(但在使用 Strawberry Perl 的 gcc 时遇到了同样的问题)。
这是我的源代码:
C文件(tc.c
):
#include <stdio.h>
#include "caml/mlvalues.h"
value print(value unused) {
printf("hello from C\n");
return Val_unit;
}
OCaml 文件(t.ml
):
external print : unit -> unit = "print"
let () =
Printf.printf "platform: %s\n" (Sys.os_type);
print ();
以下工作正常:
and@win7 $ ocamlopt t.ml tc.c -o t.exe
and@win7 $ ./t.exe
platform: Win32
hello from C
但是,如果我使用 .o
而不是 .c
,它就不起作用:
and@win7 $ gcc tc.c -c -I c:/OCaml/lib -o tc.o
and@win7 $ ocamlopt t.ml tc.o -o t.exe
** Cannot resolve symbols for tc.o:
puts
** Fatal error: Unsupported relocation kind 0004 for puts in tc.o
File "caml_startup", line 1:
Error: Error during linking
两个版本在 Linux 上都能正常工作。
我想知道这是否只是一些愚蠢的错误,我可以通过为 gcc/ocamlc/ocamlopt 提供正确的参数来快速修复,或者它是否是 OCaml 在 Windows.[=24 上的本机编译的当前限制=]
编辑: camlspotter 查明了原因,所以回想起来,我根本不需要Nuwen的MinGW。 Windows 上的 OCaml 已经包含一个基于 MinGW 的 C 编译器,除了它被称为 i686-w64-mingw32-gcc
而不是 gcc
.
您可能使用了错误的 C 编译器或没有适当的选项。最好的方法是使用与构建 OCaml 相同的 C 编译器 + 选项。您可以通过ocamlc -config
:
查看
$ ocamlc -config
version: 4.02.3
standard_library_default: C:/ocamlmgw64/lib
standard_library: C:/ocamlmgw64/lib
standard_runtime: ocamlrun
ccomp_type: cc
bytecomp_c_compiler: x86_64-w64-mingw32-gcc -O -mms-bitfields -Wall -Wno-unused
bytecomp_c_libraries: -lws2_32
native_c_compiler: x86_64-w64-mingw32-gcc -O -mms-bitfields -Wall -Wno-unused
native_c_libraries: -lws2_32
native_pack_linker: x86_64-w64-mingw32-ld -r -o
ranlib: x86_64-w64-mingw32-ranlib
...
例如,上面显示我的 OCaml 编译器是在 x86_64-w64-mingw32-gcc
的 Cygwin 32 位环境上构建的。这同样适用于链接器和 ranlib
。由于您可以使用 ocamlopt
使用 OCaml 代码编译 C,因此您的环境中必须已经安装了相同的 C 编译器。
自己构建 OCaml 编译器以确保为 C 和 OCaml 使用相同的 C 编译器可能是避免这种 C 编译器不匹配的最佳方法。
我想编译一个与 C 代码接口的 OCaml 程序,使用基于 MinGW 的 GCC,并使用单独的编译(GCC 生成 .o
,然后 ocamlopt
生成最终的可执行文件)。
我不清楚 (1) 这是否适用于 Windows,如果适用,(2) 哪些命令行参数是必需的。
我正在使用 Jonathan Protzenko 的 OCaml on Windows installer to install OCaml 4.02.1 along with a Cygwin shell (note that it uses a native windows OCaml compiler, not a Cygwin-based one). I installed gcc using Nuwen's MinGW(但在使用 Strawberry Perl 的 gcc 时遇到了同样的问题)。
这是我的源代码:
C文件(tc.c
):
#include <stdio.h>
#include "caml/mlvalues.h"
value print(value unused) {
printf("hello from C\n");
return Val_unit;
}
OCaml 文件(t.ml
):
external print : unit -> unit = "print"
let () =
Printf.printf "platform: %s\n" (Sys.os_type);
print ();
以下工作正常:
and@win7 $ ocamlopt t.ml tc.c -o t.exe
and@win7 $ ./t.exe
platform: Win32
hello from C
但是,如果我使用 .o
而不是 .c
,它就不起作用:
and@win7 $ gcc tc.c -c -I c:/OCaml/lib -o tc.o
and@win7 $ ocamlopt t.ml tc.o -o t.exe
** Cannot resolve symbols for tc.o:
puts
** Fatal error: Unsupported relocation kind 0004 for puts in tc.o
File "caml_startup", line 1:
Error: Error during linking
两个版本在 Linux 上都能正常工作。
我想知道这是否只是一些愚蠢的错误,我可以通过为 gcc/ocamlc/ocamlopt 提供正确的参数来快速修复,或者它是否是 OCaml 在 Windows.[=24 上的本机编译的当前限制=]
编辑: camlspotter 查明了原因,所以回想起来,我根本不需要Nuwen的MinGW。 Windows 上的 OCaml 已经包含一个基于 MinGW 的 C 编译器,除了它被称为 i686-w64-mingw32-gcc
而不是 gcc
.
您可能使用了错误的 C 编译器或没有适当的选项。最好的方法是使用与构建 OCaml 相同的 C 编译器 + 选项。您可以通过ocamlc -config
:
$ ocamlc -config
version: 4.02.3
standard_library_default: C:/ocamlmgw64/lib
standard_library: C:/ocamlmgw64/lib
standard_runtime: ocamlrun
ccomp_type: cc
bytecomp_c_compiler: x86_64-w64-mingw32-gcc -O -mms-bitfields -Wall -Wno-unused
bytecomp_c_libraries: -lws2_32
native_c_compiler: x86_64-w64-mingw32-gcc -O -mms-bitfields -Wall -Wno-unused
native_c_libraries: -lws2_32
native_pack_linker: x86_64-w64-mingw32-ld -r -o
ranlib: x86_64-w64-mingw32-ranlib
...
例如,上面显示我的 OCaml 编译器是在 x86_64-w64-mingw32-gcc
的 Cygwin 32 位环境上构建的。这同样适用于链接器和 ranlib
。由于您可以使用 ocamlopt
使用 OCaml 代码编译 C,因此您的环境中必须已经安装了相同的 C 编译器。
自己构建 OCaml 编译器以确保为 C 和 OCaml 使用相同的 C 编译器可能是避免这种 C 编译器不匹配的最佳方法。