0

我有一个 fortran 子例程 FortranShake 和一个 C++ 主函数 HandShakingTest.cpp。我正在尝试从 CLR C++ 调用 fortran 子例程。

我收到两批错误。我们称它们为 ERROR(1) 和 ERROR(2)。如果您能帮助我理解为什么会发生这些错误,我将不胜感激。

当我尝试使用以下内容进行编译时: cl /clr HandShakingTest.cpp

我收到以下错误(1):

HandShakingTest.obj : error LNK2028: unresolved token (0A00030A) "extern "C" void __c
ecl FortranShake(int &)" (?FortranShake@@$$J0YAXAAH@Z) referenced in function "int __
lrcall main(cli::array<class System::String ^ >^)" (?main@@$$HYMHP$01AP$AAVString@Sys
em@@@Z)
HandShakingTest.obj : error LNK2019: unresolved external symbol "extern "C" void __cd
cl FortranShake(int &)" (?FortranShake@@$$J0YAXAAH@Z) referenced in function "int __c
rcall main(cli::array<class System::String ^ >^)" (?main@@$$HYMHP$01AP$AAVString@Syst
m@@@Z)
HandShakingTest.exe : fatal error LNK1120: 2 unresolved externals

然后我使用以下命令进行编译:

ifort /c FortranShake.f //Which compiles fine
cl /c /clr HandShakingTest.cpp //compiles fine
cl /o test HandShakingTest.obj FortranShake.obj //ERROR(2) occurs

错误(2)包括:

MSVCRT.lib(ti_inst.obj) : error LNK2005: "private: __thiscall type_info::type_info(cla
ss type_info const &)" (??0type_info@@AAE@ABV0@@Z) already defined in LIBCMT.lib(typin
fo.obj)
MSVCRT.lib(ti_inst.obj) : error LNK2005: "private: class type_info & __thiscall type_i
nfo::operator=(class type_info const &)" (??4type_info@@AAEAAV0@ABV0@@Z) already defin
ed in LIBCMT.lib(typinfo.obj)
MSVCRT.lib(merr.obj) : error LNK2005: __matherr already defined in LIBCMT.lib(_matherr
_.obj)
LINK : warning LNK4098: defaultlib 'MSVCRT' conflicts with use of other libs; use /NOD
EFAULTLIB:library
HandShakingTest.obj : error LNK2028: unresolved token (0A00030A) "extern "C" void __cd
ecl FortranShake(int &)" (?FortranShake@@$$J0YAXAAH@Z) referenced in function "int __c
lrcall main(cli::array<class System::String ^ >^)" (?main@@$$HYMHP$01AP$AAVString@Syst
em@@@Z)
HandShakingTest.obj : error LNK2019: unresolved external symbol "extern "C" void __cde
cl FortranShake(int &)" (?FortranShake@@$$J0YAXAAH@Z) referenced in function "int __cl
rcall main(cli::array<class System::String ^ >^)" (?main@@$$HYMHP$01AP$AAVString@Syste
m@@@Z)
libifcoremt.lib(for_main.obj) : error LNK2019: unresolved external symbol _MAIN__ refe
renced in function _main
test.exe : fatal error LNK1120: 3 unresolved externals

这是 HandShakingTest.cpp:

#include "stdio.h"
#include <stdlib.h>
#include <Windows.h>
#using <System.DLL>
#using <System.Windows.Forms.DLL>

using namespace std;
using namespace System;
using namespace System::IO;
using namespace System::Diagnostics;
using namespace System::ComponentModel;
using namespace System::Collections;
using namespace System::Windows::Forms;

extern "C" {void FortranShake(int&);}

int main(array<System::String ^> ^args)
{
    Process^ testHand = gcnew Process();
    testHand->StartInfo->UseShellExecute = false;
    testHand->StartInfo->RedirectStandardInput = true;
    testHand->StartInfo->RedirectStandardOutput = true;
    testHand->StartInfo->ErrorDialog = true;

    int numDebug = 0;
    String^ returnedDebug = "Nothing";

    FortranShake(numDebug);

    StreamReader^ FromHandProcess = testHand->StandardOutput;
    StreamWriter^ ToHandProcess = testHand->StandardInput;

    String^ Line;

    Line = FromHandProcess ->ReadLine();

    if (Line->Equals("Enter Hand") )
    {
        Console::WriteLine(L"Hand Started!");
    }
    ToHandProcess ->WriteLine(numDebug.ToString());
    returnedDebug = FromHandProcess ->ReadLine();

    MessageBox::Show(returnedDebug);

    return 0;
}

这是 Fortran 子例程:

  SUBROUTINE FortranShake(GP_DEBUG)
  IMPLICIT DOUBLE PRECISION (A-H,O-Z)
  INN = 5

  WRITE(06,'(a)') 'Enter Hand'

  READ(INN,*) GP_DEBUG
  GP_DEBUG = GP_DEBUG + 55
  WRITE(06,*) GP_DEBUG

  RETURN
  END
4

1 回答 1

3

您的第一个错误实际上是链接器错误 - 没有/c命令行开关,您正在一步编译和链接。没有提供 Fortran 代码或目标代码。

您的第二个错误是因为:

  • 您已为 C++ 和 Fortran 指定(通过省略)不匹配的运行时库。您需要决定是要使用静态链接( Windows 上英特尔 Fortran当前版本(截至今天,但不一定是下个月......)的默认值)还是动态链接(MS C++ 的默认值)编译器)。也许添加/MD到指定动态链接的 ifort 命令行。

  • 相反,如果没有编译器选项或指令,则由该 Fortran 编译器生成的 C 代码中 Fortran 过程的等效标识符是 Fortran 过程名称的大写变体 - 即在 C++ 代码中调用过程 FORTRANSHAKE。如果您可以按照 F2003 标准编写 Fortran 代码,则应使用该语言的 C 互操作性特性 ( BIND(C,...)) 来控制 Fortran 过程的 C 绑定名称并确保调用约定等保持一致。

  • Fortran 子例程的虚拟参数具有 DOUBLE PRECISION 类型说明符,double对于这种编译器组合,它在 C++ 中等效,而不是int. 同样,F2003 引入了可以使参数类型的这种对齐更加健壮的功能。

于 2013-08-21T21:25:52.050 回答