Makefile-4-书写命令
前言
- 本笔记主要记录Makefile一些概念要点。
概念
Chapter 4:书写命令
- 每条规则中的命令和操作系统 Shell 的命令行是一致的。
- 每条命令必须以 Tab 键开头,除非,命令是紧跟在依赖规则后面的分号后的。
- make 一般是使用环境变量 SHELL 中所定义的系统 Shell 来执行命令,默认情况下是 /bin/sh —— UNIX 的标准 Shell 解释执行的。
4.1 显示命令
- 是否显示命令
- 在命令行前使用 @ 字符,那么该命令将不被make显示出来。
- 只显示,不执行
- make 执行时,带入 make 参数 -n 或 –just-print。
- 作用:一般用于调试,查看make执行的顺序。
- 全面禁止,命令的显示:
- make 参数 -s 或 –silent 或 –quiet 则是全面禁止命令的显示。
4.2 命令执行 *
- 当依赖目标新于目标时,make 会一条一条的执行其后的命令。
-
如果要让上一条命令的结果应用在下一条命令时,你应该使用分号分隔这两条命令。
- 例子1:输出为:【当前目录路径】
exec:
cd /home/lss
pwd
* 例子2:输出为:/home/lss
exec:
cd /home/lss; pwd
- 命令解析器寻找顺序
- 首先,在 SHELL 所指定的路径中找寻 –> 在当前盘符中的当前目录中寻找 –> 在 PATH 环境变量中所定义的所有路径中寻找。
- MS-DOS 中,如果定义的命令解释器没有找到,其会给你的命令解释器加上诸如 .exe 、.com 、.bat 、.sh 等后缀。
4.3 命令出错
- 每当命令运行完后,make 会检测每个命令的返回码,码为零则成功。
- 忽略命令失败
- 方法一:减号
- 在命令前( Tab 键后)加上一个减号 – 即可。
- 方法二:全局(分两种)
- 给 make 加上 -i 或是 –ignore-errors 参数
- 果一个规则是以 .IGNORE 作为目标的,该规则中的所有命令将会忽略错误。
- 方法一:减号
- 命令失败,但只终止当前规则
-
make 的参数的是 -k 或是 –keep-going
- 意思是:某规则中的命令出错了,就终止该规则的执行,但继续执行其它规则。
-
make 的参数的是 -k 或是 –keep-going
4.4 嵌套执行 make
- 总控 Makefile + 各个子目录 Makefile。
- 父级 Makefile
- -C 切换到其它目录
subsystem:
make -C subdir # 或者:cd subdir && make
-
传递变量到下级,声明方法:
export <variable ...>;
-
注:注意 等号:(意思是 export 后,就不是 延时变量)
-
export variable = value
等价于export variable := value
- 传递时默认不覆盖,加上 -e 即可覆盖下级重新赋值的变量
-
-
不想让该变量传递到下级,声明方法:
unexport <variable ...>;
-
注意两个变量 *
- SHELL 和 MAKEFLAGS 这两个变量不管你是否 export,其总是要传递到下层 Makefile 中。
-
MAKEFLAGS
- 包含了 make 的参数信息。
- 不想让 MAKEFLAGS 往下传递的做法
subsystem:
cd subdir && $(MAKE) MAKEFLAGS=
-
不往下传递的 参数
- make 命令中 -C , -f , -h, -o 和 -W 参数是不往下传递的。
-
信息打印 *
-
-w 或是 –print-directory 会在 make 的过程中输出目前的工作目录。如:执行
make -w
时
-
-w 或是 –print-directory 会在 make 的过程中输出目前的工作目录。如:执行
# 进入该目录时会答应如下信息
make: Entering directory `/home/hchen/gnu/make'.
# 在完成下层 make 后离开目录时:
make: Leaving directory `/home/hchen/gnu/make'
* 当你使用 **-C** 参数来指定 **make** 下层 **Makefile** 时,**-w** 会被自动打开的。如果参数中有 **-s(--slient)**或是 **--no-print-directory** ,那么,**-w** 总是失效的。
4.5 定义命令包
- 相当于 C 中的函数。
- 语法以 define 开始,以 endef 结束。
- 例子:
-
run-yacc
- 命令包名字
- 中间两句就是 命令包内容
-
run-yacc
define run-yacc
yacc $(firstword $^)
mv y.tab.c $@
endef
- 调用方法
- 采用 $
foo.c:foo.y
$(run-yacc)
参考
- 《GUN Makefile》
- 《跟我一起写Makefile》