【问题标题】:Makefile: depend on every file of a directoryMakefile:依赖于目录的每个文件
【发布时间】:2011-08-05 03:24:24
【问题描述】:

我想做一个使用 gnumake 或 makepp 运行的 Makefile,它可以打包给定目录下的所有文件:

DIRS:=$(shell find . -mindepth 2 -maxdepth 2 -not -name mp3 -not -name ".*" -type d)
PACKAGES = $(DIRS:%=%.npk)

all: packages

packages: $(PACKAGES)

%.npk: %/*
    npack c $@ @^

.PHONY: all packages

问题是依赖项中没有 %/* 这样的东西。 我需要目标(X.npk)依赖于目录 X 中的每个文件,但我不知道在编写 Makefile 时这些文件是什么,因为它们是稍后生成的。

一个例子:

./dirA/x
./dirA/y
./dirB/e
./dirB/f

我想创建 ./dirA.npk (取决于 x,y), ./dirB.npk (e,f) 除了第一行中使用的 find 找到所有目录之外,我对目录或文件一无所知。

【问题讨论】:

  • 我试过了:%.npk: $(wildcard %/*) 但这不起作用。我也考虑过 semafor,但遇到了同样的问题,我需要依赖 target_dir/*。

标签: makefile makepp


【解决方案1】:

尝试使用wildcard 指令:

DEPS := $(foreach dir, $(DIRS), $(wildcard $(dir)/*))

%.npk: $(DEPS)
    npack c $@ $^

编辑: 以上只是使用wildcard 的示例,它使每个 .npk 文件依赖于所有其他文件夹中的文件。您的用法会略有不同。

我认为可能有更简单的方法来解决这个问题。为什么要依赖文件夹中的所有文件?只是使用$^ 运算符吗?或者,如果任何文件发生更改,您是否需要重新构建 .npk?

另一种(可能更简洁)的解决方案是在您的配方中使用find 实用程序而不是$^,并使用.FORCE 指令始终强制重建.npk 文件。缺点是 .npk 文件可能会不必要地重建。

编辑 2: 如果没有办法使用make 命令干净地执行此操作,您可以使用.FORCE 解决它以确保始终运行配方并将“我应该重建此文件”检查移动到配方正文:

%.npk: .FORCE
    check_for_rebuild.sh $@ && npack c $@ $^

check_for_rebuild.sh 是一个执行以下操作的 shell 脚本:

#!/bin/bash
# Returns non-zero if the archive needs to be rebuilt
if [ -e $1 ]; then
    folder_name=$(basename $1 .npk)
    [ -z "$(find $folder_name -newer $1 -not -type d)" ] && return 0
fi
return 1

我不是很喜欢这个解决方案,因为它可以解决问题而不是直接解决问题,但它或许可以同时让你继续前进。如果你打算走那条路,那么在 shell 脚本中做所有事情可能会更干净、更容易,要么让 makefile 简单地调用脚本,要么完全摆脱 makefile。

【讨论】:

  • 没错,如果有任何文件更改或添加了新文件,我需要重建 npk 文件。 .FORCE 不是一个选项,我正在尝试做相反的事情(目前我有一个将所有内容重新打包到 X.npk.tmp 中的 shell 脚本,然后如果 cmp X.npk X.npk.tmp 显示已更改,我重命名 tmp 文件,否则我将其删除。但是由于我有更多的目录和文件,它变得越来越慢。
【解决方案2】:

这是我找到的解决方案: 它基于makedepend 的想法,带有一些“元”脚本。不是很好,但有效。

PACKAGES :=

all: packages

-include Makefile.depend

packages: Makefile.depend $(PACKAGES)

depend: clean Makefile.depend

Makefile.depend:
    @(PACKAGES= ; \
    for DIR in `find . -mindepth 2 -maxdepth 2 -not -name mp3 -not -name ".*" -type d` ; \
    do \
        PACKAGE=`basename $${DIR}.npk` ; \
        PACKAGES="$${PACKAGES} $${PACKAGE}" ; \
        DEPS=`find $${DIR} -not -type d | sed -e 's#\([: ]\)#\\\\\1#' -e 's#^\./\(.*\)# \1#' | tr -d "\n"` ; \
        SUBDIR=`echo $${DIR} | sed -e 's#^\./\([^/]\+\)/.*#\1#'` ; \
        FILES=`echo \ $${DEPS} | sed -e "s# $${SUBDIR}/# #g"` ; \
        echo "$${PACKAGE}:$${DEPS}" ; \
        echo "  @cd $${SUBDIR} ; \\" ; \
        echo "  npack c ../\$$@ $${FILES} ; \\" ; \
        echo ; \
    done ; \
    echo "PACKAGES = $${PACKAGES}" \
    )>> Makefile.depend ; \

cleanall: clean
    rm -f *.npk

clean:
    @rm -f Makefile.depend

.PHONY: all packages depend clean

【讨论】:

  • 使用make -j<n>depend 目标可能会失败,这对我来说是这样。移动Makefile.depend 内的目标depend 配方应该修复它。
【解决方案3】:

使用 makepp,您可以通过 :foreach 规则修饰符分两步完成此操作:

$(foreach).txt: $(foreach)/*: foreach */
    &echo $(inputs) -o $(output)

这为每个子目录提供了一个规则,只要其中的文件列表发生更改,该规则就会重新执行。

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 2018-08-27
    • 2012-12-26
    • 2016-10-21
    • 1970-01-01
    • 2021-02-24
    • 2017-10-03
    • 1970-01-01
    • 2014-01-23
    相关资源
    最近更新 更多