1.何时扩展变量

当make在处理递归变量或define指令的时候,会将变量里的每一行或宏的主体存储起来,包括换行符号,但不会予以扩展。

宏定义里的最后一个换行符号并不会被存储称宏的一部分,否则,宏被扩展时make会读进一个额外的换行符号。

当宏被扩展时,make会立即扫描被扩展的文本中是否存在宏或变量的引用,如果存在就予以扩展,如此递归进行下去。如果

宏是在命令脚本的语境中被扩展的,则宏主体的每一行都会被插入一个前导的跳格符。

下面是用来处理makefile中元素何时被扩展的准则:

1.对于变量的赋值,make会在第一阶段读进该行时,立即扩展赋值运算符左边的部分。

2.=和?=的右边部分会被延后到它们被使用的时候扩展,并且在第二阶段进行。

3.:=的右边部分会被立即扩展。

4.如果+=的左边部分原来被定义成一个简单变量,+=的右边部分就会被立即扩展,否则,它的求值动作会被延后。

5.对于宏定义,宏的变量名称会被立即扩展,宏的主体会被延后到使用的时候扩展。

6.对于规则,工作目标的必要条件总是被立即扩展,然而命令总是会延后扩展。

2.工作目标与模式的专属变量

在makefile运行期间,变量通常只有一个值,对需要经过两个处理阶段的makefile来说是这样没错。第一个阶段,make读取

makefile之后,会对变量进行赋值的扩展的动作并建立依存图。第二阶段,make会分析以及遍历依存图。所以等到make执行

命令脚本的时候,所有变量都已经处理完毕。但是如果我们想为特定的规则或模式重新定义变量,该怎么办?

现在,我们想要编译一个需要额外命令行选项-DUSE_NEW_MALLOC=1的文件,但是其他的编译项目并不需要这个额外的

命令行选项:

gui.o:gui.h

        $(COMPILE.c) -DUSE_NEW_MALLOC=1 $(OUTPUT_OPTION) $<

我们必须复制编译命令脚本以及为它加入这个必要的选项。如果有许多文件需要经过类似的特别处理,整个工作会变得冗长

乏味。

为解决此类问题,make提供了工作目标的专属变量,这些变量的定义会附在工作目标之上,且只有该工作目标以及相应的

任何必要条件被处理的时候,它们才会发生作用。通过使用此功能,我们可以改写成如下:

gui.o:CPPFLAGS += -DUSE_NEW_MALLOC=1
gui.o:gui.h
        $(COMPILE.c) $(OUTPUT_OPTION) $<

变量CPPFLAGS内置在默认在C编译规则里,用来保存供C预处理器使用的选项,通过使用+=附加运算符,我们可以把这个

新的选项附加到任何已存在的值里。

 

工作目标的专属变量的语法如下:

target...:variable = value
target...:variable := value
target...:variable += value
target...:variable ?= value

以上的语法只能用来定义工作目标的专属变量,此类变量在被赋值之前,并不需要事先存在。

此外,这类变量的赋值动作会延后到开始处理工作目标的时候进行,所以赋值运算符右边赋值的值,可由另一个目标的

专属变量来设定。同样地,此变量只有必要条件的处理期间,才会发生作用。

 

3.变量来自何处

make的变量可以有以下几个来源:

文件:

变量被定义在makefile中,或是被makefile引入。(include指令)

 

命令行:

直接在make命令行上定义或重新定义变量:

$make CFLAGS=-g CPPFLAGS='-DBSD -DDEBUG'

每个命令行参数中所包含的等号,都是一个变量赋值运算符。在命令行上,每个变量赋值运算符的右边部分必须是一个单独

的shell参数。如果变量的值包含空格,则必须为参数加上括号或者规避空格。

 

环境:

当make启动时,所有来自环境的变量都会被自动定义成make的变量。这些变量具有非常低的优先级,所以makefile文件

或命令行参数的赋值结果将会覆盖掉环境变量的值。不过,可以使用--environment-overrides(或-e)命令选项,让环境变量

覆盖掉相应的makefile变量。

 

自动创建:

make会在执行一个规则的命令脚本之前立即创建自动变量。

 

4.条件指令

当make所读进的makefile使用了条件处理指令时,makefile文件中有些部分会被省略,有些部分会被挑选处理,用来控制

是否选择的条件具有各种形式。

条件指令的基本语法如下所示:

if-condition
  text if the condition is true
endif

或:

if-condition
  text if the condition is true
else
  text if the condition is false
endif

if-condition可以是以下之一:

ifdef variable-name
ifndef variable-name
ifeq test
ifneq test

进行ifdef/ifndef的测试时,不应该以$()括住variable-name,test可以表示成下面这样:

"a" "b"
或:
(a,b)

ifeq和ifneq条件指令可用来测试其参数是否相等。条件指令里空格的处理有些微妙。举例来说,如果参数采用小括号的

形式,那么逗号之后的空格将会被忽略,除此之外所有其他的空格都是有意义的:

ifeq (a,a)
    #these are equal
endif
 
ifeq ( b, b )
    #these are not equal - ' b' != 'b '
endif

我比较喜欢使用等效的引号形式:

ifeq "a" "a"
    #these are equal
endif
 
ifeq 'b' 'b'
    #so are these
endif

 

5.include指令

一个makefile可以引入其他文件。此功能常用来引入make头文件中所存放的共同的make的定义,或是用来自动产生依存

关系的信息,include指令的用法如下:

include definitions.mk

你可以为这个指令提供任意多个文件,shell通配符以及make变量。

 

当make看到include指令时,会事先对通配符以及变量引用进行扩展的动作,然后试着读进引用文件(include file)。如果

这个文件存在,则整个处理过程会继续下去,然而,如果这个文件不存在,则make会汇报此文件并且继续读取其余的makefile。

当所有的读取动作皆已完成以后,make会从规则数据库中找出任何可用来更新引入文件的规则。如果找到一个相符的规则,

make就会按照正常的步骤更新工作目标。如果任何一个引入文件被规则更新,make接着会清除它的内部数据库并且重新

读取整个makefile。如果完成读取、更新和重新读取的过程之后,仍有include指令因为文件不存在而执行失败,那么make

就会显示错误状态并终止执行。

 

6.标准的make变量

除了自动变量,make还会为“自己的状态以及内置规则的定义”提供变量,一边对外提供相关信息:

MAKE_VERSIN:GNU make的版本编号。

CURDIR:正在执行make进程的当前工作目录,此变量的值将会时shell变量PWD的值,除非make在运行时用到了--directory

(或-C)选项。--directory选项会使得make在搜索任何makefile之前变更到不同的目录。这个选项的 完整形式是:

--directory=directory-name 或 -C directory-name。

MAKEFILE_LIST:make所读进的每个makefile文件的名称所构成的列表,包括默认的makefile以及命令行或include指令

所指定的makefile。每个makefile被读进make之前,其文件名会被附加到该变量里。

MAKECMDGOALS:对当前所运行的make而言,make运行时命令行上指定了那些工作目标。

.VARIABLES:到目前为止,make从各个makefile文件所读进的变量的名称所构成的列表,不含工作目标的专属变量,此

变量仅供读取,对它所进行的任何赋值动作都会被忽略掉。