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

Makefile:依赖于目录中的每个文件,makefile,makepp,Makefile,Makepp,我想创建一个运行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

我想创建一个运行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)
我事先对目录或文件一无所知,只知道第1行中使用的查找会查找所有目录。

尝试使用
通配符
指令:

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

%.npk: $(DEPS)
    npack c $@ $^
编辑: 以上只是使用
通配符的一个示例,使每个.npk文件依赖于所有其他文件夹中的文件。你的用法会略有不同

我想可能有一个更简单的方法。为什么要依赖文件夹中的所有文件?是否仅使用
$^
运算符?或者,如果任何文件发生更改,是否需要重新生成.npk

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

编辑2: 如果无法使用
make
命令干净地执行此操作,您可以使用
.FORCE
解决此问题,以确保配方始终运行,并将“我是否应重建此文件”复选框移到配方正文中:

%.npk: .FORCE
    check_for_rebuild.sh $@ && npack c $@ $^
其中
check\u for\u 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。

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

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

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

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

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

我试过:%.npk:$(通配符%/*)但不起作用。我还考虑了semafor,但遇到了同样的问题,我需要依赖target_dir/*。确切地说,如果任何文件发生更改或添加了新文件,我需要重建npk文件。武力不是一种选择,我正试着做相反的事情(目前我有一个shell脚本,可以将所有内容重新打包到X.npk.tmp中,然后如果
cmp X.npk X.npk.tmp
显示已更改,我将重命名tmp文件,否则我将删除它。但是由于我有更多的dir和文件,速度会变慢。使用
make-j
dependent
目标可能会失败,这在我的情况下就是如此。移动
Makefile.dependent
)>配方中的目标
dependent
recipe应该修复它。