使用 OpenMPI 从 Fortran 调用 C++

Calling C++ from Fortran with OpenMPI

我有一个编译时问题,我已将其简化为以下测试用例。我希望从 fortran 调用 C++ 例程并让 C++ 例程能够识别 MPI。

考虑以下示例代码,

Fortran 主程序:

! -- main.f90
program main
   implicit none
   external return_three
   integer return_three

   write(*,*) return_three()
end program main

C++ 子例程:

// -- subs.cpp
#include <mpi.h>

extern "C"
{
   int return_three_();
}

int return_three_()
{
   return 3;
}

请注意,为了重现问题,我只需要包含 mpi.h

使用 GCC 5.3 和 OpenMPI 1.10.1 编译(我也检查了 GCC 4.8 和 PGI 15.10)在链接期间出现以下问题:

% mpic++ -c subs.cpp
% mpifort -c main.f90
% mpifort -o main subs.o main.o -lstdc++ -lgcc_s
subs.o: In function `MPI::Intracomm::Intracomm()':
subs.cpp:(.text._ZN3MPI9IntracommC2Ev[_ZN3MPI9IntracommC5Ev]+0x14): undefined reference to `MPI::Comm::Comm()'
subs.o: In function `MPI::Intracomm::Intracomm(ompi_communicator_t*)':
subs.cpp:(.text._ZN3MPI9IntracommC2EP19ompi_communicator_t[_ZN3MPI9IntracommC5EP19ompi_communicator_t]+0x19): undefined reference to `MPI::Comm::Comm()'
subs.o: In function `MPI::Op::Init(void (*)(void const*, void*, int, MPI::Datatype const&), bool)':
subs.cpp:(.text._ZN3MPI2Op4InitEPFvPKvPviRKNS_8DatatypeEEb[_ZN3MPI2Op4InitEPFvPKvPviRKNS_8DatatypeEEb]+0x24): undefined reference to `ompi_mpi_cxx_op_intercept'
subs.o:(.rodata._ZTVN3MPI3WinE[_ZTVN3MPI3WinE]+0x48): undefined reference to `MPI::Win::Free()'
subs.o:(.rodata._ZTVN3MPI8DatatypeE[_ZTVN3MPI8DatatypeE]+0x78): undefined reference to `MPI::Datatype::Free()'
collect2: error: ld returned 1 exit status

在我看来,mpifort 缺少一些与 C++ 相关的库。不过,据我了解,应该使用 mpifort 来编译 Fortran 主程序。针对 OpenMPI 1.10.1 编译的英特尔 16.0 不会出现此问题。

我的问题是:

通过在最后一步添加 -lmpi_cxx,我能够使用 GCC 5.3.0 和 openMPI 1.10.2 编译您的代码:

% mpic++ -c subs.cpp
% mpifort -c main.f90
% mpifort -o main main.o subs.o -lstdc++ -lmpi_cxx

原因是 openMPI 包装器将 mpifort 和 mpic++ link 编译为不同的 MPI 库。您可以使用 -showme:libs 选项进行检查:

% mpifort -showme:libs
mpi_usempif08 mpi_usempi_ignore_tkr mpi_mpifh mpi
% mpic++ -showme:libs
mpi_cxx mpi

因此,为了使用 C++ MPI 库,您必须明确告诉 mpifort link 给它。