当前位置: 首页 > 文档资料 > OCaml 中文教程 >

调用 Fortran 库

优质
小牛编辑
133浏览
2023-12-01

Fortran isn't a language the many people write new codes in but it still is in extensive use in the scientific communities. Many, many libraries exist for doing numerical calculation that will never be written in C or C++. It is quite possible though to call Fortran routines from OCaml as they are normally compiled into the same object format, with minimal name mangling, as C programs.

This tutorial will step by step through the process of compiling an interface module for a fortran function. The steps that are involved here are the same steps as for wrapping a C function with a few considerations that have to be taken into account for Fortran.

The Fortran function is contained in a file called func.f and has the following signature

subroutine gtd6(integer iyd, real sec, real alt, real lat, real lon, real dens(8), real temp(2))

The iyd, sec, alt, lat, and lon parameters are input parameters while dens and temp are output parameters.

All of the examples below use the GNU Fortran 77 compiler (g77). None of these have been tested with the GNU fortran 90 compiler (gfort) and will not be until it has proven itself through some time.

Step 1: Compile the Fortran routine

Where C/C++ have only one category of subroutine (the function), Fortran has two: the function and the subroutine. The function is the equivalent to a non-void C function in that it takes parameters and always returns a value. The subroutine is equivalent to a void C function.

When g77 compiles a fortran function it creates a named function that has an underscore appended. If the fortran name for the function contains any underscores then the compiled function name will have two underscores appended. The generated function can be called by this name. Subroutines will be converted into a C function that returns an int.

To compile the funcs.f file into an object file, one can use the following:

prompt> g77 -c funcs.f

Which will generate the file 'funcs.o'. You can then see the names of the compiled functions by executing

prompt> nm funcs.o

In this output you will see a line that has the following

T gtd6_

This shows that the function gtd6_ has been created and is in the object file.

Fortran has support for both integer and real types and those are the names that they go by. In our case we have only real and integer types. Reals are equivalent to C doubles and integers are equivalent to C longs. In addition, Fortran passes everything by reference so the corresponding C prototype for our gtd6 function is

int gtd6_(integer *iyd, real* sec, real* alt, real* glat, real* glong, real* dens, real* temp);

Note that its up to the caller to know that dens and temp are actually arrays. Failure to pass an array will cause a segmentation violation since the gtd6_ function is using them as arrays (yet another reason OCaml shines).

Step 2: Create the C wrapper

Because OCaml's foreign function interface is C based, it is necessary to create a C wrapper. To avoid difficulties in passing back arrays of values, we are going to simply create a function that will return the second element of the temperature array as computed by the function and ignore the other return values (this is a very frequent use of the function). This function will be in the source file wrapper.c.

CAMLprim value gtd6_t (value iydV, value secVal, value altVal, value latVal, value lonVal) {
   CAMLparam5( iydV, secVal, altVal, latVal, lonVal );
   long iyd = Long_val( iydV );
   float    sec = Double_val( secVal );
   float    alt = Double_val( altVal );
   float    lat = Double_val( latVal );
   float    lon = Double_val( lonVal );

   gtd6_(&iyd, &sec, &alt, &glat, &glon, d, t);
   CAMLreturn( caml_copy_double( t[1] ) );
}

A few points of interest

  1. The file must include the OCaml header files alloc.h, memory.h, and mlvalue.h.
  2. The function first calls the CAMLparam5 macro. This is required at the start of any function that uses the CAML types.
  3. The function uses the Double_val and Long_val macros to extract the C types from the OCaml value object.
  4. All of the values are passed by reference to the gtd6_ routine as required by the prototype.
  5. The function uses the copy_caml_double function and the CAMLreturn macro to create a new value containing the return value and to return it respectively.

Step 3: Compile the shared library.

Now having the two source files funcs.f and wrapper.c we need to create a shared library that can be loaded by OCaml. Its easier to do this as a multistep process, so here are the commands:

prompt> g77 -c funcs.f

prompt> cc -I<ocaml include path> -c wrapper.c

prompt> cc -shared -o wrapper.so wrapper.o funcs.o -lg2c

This will create a shared object library called wrapper.so containing the fortran function and the wrapper function. The -lg2c option is required to provide the implementations of the built in fortran functions that are used.

Step 4: Now to OCaml

Now in an OCaml file (gtd6.ml) we have to define the external reference to the function and a function to call it.

external temp : int -> float -> float -> float -> float -> float = "gtd6_t"

let () =
  print_double (temp 1 2.0 3.0 4.0 5.0);
  print_newline ()

This tells OCaml that the temp function takes 5 parameters and returns a single floating point and calls the C function gtd6_t.

At this point, the steps that are given are to compile this into bytecode. I don't yet have much experience compiling to native so I'll let some else help out (or wait until I learn how to do it).

prompt> ocamlc -c gtd6.ml prompt> ocamlc -o test gtd6.cmo wrapper.so

And voila, we've called the fortran function from OCaml.