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 ...) 将被调用.
我在为我的用例编写正确的 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 ...) 将被调用.