使用 ifort/icc 时未定义对 'main' 的引用

Undefined reference to 'main' when using ifort/icc

我正在尝试编译用 Fortran 编写然后通过 C 编译器链接的代码。当我尝试使用 Intel 编译器 ifort + icc 时,问题就来了。如果我对 GNU 编译器 gfortran + gcc 做同样的事情,它就可以工作。

下面我们可以看到他们的输出。

ICC, 工作:

ifort -c -o fonts/pgpack.o -g -fno-automatic -fPIC fonts/pgpack.f ifort: command line warning #10006: ignoring unknown option '-fno-automatic'
icc -o pgpack fonts/pgpack.o -O2 -L/opt/intel/composer_xe_2013_sp1.0.080/compiler/lib/intel64 -L/opt/intel/composer_xe_2013_sp1.0.080/ipp/../compiler/lib/intel64 -L/opt/intel/composer_xe_2013_sp1.0.080/ipp/lib/intel64 -L/opt/intel/composer_xe_2013_sp1.0.080/compiler/lib/intel64 -L/opt/intel/composer_xe_2013_sp1.0.080/mkl/lib/intel64 -L/opt/intel/composer_xe_2013_sp1.0.080/tbb/lib/intel64/gcc4.4 -L/opt/intel/composer_xe_2013_sp1.0.080/compiler/lib/intel64 -L/usr/lib/gcc/x86_64-redhat-linux/4.4.7/ -L/usr/lib/gcc/x86_64-redhat-linux/4.4.7/../../../../lib64 -L/usr/lib/gcc/x86_64-redhat-linux/4.4.7/../../../../lib64/ -L/lib/../lib64 -L/lib/../lib64/ -L/usr/lib/../lib64 -L/usr/lib/../lib64/ -L/opt/intel/composer_xe_2013_sp1.0.080/compiler/lib/intel64/ -L/opt/intel/composer_xe_2013_sp1.0.080/ipp/../compiler/lib/intel64/ -L/opt/intel/composer_xe_2013_sp1.0.080/ipp/lib/intel64/ -L/opt/intel/composer_xe_2013_sp1.0.080/mkl/lib/intel64/ -L/opt/intel/composer_xe_2013_sp1.0.080/tbb/lib/intel64/gcc4.4/ -L/usr/lib/gcc/x86_64-redhat-linux/4.4.7/../../../ -L/lib64 -L/lib/ -L/usr/lib64 -L/usr/lib -L/opt/intel/composer_xe_2013_sp1.0.080/compiler/lib/intel64 -L/opt/intel/composer_xe_2013_sp1.0.080/ipp/../compiler/lib/intel64 -L/opt/intel/composer_xe_2013_sp1.0.080/ipp/lib/intel64 -L/opt/intel/composer_xe_2013_sp1.0.080/compiler/lib/intel64 -L/opt/intel/composer_xe_2013_sp1.0.080/mkl/lib/intel64 -L/opt/intel/composer_xe_2013_sp1.0.080/tbb/lib/intel64/gcc4.4 -L/opt/intel/composer_xe_2013_sp1.0.080/compiler/lib/intel64 -L/usr/lib/gcc/x86_64-redhat-linux/4.4.7/ -L/usr/lib/gcc/x86_64-redhat-linux/4.4.7/../../../../lib64 -L/usr/lib/gcc/x86_64-redhat-linux/4.4.7/../../../../lib64/ -L/lib/../lib64 -L/lib/../lib64/ -L/usr/lib/../lib64 -L/usr/lib/../lib64/ -L/opt/intel/composer_xe_2013_sp1.0.080/compiler/lib/intel64/ -L/opt/intel/composer_xe_2013_sp1.0.080/ipp/../compiler/lib/intel64/ -L/opt/intel/composer_xe_2013_sp1.0.080/ipp/lib/intel64/ -L/opt/intel/composer_xe_2013_sp1.0.080/mkl/lib/intel64/ -L/opt/intel/composer_xe_2013_sp1.0.080/tbb/lib/intel64/gcc4.4/ -L/usr/lib/gcc/x86_64-redhat-linux/4.4.7/../../../ -L/lib64 -L/lib/ -L/usr/lib64 -L/usr/lib -lifport -lifcore -limf -lsvml -lm -lipgo -lirc -lpthread -lsvml -lc -lgcc -lgcc_s -lirc_s -ldl -lc
/usr/lib/gcc/x86_64-redhat-linux/4.4.7/../../../../lib64/crt1.o: In function '_start':
(.text+0x20): undefined reference to `main'

注意:icc 链接到 gcc 库 (/usr/lib/gcc/...) 对我来说很奇怪。这是normal/right吗?

GCC,工作:

gfortran -c -o fonts/pgpack.o -O2 -Wall -fno-second-underscore -g -fno-automatic -fPIC fonts/pgpack.f
Warning: Nonconforming tab character in column 6 of line 15
Warning: Nonconforming tab character in column 6 of line 16
Warning: Nonconforming tab character in column 6 of line 17
Warning: Nonconforming tab character in column 2 of line 19
Warning: Nonconforming tab character in column 2 of line 20
Warning: Nonconforming tab character in column 2 of line 21
Warning: Nonconforming tab character in column 2 of line 24
Warning: Nonconforming tab character in column 2 of line 25
gcc -o pgpack fonts/pgpack.o -O2 -L/usr/lib/gcc/x86_64-redhat-linux/4.4.7 -L/usr/lib/gcc/x86_64-redhat-linux/4.4.7 -L/usr/lib/gcc/x86_64-redhat-linux/4.4.7/../../../../lib64 -L/lib/../lib64 -L/usr/lib/../lib64 -L/opt/intel/composer_xe_2013_sp1.0.080/compiler/lib/intel64 -L/opt/intel/composer_xe_2013_sp1.0.080/ipp/../compiler/lib/intel64 -L/opt/intel/composer_xe_2013_sp1.0.080/ipp/lib/intel64 -L/opt/intel/composer_xe_2013_sp1.0.080/compiler/lib/intel64 -L/opt/intel/composer_xe_2013_sp1.0.080/mkl/lib/intel64 -L/opt/intel/composer_xe_2013_sp1.0.080/tbb/lib/intel64/gcc4.4 -L/usr/lib/gcc/x86_64-redhat-linux/4.4.7/../../.. -L/usr/lib/gcc/x86_64-redhat-linux/4.4.7 -L/usr/lib/gcc/x86_64-redhat-linux/4.4.7 -L/usr/lib/gcc/x86_64-redhat-linux/4.4.7/../../../../lib64 -L/lib/../lib64 -L/usr/lib/../lib64 -L/opt/intel/composer_xe_2013_sp1.0.080/compiler/lib/intel64 -L/opt/intel/composer_xe_2013_sp1.0.080/ipp/../compiler/lib/intel64 -L/opt/intel/composer_xe_2013_sp1.0.080/ipp/lib/intel64 -L/opt/intel/composer_xe_2013_sp1.0.080/compiler/lib/intel64 -L/opt/intel/composer_xe_2013_sp1.0.080/mkl/lib/intel64 -L/opt/intel/composer_xe_2013_sp1.0.080/tbb/lib/intel64/gcc4.4 -L/usr/lib/gcc/x86_64-redhat-linux/4.4.7/../../.. -lgfortranbegin -lgfortran -lm -lgcc_s -lgcc -lc -lgcc_s -lgcc
: pgpack

我注意到一些正在链接的库发生了变化(例如 libgfortranbegin),但这是由 make 系统自动 added/changed。

我已经尝试过其他帖子提供的解决方案,对于类似的错误,比如使用标志 (ifort)-nofor-main 和 (icc)-nostartfiles,但它总是失败。

我错过了什么? 谢谢。

icc 的情况下,您正在 linking 的模块的 none 提供了一个函数 main() 作为程序的入口点。 gcc 然而 link 程序的成功可能部分是由于 -lgfortranbegin 包含在 link 中。这将提供一个标准(对于 gfortranmain() 函数,该函数在将控制权移交给与 [=17= 中定义的 PROGRAM 子单元相对应的不同名称的例程之前设置 Fortran 环境].

很可能 link 通过 icc 将一个 ifort 编译的目标文件编译成一个完整的程序,但显然你需要一个 ifort 特定的起点例程,并且您指定的库的 none 提供了一个。无论如何,您在 icc link 中包含 GCC 库可能是错误的。我假设您的构建系统正在识别并包括那些,但这样做是不正确的。

无论如何,当您只有一个从 Fortran 源派生的目标文件时,您 link 使用 gcc / icc 是非常奇怪的。您应该 link 使用 Fortran linker 驱动程序,在这两种情况下,它与 Fortran 编译器驱动程序 (gfortran / ifort) 相同。在那种情况下,您可能不需要当前指定的 -L-l 选项中的 any

即使您也在某些 C 例程中 linking,通常您应该使用 linker 驱动程序来编写主例程所用的语言。有一些注意事项,如果您混合使用其他一些语言(例如 C++),它会变得更加复杂,但您的特定情况看起来非常简单。