GNU Makefile自动依赖解析

GNU Makefile automatic dependency resolve

本文关键字:依赖 Makefile GNU      更新时间:2023-10-16

我过去一直使用qmakeCMake来生成生成文件,没有遇到任何问题。然而,最近我发现在我运行代码的一些集群上,这些工具更难找到/安装,所以我决定写一些空白的Makefile。此外,我还可以学习一些关于Makefiles的知识;)

我的项目涉及多个目录,其中包含源文件/头文件,并且它们之间经常存在交叉依赖关系。通过SO,我学会了使用-MM标志自动生成依赖关系信息。我的Makefile看起来像这个

include $(HEAD_DIR)/common.mk
OBJS_DIR = objs
LIBS_DIR = libs
SRCS  = Matrix.cpp MatrixFull.cpp
OBJS  = $(patsubst %,$(OBJS_DIR)/%,$(SRCS:.cpp=.o))
LIB_0 = $(patsubst %, $(LIBS_DIR)/%,libalgebra.a)
DEPS := $(patsubst %,$(OBJS_DIR)/%,$(SRCS:.cpp=.d))
-include $(DEPS)
.PHONY: all
all:    
    @echo   " ===== Building dependencies in lib/algebra ===== "
    @$(MKDIR) $(OBJS_DIR)
    @$(MAKE)  $(OBJS)
    @$(MKDIR) $(LIBS_DIR)
    @$(MAKE)  $(LIB_0)
    @echo   " ===================== done ===================== "
$(OBJS_DIR)/%.o: %.cpp
    @echo " compiling source file:   $< ..."
    $(CXX) -c $(CXXFLAGS) -I$(INCLUDE_PATH) -MM $< -o $@
$(LIB_0): $($(OBJS_DIR)/%.o)
    @echo " generating library file: $(@F) ..."
    @$(AR) $(AR_FLAGS) $@ $^
.PHONY: clean
clean:
    @$(RM) $(OBJS_DIR) $(LIBS_DIR)

这通常效果良好,但我想在以下几点上进行改进:1) 当我更改头文件并发出make时,不知何故,编译器不会选择all,并且似乎默认为DEPS。这应该发生在这个生成文件中吗?如何使all成为默认规则?2) 如何使depenency文件不可见?我试过DEPS := $(patsubst %,$(OBJS_DIR)/.%,$(SRCS:.cpp=.d)),但没用3) 如果你认为这个Makefile可以通过其他方式改进,你能给我什么具体的建议吗?

编辑:老实说,我甚至不确定make是如何调用编译器来生成依赖项的:p。。。我看不出有任何明确的规则。它是否以某种方式隐藏在DEPS := $(patsubst %,$(OBJS_DIR)/%,$(SRCS:.cpp=.d))中?

1)将-include $(DEPS)向下移动到all规则下方。第一个规则是默认规则(除非您故意设置一些特殊变量),所以如果include引入高于all:的规则,那将是默认规则。

2) 我认为"让它们不可见"的意思是你想要.foo.d而不是foo.d。你必须修改产生它们的规则:

$(OBJS_DIR)/%.o: %.cpp
    @echo " compiling source file:   $< ..."
    $(CXX) -c $(CXXFLAGS) -I$(INCLUDE_PATH) -MM $< -o $@
    @mv $(OBJS_DIR)/$*.d $(OBJS_DIR)/.$*.d

然后是找到它们的变量:

DEPS := $(patsubst %.cpp,$(OBJS_DIR)/.%.d,$(SRCS))

3) 我会去掉递归调用($(MAKE) ...),但您应该首先确保其他一切都正常工作。我很惊讶$(OBJS_DIR)/%.o规则没有出现在common.mk中。除此之外,它看起来还不错。

4(?)$(OBJS_DIR)/%.o规则中的编译器命令使用-MM标志,因此编译器生成依赖文件作为副作用。