Make 的基本用法
==============
> 本文是我对《陈皓 - 跟我一起写 Makefile》的学习笔记
makefile 的规则
---
一个 makefile 可以有很多个 rules,一个 rule 长这样:
```makefile
target ... : prerequisites ...
recipe
...
...
```
- target:可以是一个 object file(目标文件),也可以是一个可执行文件,还可以是一个标签(label)。
- prerequisites:生成该 target 所依赖的文件和/或 target。
- recipe:该 target 要执行的命令(任意的 shell 命令)。
一个 rule 包含三个部分
- 一个或多个 targets
- 0 个或多个 dependencies
- 0 个或多个 commands(recipe)
这是一个文件的依赖关系,也就是说,target 这一个或多个的目标文件依赖于 prerequisites 中的文件,其生成规则定义在 command 中。说白一点就是说:
```
prerequisites中如果有一个以上的文件比target文件要新的话,recipe所定义的命令就会被执行。
```
这就是 makefile 的规则,也就是 makefile 中最**核心**的内容。
重要参数:
- `-n` : dry run
- `-f` : 指定 makefile
- `-s` : silent/quiet,静默模式,不显示任何输出
规则说明:
- recipe 中的命令默认使用`/bin/sh`解释 shell 命令
- 输入`make target`意味着
1. 确定所有的依赖都是最新的
2. 如果 target 比任何一个 dependency 旧,则重新构建 target
- 输入`make`默认构建 Makefile 中的第一个 target
- Phony target(伪目标):伪目标的名字并不表示真的要生成这样一个文件,伪目标仅包含 recipe 和 target,不包含任何 dependency
命令的开头
---
- recipe 中的命令一定要以一个Tab键作为开头,**不能用空格代替**
- recipe 中的命令若以`-`开头,表示如果命令执行出错,继续执行下一条命令
- recipe 中的命令若以`@`开头,表示命令本身不会输出,但命令的输出(如有)会输出
命令的执行
---
make 会一条一条执行 recipe 中的命令,需要注意的是,如果你要让上一条命令的结果应用在下一条命令时,你应该使用分号分隔这两条命令。比如你的第一条命令是 cd 命令,你希望第二条命令得在 cd 之后的基础上运行,那么你就不能把这两条命令写在两行上,而应该把这两条命令写在一行上,用分号分隔。如:
```makefile
#1
exec:
cd /home/hchen
pwd
#2
exec:
cd /home/hchen; pwd
```
当我们执行 `make exec` 时,第一个例子中的 cd 没有作用,pwd 会打印出当前的 Makefile 目录,而第二个例子中,cd 就起作用了,pwd 会打印出“/home/hchen”。
嵌套执行 make
---
在一些大的工程中,我们会把我们不同模块或是不同功能的源文件放在不同的目录中,我们可以在每个目录中都书写一个该目录的 Makefile,这有利于让我们的 Makefile 变得更加地简洁,而不至于把所有的东西全部写在一个 Makefile 中,这样会很难维护我们的 Makefile,这个技术对于我们模块编译和分段编译有着非常大的好处。
例如,我们有一个子目录叫 subdir,这个目录下有个 Makefile 文件,来指明了这个目录下文件的编译规则。那么我们总控的 Makefile 可以这样书写:
```makefile
subsystem:
cd subdir && $(MAKE)
```
其等价于:
```makefile
subsystem:
$(MAKE) -C subdir
```
定义$(MAKE) 宏变量的意思是,也许我们的 make 需要一些参数,所以定义成一个变量比较利于维护。这两个例子的意思都是先进入“subdir”目录,然后执行 make 命令。
我们把这个 Makefile 叫做“总控 Makefile”,总控 Makefile 的变量可以传递到下级的 Makefile 中(如果你显示的声明),但是不会覆盖下层的 Makefile 中所定义的变量,除非指定了 `-e` 参数。
定义命令包
---
如果 Makefile 中出现一些相同命令序列,那么我们可以为这些相同的命令序列定义一个变量。定义这种命令序列的语法以 `define` 开始,以 `endef` 结束,如:
```makefile
define run-yacc
yacc $(firstword $^)
mv y.tab.c $@
endef
```
这里,“run-yacc”是这个命令包的名字,其不要和 Makefile 中的变量重名。在 `define` 和 `endef` 中的两行就是命令序列。这个命令包中的第一个命令是运行 Yacc 程序,因为 Yacc 程序总是生成“y.tab.c”的文件,所以第二行的命令就是把这个文件改改名字。还是把这个命令包放到一个示例中来看看吧。
```makefile
foo.c : foo.y
$(run-yacc)
```
我们可以看见,要使用这个命令包,我们就好像使用变量一样。在这个命令包的使用中,命令包“run-yacc”中的 `$^` 就是 `foo.y` , `$@` 就是 `foo.c` (有关这种以 `$` 开头的特殊变量,我们会在后面介绍),make 在执行命令包时,命令包中的每个命令会被依次独立执行。
使用变量
---
在 Makefile 中的定义的变量,就像是 C/C++语言中的宏一样,他代表了一个文本字串,在 Makefile 中执行的时候其会自动原模原样地展开在所使用的地方。其与 C/C++所不同的是,你可以在 Makefile 中改变其值。在 Makefile 中,变量可以使用在“目标”,“依赖目标”, “命令”或是 Makefile 的其它部分中。
命名规则:变量的命名字可以包含字符、数字,下划线(可以是数字开头),但不应该含有 `:` 、 `#` 、 `=` 或是空字符(空格、回车等)。变量是大小写敏感的,“foo”、“Foo”和“FOO”是三个不同的变量名。传统的 Makefile 的变量名是全大写的命名方式,但我推荐使用大小写搭配的变量名,如:MakeFlags。这样可以避免和系统的变量冲突,而发生意外的事情。
有一些变量是很奇怪字串,如 `$<` 、 `$@` 等,这些是自动化变量,我会在后面介绍。
> macro `@` evaluates to the name of the current target.
> 可用`make -p`打印内部宏
变量在声明时需要给予初值,而在使用时,需要给在变量名前加上 `$` 符号,但最好用小括号 `()` 或是大括号 `{}` 把变量给包括起来。如果你要使用真实的 `$` 字符,那么你需要用 `$$` 来表示。
变量可以使用在许多地方,如规则中的“目标”、“依赖”、“命令”以及新的变量中。先看一个例子:
```makefile
objects = program.o foo.o utils.o
program : $(objects)
cc -o program $(objects)
$(objects) : defs.h
```
变量会在使用它的地方精确地展开,就像 C/C++中的宏一样,例如:
```makefile
foo = c
prog.o : prog.$(foo)
$(foo)$(foo) -$(foo) prog.$(foo)
```
展开后得到:
```makefile
prog.o : prog.c
cc -c prog.c
```
当然,千万不要在你的 Makefile 中这样干,这里只是举个例子来表明 Makefile 中的变量在使用处展开的真实样子。可见其就是一个“替代”的原理。
另外,给变量加上括号完全是为了更加安全地使用这个变量,在上面的例子中,如果你不想给变量加上括号,那也可以,但我还是强烈建议你给变量加上括号。
与C/C++不同,为变量赋值时,右侧变量可以是后面定义的变量:
```makefile
foo = $(bar)
bar = $(ugh)
ugh = Huh?
all:
echo $(foo)
```
我们执行“make all”将会打出变量 `$(foo)` 的值是 `Huh?` ( `$(foo)` 的值是 `$(bar)` , `$(bar)` 的值是 `$(ugh)` , `$(ugh)` 的值是 `Huh?` )可见,变量是可以使用后面的变量来定义的。
还有另一种使用变量的方式(推荐):
```makefile
x := foo
y := $(x) bar
x := later
```
其等价于:
```makefile
y := foo bar
x := later
```
值得一提的是,这种方法,**前面的变量不能使用后面的变量**,只能使用前面已定义好了的变量。如果是这样:
```makefile
y := $(x) bar
x := foo
```
那么,y 的值是“bar”,而不是“foo bar”。
总结一下:
- `=`操作符允许先使用变量,后为变量赋值,但容易引发递归定义的问题
- `:=`操作符遵循常规变量先定义后使用的原则,推荐使用
### 行尾注释的副作用
```makefile
nullstring :=
space := $(nullstring) # essential for one space
dir := /foo/bar # dir for xxx
all:
@echo "$(space),$(dir),$(nullstring),hehe"
```
```bash
$ make -n
echo " ,/foo/bar ,,hehe"
$ make
,/foo/bar ,,hehe
```
注意其中的空格,由此可见,行尾注释之前的空格也会被附加到变量值中。如果行尾没有注释,space 变量将没有空格,dir 变量也将恢复正常,没有后面的空格。**用“#”注释符来表示变量定义的终止**。这样,我们可以定义出其值是一个空格的变量。
### `?=` 操作符
```makefile
FOO ?= bar
```
其含义是,如果 FOO 没有被定义过,那么变量 FOO 的值就是“bar”,如果 FOO 先前被定义过,那么这条语将什么也不做,其等价于:
```makefile
ifeq ($(origin FOO), undefined)
FOO = bar
endif
```
### `+=` 操作符
我们可以使用 `+=` 操作符给变量追加值,如:
```makefile
objects = main.o foo.o bar.o utils.o
objects += another.o
```
于是,我们的 `$(objects)` 值变成:“main.o foo.o bar.o utils.o another.o”(another.o 被追加进去了)。它等价于下面的写法:
```makefile
objects = main.o foo.o bar.o utils.o
objects := $(objects) another.o
```
很明显,`+=` 更简洁。
如果变量之前没有定义过,那么, `+=` 会自动变成 `=` ,如果前面有变量定义,那么 `+=` 会继承于前次操作的赋值符。如果前一次的是 `:=` ,那么 `+=` 会以 `:=` 作为其赋值符。
仍然,小心使用`=`和`+=`时引发的递归定义:
```makefile
v = $(value)
value += $v
all:
@echo "value is $(value)"
```
```bash
$ make
Makefile:6: *** Recursive variable 'value' references itself (eventually). Stop.
```
### 目标变量
前面我们所讲的在 Makefile 中定义的变量都是“全局变量”,在整个文件,我们都可以访问这些变量[^a]。当然,我也同样可以为某个目标设置局部变量,这种变量被称为“Target-specific Variable”,它可以和“全局变量”同名,因为它的作用范围只在这条规则以及连带规则中,所以其值也只在作用范围内有效。而不会影响规则链以外的全局变量的值。
其语法是:
```makefile
: ;
: overide
```
`;`可以是前面讲过的各种赋值表达式,如 `=` 、 `:=` 、 `+=` 或是 `?=` 。第二个语法是针对于 make 命令行带入的变量,或是系统环境变量。
这个特性非常的有用,当我们设置了这样一个变量,这个变量会作用到由这个目标所引发的所有的规则中去。如:
```makefile
prog : CFLAGS = -g
prog : prog.o foo.o bar.o
$(CC) $(CFLAGS) prog.o foo.o bar.o
prog.o : prog.c
$(CC) $(CFLAGS) prog.c
foo.o : foo.c
$(CC) $(CFLAGS) foo.c
bar.o : bar.c
$(CC) $(CFLAGS) bar.c
```
在这个示例中,不管全局的 `$(CFLAGS)` 的值是什么,在 prog 目标,以及其所引发的所有规则中(prog.o foo.o bar.o 的规则), `$(CFLAGS)` 的值都是 `-g`.
### 高级用法
拼接:
```makefile
first_second = Hello
a = first
b = second
all = $($a_$b)
```
这里的 `$a_$b` 组成了“first_second”,于是,`$(all)` 的值就是“Hello”。当然,“把变量的值再当成变量”这种技术,同样可以用在操作符的左边:
```makefile
dir = foo
$(dir)_sources := $(wildcard $(dir)/*.c)
define $(dir)_print
lpr $($(dir)_sources)
endef
```
这个例子中定义了三个变量:“dir”,“foo_sources”和“foo_print”。
Reference
---
- [跟我一起写 Makefile - 陈皓](https://seisman.github.io/how-to-write-makefile)
- [A Short Introduction to Makefile](https://www3.nd.edu/~zxu2/acms60212-40212/Makefile.pdf)
[^a]:当然,“自动化变量”除外,如 `$<` 等这种类量的自动化变量就属于“规则型变量”,这种变量的值依赖于规则的目标和依赖目标的定义。