如何使用GNUMake指定目标相关的先决条件?

时间:2016-04-29 12:40:34

标签: build-process gnu-make

假设我有一个像这样的源文件列表:

SOURCE=aaa.src bbb.src ccc.src

我想创建一个规则来在自己的文件夹中构建每个相应的目标,其名称取决于soruce文件。

  

Sourcefile ---->对应的输出文件/目标

           

aaa.src -------------------> AAA / aaa.out

     

bbb.src -------------------> BBB / bbb.out

     

ccc.src -------------------> CCC / ccc.out

如何使用GNUMake为此编写规则?我最大的努力是以下Makefile:

.PHONY: all clean

CC=somecompiler

SOURCE := aaa.src bbb.src ccc.src
RTARGS := $(SOURCE:%.src=%.out)
TDIRS := $(addsuffix /,$(basename $(SOURCE)))
TARGS := $(join $(TDIRS), $(RTARGS))

all:$(TARGS)

%.out: $(SOURCE)     # Better source specification?
    @[ ! -d "$(dir $*)" ] && mkdir "$(dir $*)"
    $(CC) "$(patsubst %/,%,$(dir $*)).src" "$@"

clean:
    rm -f $(TARGS)
    rmdir --ignore-fail-on-non-empty $(TDIRS)

这里的问题是,任何一个目标(即.out文件)依赖于每个源(即.src文件),而不仅仅是具有相同的基名。我可以将注释行更改为%.out: %.src,但源文件必须与输出文件位于同一目录中。我也可以这样编译:

%.out: %.src
    $(CC) "$>" -o "$*/$@"

但是make会始终编译每个目标,无论它是否已经存在。

如何使make仅使用适当的源文件。 在通用规则中单独指定每个目标的依赖关系的正确方法是什么?

1 个答案:

答案 0 :(得分:1)

在GNU Make中,您可以与食谱分开指定先决条件,并且仍然具有所有.out文件的通用/模式规则:

.PHONY: all clean

all :

SOURCES := aaa.src bbb.src ccc.src
OUTPUTS := $(foreach src,${SOURCES},$(basename ${src})/${src:%.src=%.out})

# Build dependencies in the form of x/x.out : x.src | x
define ESTABLISH_DEPENDENCY =
$(basename ${1})/${1:%.src=%.out} : ${1} | $(basename ${1}) # Also depend on the output directory.
$(basename ${1}) : # Rule to create the output directory.
    mkdir $$@
endef

$(foreach src,${SOURCES},$(eval $(call ESTABLISH_DEPENDENCY,${src})))

all : ${OUTPUTS}

# Generice rule for all .out files.
%.out :
    @echo "build $@ from $^"
    touch $@

%.src : # For debugging only.
    touch $@

.PHONY: all

输出:

$ make
touch aaa.src
mkdir aaa
build aaa/aaa.out from aaa.src
touch aaa/aaa.out
touch bbb.src
mkdir bbb
build bbb/bbb.out from bbb.src
touch bbb/bbb.out
touch ccc.src
mkdir ccc
build ccc/ccc.out from ccc.src
touch ccc/ccc.out

请注意,对目录使用仅依赖顺序依赖并让make创建它们比使每个配方检查目录首先存在更有效和优雅。