make:设置一个复杂的先决条件列表

make: setting up a complex prerequisit list

我在为我的用例编写正确的 makefile 目标时遇到问题。

我有一个可以生成 header 和源文件的脚本。生成 header myHeader.h 的输入文件是 config/myHeader.h.template 和 config/myHeader.xlsx。 可以看到,输入文件不在 header 需要的目录中,而是在 "config" 子目录中。

在 makefile 中,我有一个列表,包含脚本生成的所有 headers 和来源。

MY_GENERATED_HEADERS =  $(sort \
            foo/bar/myHeader1.h \
            foo/bar/myHeader2.h)

此外,我有一个可以从 "outside":

调用的目标
generate: $(MY_GENERATED_HEADERS)

所以当我写

make generate

在命令行上,我想检查列表中的每个 header 是否需要再次生成。

所以,现在是复杂的部分。 我必须编写一个目标,当 .h.template 或 .xlsx 发生变化时最终调用我的脚本。

$(MY_GENERATED_HEADERS): %.h: %.h.template %.xlsx
    @echo $@
    #invoke the script...

在规则中,我可以设置调用我的脚本所需的绝对文件名。没问题,只要使用一些自动变量和字符串操作函数即可。

实际问题是先决条件列表:在上述解决方案中,假设 .h.template 和 .xlsx 文件与输出 header 在同一目录中(foo/bar /),但是,我希望它们位于 foo/bar/config 中以稍微清理文件系统。

在上面的解决方案中,我会得到错误信息

No rule to make target 'foo/bar/myHeader1.h.template'.

是的,没错,因为 myHeader1.h.template 在 foo/bar/config...

我已将所有目录添加到 VPATH,但这根本没有改变任何内容。

有什么想法吗?

什么Garf365 suggested cannot work unless you use .SECONDEXPANSION 开启二次扩展依赖。下面是一个 Makefile,它说明了解决方案的工作原理:

# Some value for the sake of illustration.
MY_GENERATED_HEADERS=foo/bar/1.h foo/baz/2.h

all: $(MY_GENERATED_HEADERS)

.SECONDEXPANSION:
$(MY_GENERATED_HEADERS): %.h: $$(dir %)config/$$(notdir %).h.template $$(dir %)config/$$(notdir %).xlsx
    @echo Target: $@
    @echo Dependencies: $^

我在发布示例之前已经测试过它。

当您使用 .SECONDEXPANSION 时,您必须注意在正确的时间扩展您的变量和函数调用。第一次扩展是在第一次读取规则时完成的 before % 获取任何值(自动变量相同)。第二次扩展是在 Make 即将对特定目标应用规则时完成的:那时,% 有一个值(自动变量也是如此)。您在第二次扩展时使用 $$ 来表示您想要扩展的内容。第一次扩容时,依赖会转化为:

$(dir %)config/$(notdir %).h.template $(dir %)config/$(notdir %).xlsx

然后在第二次扩展时(每个目标发生一次)% 将被替换为它的值并且 $(dir ...) 和 $(notdir ...) 将被调用.