【问题标题】:Generate gcc dependencies with full path生成具有完整路径的 gcc 依赖项
【发布时间】:2017-12-12 23:17:34
【问题描述】:

我有一个看起来像这样的简单项目

.
├── build
│   ├── file1.o
│   └── one
│       ├── file1.o
│       └── file2.o
├── .depend
├── Makefile
└── src
    ├── file1.cpp
    └── one
        ├── file1.cpp
        └── file2.cpp

Makefile 是这样的:

# Get all of the source files
SRC = $(shell find src/ -name "*.cpp")
# Get all of the object files
OBJ = $(subst src,build,$(SRC:.cpp=.o))

$(OBJ):
    @mkdir -p $(shell dirname $@)
    g++ -g -c $(subst build,src,$(subst .o,.cpp,$@)) -o $@

all: depend build

build: $(OBJ)
    gcc -o project $^

depend:
    g++ -MM $(SRC) > .depend
    sed -i 's/.\/src/.\/build\//g' .depend

sinclude .depend

我正在尝试通过运行 g++ -MM src/file1.cpp src/one/file1.cpp src/one/file2.cpp > .depend 来生成 makefile 依赖项,它会生成以下指令:

file1.o: src/file1.cpp <other headers>
file1.o: src/one/file1.cpp <other headers>
file2.o: src/one/file2.cpp <other headers>

问题在于build/file1.ofile1.o 不匹配,因此,更改src/file1.cpp 或它所依赖的任何标头都不会导致目标文件被重建。起初我认为这可能是在生成 .depend 文件之前运行sinclude .depend 的问题,但即使我运行make depend 后跟make build,问题仍然存在。从我读过的所有内容来看,没有g++ 参数或选项可以保留名称的路径。

是否可以通过这种方式生成依赖文件,或者这是构建项目的根本错误方法?

我查看了这个问题被标记为可能重复的问题的答案,但似乎问题是询问如何为项目创建完整的 makefile,而我的问题不在于创建Makefile,而是gcc -MM 依赖生成的问题。这个问题的答案并没有解决我的问题。

【问题讨论】:

  • 我在latedev.wordpress.com/2014/11/08/… 有几篇关于此的博客文章可能会有所帮助
  • 您希望目标是什么样的?你不能再做一个sed 替换来按照你想要的方式修复它吗?
  • 感谢您的链接,但我在那里找不到我的问题的答案。我仍然不知道如何让g++ -MM 包含完整路径,或者如何让%.o: %.cpp 匹配build/file1.o
  • 据我所知,在这种情况下我无法使用sed 替换,因为这种情况涉及完全丢失的信息。 sed 将允许我删除与模式匹配的路径信息,但我遇到的问题是路径信息从不存在。
  • 有一种方法可以做到这一点,但是您确定要拥有一个大的依赖文件 (.depend),而不是每个目标文件都有一个吗?后一种方法有一些优势,但这是您的决定。

标签: c++ makefile g++


【解决方案1】:

怎么样:

# Get all of the source files
SRC = $(shell find src/ -name "*.cpp")
# Get all of the object files
OBJ = $(patsubst src/%.cpp,build/%.o,$(SRC))

.PHONY: all

all: project

project: $(OBJ)
    gcc -o $@ $^

$(OBJ): build/%.o: src/%.cpp
    @mkdir -p $(dir $@)
    g++ -g -c $< -o $@

.depend: $(SRC)
    g++ -MM $^ > $@ && \
    sed -Ei 's#^(.*\.o: *)src/(.*/)?(.*\.cpp)#build/\2\1src/\2\3#' $@

include .depend

依赖计算

sed 命令替换任何:

file.o: src/file.cpp ...

作者:

build/file.o: src/file.cpp ...

和任何:

file.o: src/X/Y/Z/file.cpp ...

作者:

build/X/Y/Z/file.o: src/X/Y/Z/file.cpp ...

目标直接是.depend,它具有所有源文件作为依赖项,因此如果丢失或比任何源文件旧,它会自动重建。无需使用depend 假目标或将其添加为all 的先决条件(如果需要,make 会自动尝试重建include 中包含的文件)。

注意

我添加了一些 GNU make 功能(patsubst、静态模式规则、自动变量的系统使用...)如果您使用其他 make,请重做不支持的功能。

