【发布时间】:2011-07-04 00:31:39
【问题描述】:
我刚刚开始使用 makefile,我已经为我的项目编写了一个具有两个目标的文件。如果我运行一个新的 make 一切正常。但是如果在那之后我不清理,例如我编辑并保存一个源文件,然后我再次运行 make,我的项目只会编译我编辑的源文件,然后不会尝试将我的源链接到一个新的可执行。
换句话说:我第一次运行 make 时,每个源都会得到 $(CC) -c <sourcefile>,然后是 $(CC) -o <sourceobject1> <sourceobject2>...,一切正常。但是如果我然后去编辑一个源文件,并尝试再次制作,我只会得到$(CC) -c <editedsourcefile>,之后就没有链接了!至少 make 不会尝试再次编译我所有的源文件,但为什么它不链接我一个新的可执行文件?这是预期的行为吗?
额外问题:有没有比在目标规则顶部粘贴mv *.o 更好的方法将我的目标文件移动到子目录中?我看过this answer,但我不明白发生了什么,也无法弄清楚如何让它适应我有两个目标的情况。
这是我的生成文件:
CC = clang++
OPTFLAGS = -O3 -flto -m64
CFLAGS = -Wall -c $(OPTFLAGS)
LFLAGS = -Wall $(OPTFLAGS)
MAKE = make
RM = \rm -f
TARGET-DIR = build
OBJ-DIR = $(TARGET-DIR)/obj
OBJS = $(addprefix $(OBJ-DIR)/, MCTS.o NodePool.o Node.o FastMath.o \
Board.o Patterns.o FastSet.o FastHash.o FastHash2.o FastHashMap.o)
GTP-OBJS = $(addprefix $(OBJ-DIR)/, GtpPlayer.o)
GTP-TARGET = $(TARGET-DIR)/go-gtp
TEST-OBJS = $(addprefix $(OBJ-DIR)/, TestsMain.o MCTSTests.o NodePoolTests.o \
NodeTests.o FastMathTests.o BoardTests.o FastSetTests.o \
FastHashTests.o FastHash2Tests.o FastHashMapTests.o)
TEST-TARGET = $(TARGET-DIR)/go-test
.PHONY : all
all : $(GTP-TARGET) $(TEST-TARGET)
$(GTP-TARGET) : $(GTP-OBJS) $(OBJS)
mv *.o $(OBJ-DIR)
$(CC) $(LFLAGS) $(GTP-OBJS) $(OBJS) -o $(GTP-TARGET)
$(OBJ-DIR)/GtpPlayer.o : GtpPlayer.hpp GtpPlayer.cpp mcts/MCTS.hpp
$(CC) $(CFLAGS) GtpPlayer.cpp
$(OBJ-DIR)/MCTS.o : mcts/MCTS.hpp mcts/MCTS.cpp mcts/NodePool.hpp
$(CC) $(CFLAGS) mcts/MCTS.cpp
$(OBJ-DIR)/NodePool.o : mcts/NodePool.hpp mcts/NodePool.cpp mcts/Node.hpp
$(CC) $(CFLAGS) mcts/NodePool.cpp
$(OBJ-DIR)/Node.o : mcts/Node.hpp mcts/Node.cpp mcts/FastMath.hpp board/Board.hpp
$(CC) $(CFLAGS) mcts/Node.cpp
$(OBJ-DIR)/FastMath.o : mcts/FastMath.hpp mcts/FastMath.cpp
$(CC) $(CFLAGS) mcts/FastMath.cpp
$(OBJ-DIR)/Board.o : board/Board.hpp board/Board.cpp board/Patterns.hpp struct/FastSet.hpp struct/FastHash.hpp
$(CC) $(CFLAGS) board/Board.cpp
$(OBJ-DIR)/Patterns.o : board/Patterns.hpp board/Patterns.cpp struct/FastHash2.hpp struct/FastHashMap.hpp
$(CC) $(CFLAGS) board/Patterns.cpp
$(OBJ-DIR)/FastSet.o : struct/FastSet.hpp struct/FastSet.cpp
$(CC) $(CFLAGS) struct/FastSet.cpp
$(OBJ-DIR)/FastHash.o : struct/FastHash.hpp struct/FastHash.cpp
$(CC) $(CFLAGS) struct/FastHash.cpp
$(OBJ-DIR)/FastHash2.o : struct/FastHash2.hpp struct/FastHash2.cpp
$(CC) $(CFLAGS) struct/FastHash2.cpp
$(OBJ-DIR)/FastHashMap.o : struct/FastHashMap.hpp struct/FastHashMap.cpp
$(CC) $(CFLAGS) struct/FastHashMap.cpp
$(TEST-TARGET) : $(TEST-OBJS) $(OBJS)
mv *.o $(OBJ-DIR)
$(CC) $(LFLAGS) $(TEST-OBJS) $(OBJS) -o $(TEST-TARGET)
$(OBJ-DIR)/TestsMain.o : test/TestsMain.cpp test/MCTSTests.hpp test/NodePoolTests.hpp \
test/NodeTests.hpp test/FastMathTests.hpp test/BoardTests.hpp \
test/FastSetTests.hpp test/FastHashTests.hpp test/FastHash2Tests.hpp test/FastHashMapTests.hpp
$(CC) $(CFLAGS) test/TestsMain.cpp
$(OBJ-DIR)/MCTSTests.o : test/MCTSTests.hpp test/MCTSTests.cpp mcts/MCTS.hpp
$(CC) $(CFLAGS) test/MCTSTests.cpp
$(OBJ-DIR)/NodePoolTests.o : test/NodePoolTests.hpp test/NodePoolTests.cpp
$(CC) $(CFLAGS) test/NodePoolTests.cpp
$(OBJ-DIR)/NodeTests.o : test/NodeTests.hpp test/NodeTests.cpp
$(CC) $(CFLAGS) test/NodeTests.cpp
$(OBJ-DIR)/FastMathTests.o : test/FastMathTests.hpp test/FastMathTests.cpp
$(CC) $(CFLAGS) test/FastMathTests.cpp
$(OBJ-DIR)/BoardTests.o : test/BoardTests.hpp test/BoardTests.cpp
$(CC) $(CFLAGS) test/BoardTests.cpp
$(OBJ-DIR)/FastSetTests.o : test/FastSetTests.hpp test/FastSetTests.cpp
$(CC) $(CFLAGS) test/FastSetTests.cpp
$(OBJ-DIR)/FastHashTests.o : test/FastHashTests.hpp test/FastHashTests.cpp
$(CC) $(CFLAGS) test/FastHashTests.cpp
$(OBJ-DIR)/FastHash2Tests.o : test/FastHash2Tests.hpp test/FastHash2Tests.cpp
$(CC) $(CFLAGS) test/FastHash2Tests.cpp
$(OBJ-DIR)/FastHashMapTests.o : test/FastHashMapTests.hpp test/FastHashMapTests.cpp
$(CC) $(CFLAGS) test/FastHashMapTests.cpp
.PHONY : clean
clean :
$(RM) $(OBJ-DIR)/*.o
.PHONY : distclean
distclean :
$(MAKE) clean
$(RM) $(GTP-TARGET) $(TEST-TARGET)
.PHONY : again
again:
$(MAKE) distclean
$(MAKE) $(GTP-TARGET)
$(MAKE) $(TEST-TARGET)
.PHONY : tar
tar:
tar cfv $(GTP-TARGET).tar $(GTP-OBJS) $(OBJS)
tar cfv $(TEST-TARGET).tar $(TEST-OBJS) $(OBJS)
示例输出:
$ make
clang++ -Wall -c -O3 -flto -m64 GtpPlayer.cpp
clang++ -Wall -c -O3 -flto -m64 mcts/MCTS.cpp
clang++ -Wall -c -O3 -flto -m64 mcts/NodePool.cpp
clang++ -Wall -c -O3 -flto -m64 mcts/Node.cpp
clang++ -Wall -c -O3 -flto -m64 mcts/FastMath.cpp
clang++ -Wall -c -O3 -flto -m64 board/Board.cpp
clang++ -Wall -c -O3 -flto -m64 board/Patterns.cpp
clang++ -Wall -c -O3 -flto -m64 struct/FastSet.cpp
clang++ -Wall -c -O3 -flto -m64 struct/FastHash.cpp
clang++ -Wall -c -O3 -flto -m64 struct/FastHash2.cpp
clang++ -Wall -c -O3 -flto -m64 struct/FastHashMap.cpp
mv *.o build/obj
clang++ -Wall -O3 -flto -m64 build/obj/GtpPlayer.o build/obj/MCTS.o build/obj/NodePool.o build/obj/Node.o build/obj/FastMath.o build/obj/Board.o build/obj/Patterns.o build/obj/FastSet.o build/obj/FastHash.o build/obj/FastHash2.o build/obj/FastHashMap.o -o build/go-gtp
clang++ -Wall -c -O3 -flto -m64 test/TestsMain.cpp
clang++ -Wall -c -O3 -flto -m64 test/MCTSTests.cpp
clang++ -Wall -c -O3 -flto -m64 test/NodePoolTests.cpp
clang++ -Wall -c -O3 -flto -m64 test/NodeTests.cpp
clang++ -Wall -c -O3 -flto -m64 test/FastMathTests.cpp
clang++ -Wall -c -O3 -flto -m64 test/BoardTests.cpp
clang++ -Wall -c -O3 -flto -m64 test/FastSetTests.cpp
clang++ -Wall -c -O3 -flto -m64 test/FastHashTests.cpp
clang++ -Wall -c -O3 -flto -m64 test/FastHash2Tests.cpp
clang++ -Wall -c -O3 -flto -m64 test/FastHashMapTests.cpp
mv *.o build/obj
clang++ -Wall -O3 -flto -m64 build/obj/TestsMain.o build/obj/MCTSTests.o build/obj/NodePoolTests.o build/obj/NodeTests.o build/obj/FastMathTests.o build/obj/BoardTests.o build/obj/FastSetTests.o build/obj/FastHashTests.o build/obj/FastHash2Tests.o build/obj/FastHashMapTests.o build/obj/MCTS.o build/obj/NodePool.o build/obj/Node.o build/obj/FastMath.o build/obj/Board.o build/obj/Patterns.o build/obj/FastSet.o build/obj/FastHash.o build/obj/FastHash2.o build/obj/FastHashMap.o -o build/go-test
现在,例如,如果我编辑并保存 Board.cpp 并再次运行 make,我会得到:
$ make
clang++ -Wall -c -O3 -flto -m64 board/Board.cpp
链接命令发生了什么?我想要一个新目标!
【问题讨论】:
-
为什么要移动
*.o文件?如果你在不移动的情况下尝试你的makefile怎么办? (显然,您必须更改几件事才能使其正常工作。)这绝对是一种非标准的做事方式。我看到的“我希望我的.o文件在其他地方”问题的解决方案通常涉及告诉编译器直接在那个地方生成它们,而不是在makefile 中移动它们(这势必会混淆make)。 -
不相关,但您可以将大部分规则简化为
$(OBJ_DIR)/%.0: test/%.hpp test/%.cpp。 -
如果您将
-o $@添加到您的编译器命令中,它们会自动将对象发送到您的对象目录中(因为您的目标期望它:$(OBJ-DIR)/FastHash2.o)。这将使您删除我认为会给您带来额外麻烦的mv命令。我不完全确定如何解决链接器问题,但我认为移动整个 Makefile 以仅列出依赖项将有助于更容易发现。 -
@Greg 我正在移动我的目标文件,而不是告诉编译器将它们输出到其他地方,因为这是我的第一个 makefile,我不知道我在做什么! :)
-
@Maxpm 和 @sarnold 我会尝试这些规则来简化事情。