[英]Makefile improvements, dependency generation not functioning
我正在尝试构建一个正确的Makefile。
我想要的是完全控制正在发生的事情,所以我不想要任何第三方软件。
我当前的尝试对我来说似乎是逻辑,但由于依赖关系生成无效,我有点卡住了。
为了更好的可读性,完整的Makefile被分成几小块。 如果有什么需要改进,我将不胜感激任何评论。
首先,我有以下静态定义
CXX = g++
CXXFLAGS = -Wall \
-Wextra \
-Wuninitialized \
-Wmissing-declarations \
-pedantic \
-O3 \
-p -g -pg
LDFLAGS = -p -g -pg
DEPFLAGS = -MM
Afaik这应该没问题。 使分析标志可选是完美的,但这并不重要。
SRC_DIR = ./src
OBJ_DIR = ./obj
SRC_EXT = .cpp
OBJ_EXT = .o
TARGET = ./bin/my_target
SRCS = $(wildcard $(SRC_DIR)/*$(SRC_EXT))
OBJS = $(subst $(SRC_DIR), $(OBJ_DIR), $(SRCS:$(SRC_EXT)=$(OBJ_EXT)))
DEP = depend.main
基本上,这应该只是提取所有的*.cpp
文件从子文件夹的src
和另外更换./src
与./obj
和.cpp
用.o
为对象的名称。
.PHONY: clean all depend
all: $(TARGET)
$(TARGET): $(OBJS)
@echo "-> linking $@"
@$(CXX) $^ $(LDFLAGS) -o $@
$(OBJ_DIR)/%.$(EXT_OBJ):
@echo "-> compiling $@"
@$(CXX) $(CXXFLAGS) -c $< -o $@
Afaik,这个块 - 提供了一个有效的依赖文件 - 应该做所有必要的编译和链接。
clean:
@echo "removing objects and main file"
@rm -f $(OBJS) $(TARGET)
应该是不言自明和正确的,还是我在这里遗漏了什么?
$(SRC_DIR)/%.$(SRC_EXT):
$(CXX) $(DEPFLAGS) -MT \
"$(subst $(SRC_DIR),$(OBJ_DIR),$(subst $(SRC_EXT),$(OBJ_EXT),$@))" \
$(addprefix ,$@) >> $(DEP);
clear_dependencies:
@echo "-> (re-)building dependencies";
@$(RM) $(DEP)
depend: clear_dependencies $(SRCS)
这是非功能性部分。 我打算做的是使用g++
Compiler标志-MM
来自动创建依赖项,并使用-MT
来使用与默认路径不同的路径。 产生的依赖关系应该是这样的
./obj/main.o: ./src/main.cpp ./src/some_header_file.h
不幸的是,这永远不会被调用,我不知道为什么会这样。 在一个类似的问题中 ,用户Beta很乐意通过添加.Phony
提供临时解决方案,但这对重建每个对象没有任何改变有副作用。
最后,只有一行
-include $(DEP)
一旦创建,就包含依赖文件。
任何提供任何部分提示的答案都非常受欢迎。 所以我的问题是:我能做得更好或者更“干净”,为什么依赖生成不起作用?
开始。
SRCS := $(wildcard $(SRC_DIR)/*$(SRC_EXT))
来自GNU Make手册 :
[递归扩展变量]的另一个缺点是每次扩展变量时都会执行定义中引用的任何函数。 这使得
make
运行速度变慢; 更糟糕的是,它会导致wildcard
和shell
函数产生不可预测的结果,因为您无法轻易控制它们何时被调用,甚至无法控制多少次。
使用替换引用或patsubst
函数将源转换为对象:
OBJS := $(SRCS:$(SRC_DIR)/%$(SRC_EXT)=$(OBJ_DIR)/%$(OBJ_EXT))
在编译模式规则中指定适当的先决条件。 这是必须的,以使您的目标文件保持最新并在源更改时更新它们。
$(OBJ_DIR)/%$(OBJ_EXT) : $(SRC_DIR)/%$(SRC_EXT)
@echo "-> compiling $@"
@$(CXX) $(CXXFLAGS) -o $@ -c $<
编译源并同时为它们生成依赖文件。 使用-MMD -MP
标志可以使事情正常工作(只需将它们附加到CXXFLAGS
)。
CXXFLAGS += -MMD -MP
-include $(OBJS:$(OBJ_EXT)=.d)
从GCC手册 :
-MD
-MD
等效于-M -MF
文件 ,但不暗示-E
。 驱动程序根据是否给出-o选项来确定文件。 如果是,则驱动程序使用其参数但后缀为.d
,否则它将获取输入文件的名称,删除任何目录组件和后缀,并应用.d
后缀。
-MMD
像
-MD
一样,只提到用户头文件,而不是系统头文件。
-MP
此选项指示CPP为除主文件之外的每个依赖项添加虚假目标,从而使每个依赖项都不依赖。 这些假规则将避开错误
make
如果删除头文件时不更新给Makefile
相匹配。
还要考虑学习Paul Smith的这篇文章 (他是GNU Make的维护者)。 它给出了不同自动生成方法的相当好的概述。
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.