在Mac OS上使用gfortran进行编译时出现Makefile错误

时间:2012-03-07 10:20:03

标签: macos linker makefile ld gfortran

我使用最新的Xcode(4.3)运行Mac OS X Lion:

gfortran --version # -> GNU Fortran (GCC) 4.6.1
gcc --version # -> i686-apple-darwin11-llvm-gcc-4.2 (GCC) 4.2.1 (Based on Apple Inc. build 5658)

我的makefile出错:

gfortran temp/modules.o temp/testModules.o temp/mathFunctions.o temp/functionTest.o -o arenatest
ld: duplicate symbol ___rng_MOD_uni in temp/testModules.o and temp/modules.o for architecture x86_64
collect2: ld returned 1 exit status
make: *** [arenatest] Error 1

我不明白原因是什么,所以我尝试编写一个构建脚本来做同样的事情,并且令人惊讶的是它可以创造奇迹。我想要一个makefile而不是一个脚本,所以有人能发现差异吗?我完全没有想法。

我知道有一个时髦的循环替换脚本,用于在正确的子目录中生成.o对象。然而,build / compile-commands似乎完全匹配。如果有一种更聪明的方式,我愿意接受建议。

脚本:

echo remove old files
rm *.mod
rm *.o
rm -rf temp
rm arenatest

echo create directories and compile:
mkdir -p temp
gfortran -cpp -c ../modules/modules.f90 -o temp/modules.o -J../buildtest -I../buildtest
gfortran -cpp -c ../modules/mathFunctions.f90 -o temp/mathFunctions.o -J../buildtest -I../buildtest
gfortran -cpp -c testModules.f90 -o temp/testModules.o -J../buildtest -I../buildtest
gfortran -cpp -c functionTest.f90 -o temp/functionTest.o -J../buildtest -I../buildtest

echo do linking: gfortran -cpp temp/modules.o temp/testModules.o temp/mathFunctions.o temp/functionTest.o -o arenatest
gfortran temp/modules.o temp/testModules.o temp/mathFunctions.o temp/functionTest.o -o arenatest

echo DONE!

makefile:

PROGRAM = arenatest
BUILDDIR = temp
FC = gfortran
SRC = ../modules/modules.f90 ../modules/mathFunctions.f90 testModules.f90 functionTest.f90
OBJ = $(SRC:.f90=.o)
OBJECTS = $(foreach var, $(OBJ), $(BUILDDIR)/$(lastword $(subst /, , $(var))) )
FLAGS =-J../buildtest -I../buildtest


all: buildtest

buildtest: $(PROGRAM)

build_message:
    @echo
    @echo sources:
    @echo $(SRC)
    @echo objects:
    @echo $(OBJECTS)

clean:
    rm -rf temp
    rm arenatest
    rm *.mod
    rm *.o


mkdir:
    @echo create directories and compile:
    mkdir -p temp

# 
# gfortran -cpp -c ../modules/modules.f90 -o temp/modules.o -J../buildtest -I../buildtest
# gfortran -cpp -c ../modules/mathFunctions.f90 -o temp/mathFunctions.o -J../buildtest -I../buildtest
# gfortran -cpp -c testModules.f90 -o temp/testModules.o -J../buildtest -I../buildtest
# gfortran -cpp -c functionTest.f90 -o temp/functionTest.o -J../buildtest -I../buildtest
# 
$(OBJECTS) : $(SRC) | mkdir
    $(FC) -c $(FLAGS) $< -o $@
# link: #echo do linking: gfortran -cpp temp/modules.o temp/testModules.o temp/mathFunctions.o temp/functionTest.o -o arenatest
$(PROGRAM): $(OBJECTS) | build_message
    $(FC) $(FLAGS) $(OBJECTS) -o $(PROGRAM)

#echo DONE!

1 个答案:

答案 0 :(得分:0)

编译目标文件的方法是错误的。扩展变量后,它将如下所示:

temp/modules.o temp/mathFunctions.o temp/testModules.o temp/functionTest.o: ../modules/modules.f90 ../modules/mathFunctions.f90 testModules.f90 functionTest.f90 | mkdir
    $(FC) -c $(FLAGS) $< -o $@

实际上,这意味着每个目标文件都依赖于所有源文件。此外,每次调用此配方都会将自动变量$<设置为此源文件列表中的第一个。这将导致所有目标文件都是从同一个源文件中编译的(因此当然所有目标文件都有相同符号的定义)。