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