Makefile-关键字和函数(五)
流程控制
-
ifeq
ifeq (arg1, arg2)
commond
endif
如果arg1和arg2相等,那么执行命令commond,或者
ifeq (arg1, arg2)
commond1
else
commond2
endif
如果表达式如果arg1和arg2相等,那么执行命令commond1,否则执行commond2 -
ifneq
语法和ifeq一样,但是含义为arg1和arg2不相等则执行commond或者执行commond1 -
ifdef
ifdef arg1
commond
endif
如果arg1非空,那么执行命令commond,或者
ifdef arg1
commond1
else
commond2
endif
如果arg1飞空,那么执行命令commond1,否则执行commond2 -
ifndef
语法和ifdef一样,只不过含义相反 -
vpath
make默认只在当前被执行的Makefile所在目录搜索目标文件,用来指定搜索路径,那么make将按照当前目录,指定路径的顺序搜索目标文件)。使用方式:
vpath <pattern> <directories>;定义符合模式的文件的搜索路径
vpath <pattern> ;取消符合模式的文件的搜索路径
vpath;取消所有之前定义的搜索路径,采用默认 -
%模式匹配
%的使用场景很多,如下:
1、变量中使用模式匹配
$(var %.c=%.o);代表将变量var中所有.c结尾的字串替换成.o结尾的字串
$(var %=%.o);代表将变量var中每个字串结尾添加.o
$(var %=test/%);代表将变量var中每个字串的前面添加test/
objects=a.c b.c c.c d.c
$(info $(objects %.c=%.o))
$(info $(objects %=test/%))
2、目标规则中使用模式匹配
假如有如下的文件集合
target_1_test.o target_2_test.o target_3_test.o test1.o test2.o
target_%_test:target_%_test.c test1.o test2.o
cc $< $^ -o $@
相当于:
target_1_test:target_1_test.o
cc target_1_test.o test1.o test2.o -o target_1_test
target_2_test:target_2_test.o
cc target_2_test.o test1.o test2.o -o target_2_test
target_3_test:target_3_test.o
cc target_3_test.o test1.o test2.o -o target_3_test
@ 一般成对使用于目标规则含有模式匹配时,Makefile会自动展开。
@表示模式匹配到的每一个目标文件
-
备注:例子中test1.o test2.o虽然也是依赖文件,但不属于模式匹配中的文件,所以不包括在^则表示所有的依赖目标文件(会去重),例子中^的每一次展开后未去重应该分是
target_1_test.o target_1_test.o test1.o test2.o
target_2_test.o target_2_test.o test1.o test2.o
target_3_test.o target_3_test.o test1.o test2.o
故去重后就是上面的结果了。^一样,只不过不去重。 -
.SUFFIXES:
代表删除makefile默认的后缀隐含规则,有助于加快makefile的编译解析速度 -
.PHONY:
定义一个目标为伪目标
# 清除makefile默认的后缀隐含规则,加快解析编译速度
.SUFFIXES:
# 定义伪目标(伪目标代表着这个目标下的规则一定会被执行,而正常目标是如果依赖文件没有更新则目标下的规则不会被执行)
.PHONY: all all-yes alltools build check config testprogs
.PHONY: *clean install* uninstall*
函数定义与调用
-
$(function arguments)
这是Makefile内建函数的调用方式。function为函数名,arguments为参数,如果有多个参数,以逗号","分隔。 -
Makefile自定义函数
define funname
command1
command2
....
endef
它其实不叫函数,称为命令包,调用方式为 $(call funname,arg1,arg2,...argn);在命令包内部,通过$(1),$(2),...$(n)来代表对应的参数arg1,arg2,...argn
define funtest
$(1)-yes :=$(2)
endef
$(info $(call funtest,config,true))
最终config的值为config-yes的值为true。
Makefile其实没有自定义函数的概念,上面define关键字只是将多个命令组合在一起执行罢了,称为命令包,它的返回值是命令包内的所有语句组成的makefile代码字符串。比如对于上面(call funtest,config,true)),$(call funtest,config,true)的返回值就是 config-yes=true这一条代码字符串
define RESET
$(1) :=
$(1)-yes :=
endef
$(call RESET,config)
这个函数的功能就是将config和config-yes这两个变量赋值为空
常用函数
1、$(call expression,parm1,parm2,...,parmn)
执行表达式expression(由$(1),$(2),...$(n)组成,方式顺序不自由组合),该表达式的中的$(1),$(2),...$(n)则对应到parm1,parm2,...,parmn的值
reverse = $(2) $(1)
foo := $(call reverse,a,b)
最终foo的值为 b a
2、$(foreach var,list,text)
依次从list变量中取出每个字串(空格分开)赋值给var,然后执行text表达式,直到循环完毕将每次text表达式的值添加到一起作为foreach的返回值
names := a b c d
files := $(foreach n,$(names),$(n).o)
最终files的值为 a.o b.o c.o d.o
3、$(eval text)
将text 的内容作为makefile的一部分而被make解析和执行。
pointer := pointed_value
define foo
var := 123
arg := $1
$$($1) := ooooo
endef
$(info $(call foo,pointer))
$(eval $(call foo,pointer))
target:
@echo -----------------------------
@echo var: $(var), arg: $(arg)
@echo pointer: $(pointer), pointed_value: $(pointed_value)
@echo done.
@echo -----------------------------
最终执行结果为:
var := 123
arg := pointer
$(pointer) := ooooo
-----------------------------
var: 123, arg: pointer
pointer: pointed_value, pointed_value: ooooo
done.
eval 常常和call函数,自定义命令包一起使用,用来动态的执行命令包所定义的代码,如上例子。