【讨论】:

  • 谢谢,这正是我想要的。我唯一注意到的是@mkdir $(dir $@) 将在构建文件夹不存在的情况下失败,并且创建的第一个目录的深度超过一层(例如build/folder1/file.o)。我可以通过在命令中添加-p 选项来解决此问题,将其转换为@mkdir -p $(dir $@)
  • @NicholasHollander 哎呀,对不起。编辑我的答案以添加 -p 选项。
  • 是否有充分的理由编写 .depend 文件,然后编写 sed -i 而不是通过 sed 的简单管道?
  • @TobySpeight 除了美观之外没有其他原因:我不喜欢长线,也不喜欢在管道上断线。没什么重要的。
【解决方案2】:

这里有三种方法。

一、用 sed 修改输出(类似于 Renaud Pacalet 的回答):

depend:
    g++ -MM $(SRC) | sed 's/.*: src\([^ ]*\)cpp/build\1o: src\1cpp/' > .depend                  

二、使用shell循环:

STEMS := file1 one/file1 one/file2
depend:
    rm -f .depend
    for x in $(STEMS); do g++ -MM -MT build/$$x.o src/$$x.cpp >> .depend; done

三、Make方法:

DEPENDENCIES := $(addsuffix -depend,$(STEMS))

clear-depend:
    rm -f .depend

depend: $(DEPENDENCIES)

%-depend: clear-depend
    g++ -MM -MT build/$*.o src/$*.cpp >> .depend

(我最喜欢的做法是为每个目标文件建立一个单独的依赖文件,而不是一个大的.depend 文件。它有几个优点,但是需要一些时间来解释,如果有名称冲突也很棘手在您的源代码树中,例如 file1.cppfile1.cpp。)

【讨论】:

    【解决方案3】:

    我使用以下顺序进行构建:

    define req
    $(subst ..,__,$(dir build-$(TARGET)$(build_dir_ext)/$(1)))%.o: $(dir $1)%.cpp
        mkdir -p $$(dir $$(subst ..,__,$$@))
        $$(CXX) -MM $$(CXXFLAGS) $$< -MT   $$(subst ..,__,$$@) > $$(patsubst %.o,%.d,$$(subst ..,__,$$@))
        $$(CXX)     $$(CXXFLAGS) $$< -c -o $$(subst ..,__,$$@)
    endef
    
    $(eval $(foreach x,$(OBJ),$(call req,$(x))))
    

    因此,make 现在能够处理可以在源树“外部”的路径,只需使用 '__' 而不是 '..' 并根据找到的模式设置构建目录,所以有srcbuild 不再有问题。我需要“外部”文件来使用源池,其中本地构建目录位于源池和项目目录的根目录下。

    希望对您有所帮助...

    编辑 1:为什么要替换 '..'

    想想下面的源代码树:

    ./sourcepool/lib1/src/one.cpp
    ./sourcepool/project/build
    

    如果您的 Makefile 位于 ./sourcepool/project 路径中且 OBJ 之一是“../lib1/src/one.o”,则 Makefile 应在构建目录中创建等效路径。也就是说,如果使用“..”,则不可能,因为路径不再是构建中的,而是更高的一个深度。如果将 .. 替换为 __ 结果如下:

    ./sourcepool/project/build/__/lib1/src/one.o
    

    这使得不将所有使用的目录复制或链接到本地​​项目和构建文件树成为可能。

    【讨论】:

    • 我完全不明白:你为什么要替换..
    • 为了避免这种诡计,GNUmake 之神引入了abspath 函数。它将所有... 解析为规范路径名,之后可以以正常方式与其他替换和路径函数一起使用。
    • @Vroomfondel:如果使用 abspath,我将在我的构建路径下复制完整的文件系统结构。我的解决方案只增加了所需的深度。
    • abspath 将您的../lib1/src/one.cpp 变成/home/klaus/sourcepool/lib1/src/one.cpp(或它所在的任何路径)。我不知道您所说的构建路径下的整个结构是什么意思。它只是一个字符串函数。
    • 试试这个:PRJROOT = $(abspath ..)OUTOFTREE := $(wildcard $(PRJROOT)/lib1/src/*.cpp)$(info $(foreach f,$(OUTOFTREE),$(subst $(PRJROOT),$(PRJROOT)/build,$(f))))
    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 2014-02-21
    • 2016-02-13
    • 2022-01-01
    • 2021-01-12
    • 2012-01-03
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多