Makefile: 依赖于目录中的每个文件

7
我想做一个 Makefile 文件,可以在 GNU Make 或 Makepp 中运行,并打包给定目录下的所有文件:
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 %/*),但是不起作用。我也考虑过信号量,但遇到了同样的问题,我需要依赖于target_dir/*。 - Gavriel
3个回答

4
尝试使用“通配符”指令: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不是一个选项,我试图做相反的事情(目前我有一个shell脚本将所有内容重新打包成X.npk.tmp,然后如果cmp X.npk X.npk.tmp显示已更改,我将重命名临时文件,否则我将删除它。但随着我拥有更多的目录和文件,速度变慢了。 - Gavriel

1
这是我找到的解决方案: 它基于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配方中应该可以解决它。 - akhan

0

使用makepp,您可以通过:foreach规则修改器在2个步骤中完成此操作:

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

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


网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接