GUN make 的执行过程分为两个阶段。
MAKIFILES
变量指定的、指示符 include
指定的、以及命令行选项 -f(--file)
指定的 makefile 文件),内建所有的变量、明确规则和隐含规则,并建立所有目标和依赖之间的依赖关系结构链表。在 make 执行的第一阶段中如果变量和函数被展开,那么称此展开是“立即”的,此时所有的变量和函数被展开在需要构建的结构链表的对应规则中(此规则在建立链表是需要使用)。其他的展开称之为“延后”的。这些变量和函数不会被“立即”展开,而是直到后续某些规则需要使用时或者在 make 处理的第二阶段它们才会被展开。
变量和函数的展开时机在变量取值,条件语句和规则的定义中的如下:
变量取值的解析规则如下:
IMMEDIATE = DEFERRED
IMMEDIATE ?= DEFERRED
IMMEDIATE := IMMEDIATE
IMMEDIATE += DEFERRED or IMMEDIATE
define IMMEDIATE
DEFERRED
Endef
条件语句的展开是“立即”的。其中包括:ifdef
、ifeq
、ifndef
和 ifneq
所确定的所有分支命令。
规则的定义按照如下的模式展开:
IMMEDIATE : IMMEDIATE ; DEFERRED
DEFERRED
以上内容介绍的 Makefile 的解析流程,需要有一定的基础。可以先查看后续章节对Makefile的组成元素有一定了解,再返回来查看此章节。本文大部分内容摘自《Makefile中文手册》,如需详细了解请查看原著。后续章节将介绍组成 Makefile 的规则、变量、条件判断、函数和注释。
Makefile 中,规则描述了在何种情况下使用什么命令来重建一个特定的文件。规则中的命令执行的条件是:1.目标文件不存在; 2.存在依赖文件比目标文件新。这是 Makefile 的核心思想。
通常规则的语法格式如下:
TARGETS : PREREQUISITES
COMMAND
...
或者
TARGETS : PREREQUISITES; COMMAND
COMMAND
...
我们需要注意几点:
[tab]
字符开始,命令行运行在系统shell之上。clean
,all
)。Bourne shell
完全相同,比如 *
、?
、[...]
等。\
可以延申物理行,即把几行合并为一行,方便书写和阅读。通过 .PHONY
申明一个目标为伪目标,那么此目标的生成命令总是会被执行。比如:
PHONY : clean
可以避免工作目录下存在同名文件 clean
的情况。
如果一个规则没有命令或者依赖,并且它的目标不是一个存在的文件名。在执行此规则时,目标总会被认为是最新的。这样的目标在作为一个规则的依赖时,规则中定义的命令总会被执行。比如:
clean: FORCE
rm $(objects)
FORCE:
此处的申明方式,和把 clean
申明为伪目标效果相同。
其他特殊目标如 .SUFFIXES
,.DEFAULT
等,请参考《Makefile中文手册》第4.9章。
一个规则可以有多个目标,规则所定义的命令对所有的目标有效。多目标规则意味着所有的目标具有相同的依赖文件。比如:
bigoutput littleoutput : text.g
generate text.g -$(subst output,,$@) > $@
一个文件可以作为多个规则的目标(多个规则中只能有一个规则定义命令)。这种情况时,以这个文件为目标的规则的所有依赖文件将会被合并成此目标一个依赖文件列表,当其中任何一个依赖文件比目标更新(比较目标文件和依赖文件的时间戳)时,make 将会执行特定的命令来重建这个目标。
foo.o : defs.h
foo.o : config.h
规则存在多个目标,并且不同的目标可以根据目标文件的名字来自动构造出依赖文件。
TARGETS ...: TARGET-PATTERN: PREREQ-PATTERNS ...
COMMANDS
...
首先在目标模式和依赖模式中,一般需要包含模式字符 %
。在目标模式(TAGET-PATTERN)中 %
可以匹配目标文件的任何部分,用模式字符 %
匹配的部分替代依赖模式(PREREQ-PATTERNS)中的相应部分来产生对应目标的依赖文件。
模式规则中,目标名中需要包含有模式字符 %
(一个),包含有模式字符 %
的目标被用来匹配一个文件名,%
可以匹配任何非空字符串。例如:
%.o : %.c ; COMMAND...
...
需要注意以下几点:
%
的匹配和替换发生在规则中所有变量和函数引用展开之后。双冒号规则和普通规则的不同体现在以下两个方面:
比如下面例子会根据改变的依赖文件,而执行相应的规则:
Newprog :: foo.c
$(CC) $(CFLAGS) $< -o $@
Newprog :: bar.c
$(CC) $(CFLAGS) $< -o $@
规则的命令由一些shell命令行组成,它们一条一条的执行。每一行必须以 [Tab]
字符开始。多个命令行之间可以有空行和注释行。
如果规则的命令行以字符 @
开始,则 make 在执行这个命令时就不会回显这个将要被执行的命令。比如:@echo 开始编译 XXX 模块 ......
对于多行命令,每一行命令将在一个独立的子 shell 进程中被执行。命令的解析使用环境变量 SHELL
所指定的那个程序,默认的程序是 /bin/sh
。
可以通过 make 的命令行选项 -j
或者 --job
来告诉 make 在同一时刻可以允许多条命令同时被执行。
在 Makefile 中使用 make
作为一个命令来执行本身或者其它 makefile 文件。比如:
subsystem:
$(MAKE) -C subdir
...
为了防止 make 在执行时试图为重建这个目标去查找隐含命令(包括了使用隐含规则中的命令和 .DEFAULT
指定的命令,可以定义一个空命令:
target: ;
Makefile 变量的定义有两种风格,主要区别在展开时机。
IMMEDIATE = DEFERRED
IMMEDIATE ?= DEFERRED
IMMEDIATE := IMMEDIATE
IMMEDIATE += DEFERRED or IMMEDIATE
define IMMEDIATE
DEFERRED
Endef
变量的引用方式是:$(VARIABLE)
或者 ${VARIABLE}
。对一个变量的引用可以在 Makefile 的任何上下文中,目标、依赖、命令、绝大多数指示符和新变量的赋值中。
变量引用的展开过程是严格的文本替换过程,就是说变量值的字符串被精确的展开在变量被引用的地方。
注意:由于 $
符号有特殊意义,因此当需要直接使用 $
时需要转义,需要写两个 $
,即$$
。
使用“替换引用”将其值中的后缀字符(串)使用指定的字符(字符串)替换。格式为$(VAR:A=B)
或者 ${VAR:A=B}
。例如:
foo := a.o b.o c.o
bar := $(foo:.o=.c)
另一种替换的技术使用功能更强大的 patsubst
函数,需要包含模式字符 %
,例如:
foo := a.o b.o c.o
bar := $(foo:%.o=%.c)
从最里层的变量引用开始,逐步向外进行替换。一层层展开直到最后计算出需要应用的具体的变量,之后进行替换展开得到实际的引用值。
x = y
y = z
z = u
a := $($($(x)))
如果通过命令行定义了一个变量,那么它将替代在 Makefile 中出现的同名变量的定义。override 指示符可以避免这种作用。例如:
override VARIABLE = VALUE
自动化变量的取值是根据具体所执行的规则来决定的,取决于所执行规则的目标和依赖文件名。
$@
表示规则的目标文件名。$%
代表静态库的一个成员名,如果目标不是静态库文件,其值为空。$<
规则的第一个依赖文件名。$?
所有比目标文件更新的依赖文件列表,空格分割。$^
规则的所有依赖文件列表,使用空格分隔。$+
类似“$^”,但是它保留了依赖文件中重复出现的文件。$*
在模式规则和静态模式规则中,代表“茎”。其他如 $(@D)
,$(@F)
等请参考 《Makefile中文手册》第10.5.3章
make 在运行时,系统中的所有环境变量对它都是可见的。在 Makefile 中,可以引用任何已定义的系统环境变量。通过 export
指示符可以将一个普通变量变为系统环境变量。
MAKEFILES
VPATH
SHELL
MAKESHELL
MAKE
等,详细介绍请参考《Makefile中文手册》附录第4节
目标指定变量值只在指定它的目标的上下文中有效,对于其他的目标没有影响。语法如下:
TARGET ... : VARIABLE-ASSIGNMENT
模式指定变量定义是将一个变量值指定到所有符合此模式的目标上。语法如下:
PATTERN ... : VARIABLE-ASSIGNMENT
条件语句可以根据一个变量的值来控制 make 执行或者忽略 Makefile 的特定部分。
基本语法如下:
CONDITIONAL-DIRECTIVE
TEXT-IF-TRUE
Endif
或者
CONDITIONAL-DIRECTIVE
TEXT-IF-TRUE
else
TEXT-IF-FALSE
Endif
ifeq
: 参数是否相等ifneq
: 参数是否不相等ifdef
: 变量是否已经定义ifndef
: 变量是否未定义GNU make 函数的调用格式类似于变量的引用,以 $
开始表示一个引用。 语法格式如下:
$(FUNCTION ARGUMENTS)
或者:
${FUNCTION ARGUMENTS}
对于函数调用的格式有以下几点说明:
FUNCTION
是指make 内嵌的函数名,用户自己的函数需要通过 make 的 call
函数来间接调用。ARGUMENTS
是函数的参数,参数和函数名之间使用若干个空格或者 [tab]
字符分割。如果存在多个参数时,参数之间使用逗号 ,
分开。,
和空格。在实际书写 Makefile 时,当有逗号或者空格作为函数的参数时,需要把它们赋值给一个变量,在函数的参数中引用这个变量来实现。下面列出了所有make函数,每个函数的具体使用使用说明请参考 《Makefile中文手册》第8章
#文本处理函数
$(subst FROM,TO,TEXT)
$(patsubst PATTERN,REPLACEMENT,TEXT)
$(strip STRINT)
$(findstring FIND,IN)
$(filter PATTERN…,TEXT)
$(filter-out PATTERN...,TEXT)
$(sort LIST)
$(word N,TEXT)
$(wordlist S,E,TEXT)
$(words TEXT)
$(firstword NAMES…)
#文件名处理函数
$(dir NAMES…)
$(notdir NAMES…)
$(suffix NAMES…)
$(basename NAMES…)
$(addsuffix SUFFIX,NAMES…)
$(addprefix PREFIX,NAMES…)
$(join LIST1,LIST2)
$(wildcard PATTERN)
$(foreach VAR,LIST,TEXT)
$(if CONDITION,THEN-PART[,ELSE-PART])
$(call VARIABLE,PARAM,PARAM,...)
$(value VARIABLE)
$(eval VARIABLE)
$(origin VARIABLE)
$(shell VARIABLE)
$(error TEXT…)
$(warning TEXT…)