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

时间:2011-04-13 23:32:54

标签: makefile makepp

我想做一个使用gnumake或makepp运行的Makefile,它包含给定directiories下的所有文件:

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) 除了第一行中使用的查找找到所有dirs之外,我没有提前知道dirs或文件。

3 个答案:

答案 0 :(得分:2)

尝试使用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。

答案 1 :(得分: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

答案 2 :(得分:0)

使用makepp,您可以通过:foreach rule modifier:

分两步完成
$(foreach).txt: $(foreach)/*: foreach */
    &echo $(inputs) -o $(output)

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