16

我正在尝试从 C 调用 FORTRAN 函数

我的问题是:

  1. 如果fortRoutine是我的 fortran 子例程的名称,那么我从 C 中将其称为fortRoutine_. 如果fortRoutine只包含一个字符数组参数,那么我可以这样传递:

    fortRoutine_("I am in fortran");
    
  2. 在调用 FORTRAN 子例程时,我应该何时使用按值传递以及何时使用按引用传递?

由于我是 C 新手,所以我对此一无所知。如果可能的话,请推荐一些好的教程链接。

4

3 回答 3

28

现在执行此操作的方法是在 Fortran 端使用 Fortran ISO C 绑定。这是 Fortran 2003 语言标准的一部分,可用于许多编译器;它不是特定于 gcc 的。该网站的许多答案中已对此进行了描述。作为语言标准的一部分,它独立于编译器和平台。而且您不需要了解编译器的内部传递约定。ISO C 绑定在 Fortran 子例程或函数的声明中使用时,会导致 Fortran 编译器使用 C 调用约定,以便可以直接从 C 调用该过程。您无需添加隐藏参数或名称修改Fortran 子程序名称,即没有下划线。链接器使用的名称来自“绑定”选项。

字符串是一个困难的情况,因为从技术上讲,它们在 C 中是字符数组,您必须在 Fortran 中匹配它。您还必须处理字符串的不同定义:C 以空结尾,Fortran 固定长度并用空格填充。该示例显示了这是如何工作的。数字更容易。数组的唯一问题是 C 是行优先的,而 Fortran 是列优先的,因此多维数组被转置。

int main ( void ) {

   char test [10] = "abcd";

   myfortsub (test);

   return 0;

}

subroutine myfortsub ( input_string ) bind ( C, name="myfortsub" )

   use iso_c_binding, only: C_CHAR, c_null_char
   implicit none

   character (kind=c_char, len=1), dimension (10), intent (in) :: input_string
   character (len=10) :: regular_string
   integer :: i

   regular_string = " "
   loop_string: do i=1, 10
      if ( input_string (i) == c_null_char ) then
         exit loop_string
      else
         regular_string (i:i) = input_string (i)
      end if
   end do loop_string

   write (*, *) ">", trim (regular_string), "<", len_trim (regular_string)

   return

end subroutine myfortsub

您将 C 编译为目标文件并使用 gfortran 编译 fortran 并链接两者:

gcc-mp-4.6   \
         -c  \
         test_fortsub.c

gfortran-mp-4.6   \
     test_fortsub.o  \
     myfortsub.f90  \
     -o test_fortsub.exe

输出是:

 >abcd<           4
于 2011-11-21T08:15:27.657 回答
4

当然这一切都取决于你的 FORTRAN 编译器,但一般来说:

  1. 不,您需要为您的字符串传递一个隐藏的长度参数。一些编译器将这些与其他参数交错,直接在字符串之后。其他人,将所有字符串长度参数分组在参数列表的末尾。

    char str[11] = {0};
    fortranFunc_(str, sizeof(str) - 1);
    // remember that 'str' will need to be null terminated
    // and will be padding with spaces to fit the length
    // so for C passing strings to Fortran specify the length
    // less 1 so you can add a nul terminator, and on all strings
    // being filled in by FORTRAN, trim-end all spaces.
    
  2. 它几乎总是通过引用传递,但您可以使用 FORTRAN 端的虚拟参数上的属性来切换此行为。

    int value = 10;
    fortranFunc_(&value);
    // INTEGER I
    

以下是一些参考资料,适用于各种编译器:

于 2011-11-21T06:34:24.443 回答
1

答案取决于编译器和系统(技术上,它的 ABI)。对于 GCC(它是一个 C、一个 C++、一个 Ada 和一个 Fortran 编译器),请阅读fortran 混合编程章节。

于 2011-11-21T06:23:17.883 回答