假设我有一个像这样的源文件列表:
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仅使用适当的源文件。 在通用规则中单独指定每个目标的依赖关系的正确方法是什么?
答案 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创建它们比使每个配方检查目录首先存在更有效和优雅。