Makefile入门

1.Makefile概述:

  什么是makefile?或许不少Winodws的程序员都不知道这个东西,由于那些Windows的IDE都为你作了这个工做,但我以为要做一个好的和professional的程序员,makefile仍是要懂。这就好像如今有这么多的HTML的编辑器,但若是你想成为一个专业人士,你仍是要了解HTML的标识的含义。特别在Unix下的软件编译,你就不能不本身写makefile了,会不会写makefile,从一个侧面说明了一我的是否具有完成大型工程的能力。c++

  由于,makefile关系到了整个工程的编译规则。一个工程中的源文件不计数,其按类型、功能、模块分别放在若干个目录中,makefile定义了一系列的规则来指定,哪些文件须要先编译,哪些文件须要后编译,哪些文件须要从新编译,甚至于进行更复杂的功能操做,由于makefile就像一个Shell脚本同样,其中也能够执行操做系统的命令。程序员

  makefile带来的好处就是——“自动化编译”,一旦写好,只须要一个make命令,整个工程彻底自动编译,极大的提升了软件开发的效率。make是一个命令工具,是一个解释makefile中指令的命令工具,通常来讲,大多数的IDE都有这个命令,好比:Delphi的make,Visual C++的nmake,Linux下GNU的make。可见,makefile都成为了一种在工程方面的编译方法。编辑器

  在这篇文档中,将以C/C++的源码做为咱们基础,因此必然涉及一些关于C/C++的编译的知识,相关于这方面的内容,还请各位查看相关的编译器的文档。这里所默认的编译器是UNIX下的GCC和CC。ide

2.Makefile规则(显示规则, 隐晦规则, 变量定义, 文件指示, 注释)

   2.1Makefile基本编写格式:

target ... : prerequisites ...
    command
    ...
    ...

  target:    目标文件,能够是Object File,也能够是执行文件。还能够是一个标签(Label)工具

  prerequisites: 要生成那个target所依赖的文件或是目标。优化

  command:   也就是make须要执行的命令。(任意的Shell命令)ui

  这是一个文件的依赖关系,也就是说,target这一个或多个的目标文件依赖于prerequisites中的文件,其生成规则定义在command中。说白一点就是说,prerequisites中若是有一个以上的文件比target文件要新的话,command所定义的命令就会被执行。这就是Makefile的规则。也就是Makefile中最核心的内容。spa

  2.2编写规则说明:

  1. 显示规则 :: 说明如何生成一个或多个目标文件(包括 生成的文件, 文件的依赖文件, 生成的命令)
  2. 隐晦规则 :: make的自动推导功能所执行的规则($@ $* $^ ...)
  3. 变量定义 :: Makefile中定义的变量
  4. 文件指示 ::其包括了三个部分,一个是在一个Makefile中引用另外一个Makefile,就像C语言中的include同样;另外一个是指根据某些状况指定Makefile中的有效部分,就像C语言  中的预编译#if同样;还有就是定义一个多行的命令。有关这一部分的内容,我会在后续的部分中讲述。
  5. 注释     :: Makefile只有行注释 "#", 若是要使用或者输出"#"字符, 须要进行转义, "\#"

  简单Makefile展现以下:操作系统

include ../Make.defines

PROGS =    test01 \
           test02

all:    ${PROGS}
test01:    test01.o
        ${CC} ${CFLAGS} -o $@ test01.o ${LIBS}
test02:    test02.o
        ${CC} ${CFLAGS} -o $@ test02.o ${LIBS}
        
clean:
        rm -f ${PROGS} ${CLEANFILES}

   事例中几点注意以下:命令行

  1.首行导入了其余的Makefile相关文件,事例假定此文件为二级目录,导入文件为一级目录下的Make.define文件

  2.第二行显示定义了变量PROGS, test01后面的 \为换行符,在参数较多时方便代码查看。

  3.all表示一个标签,在此例中表示全部目标,即${PROGS}所指定的全部执行文件。clean同理。

  4.命令行必定要以TAB开头(Makefile规定)

  5.命令行定义中使用了隐晦规则,$@表示目标文件,在此例中即指代test01或test02.

  6.命令行中的变量均为Make.define文件中定义,表示编译所用的参数
  

3.Makefile工做方式

  为方便说明,简化上述Makefile以下:

include ../Make.defines

test01:    test01.o
        ${CC} ${CFLAGS} -o $@ test01.o ${LIBS}
        
clean:
        rm -f ${PROGS} ${CLEANFILES}

  在默认的方式下,在make命令后:

    一、make会在当前目录下找名字叫“Makefile”或“makefile”的文件。
    二、若是找到,它会找文件中的第一个目标文件(target),即显示定义的test01。他会找到“test01”这个文件,并把这个文件做为最终的目标文件。
    三、若是test01文件不存在,或是test01所依赖的后面的 .o 文件的文件修改时间要比all这个文件新,那么,他就会执行后面所定义的命令来生成test01这个文件。
    四、若是test01所依赖的.o文件也存在,那么make会在当前文件中找目标为.o文件的依赖性,若是找到则再根据那一个规则生成.o文件。(这有点像一个堆栈的过程)
    五、固然,你的C文件和H文件是存在的啦,因而make会生成 .o 文件,而后再用 .o 文件生命make的终极任务,也就是执行文件test01了。

 4.Makefile中一些杂项说明

  4.1 文件包含

  Makefile支持文件包含功能,相似于c/c++中的#include功能,最开始的例子已经使用,实例模型能够参考。其具体使用方法介绍以下:

  include <filename>        #  filename能够是当前操做系统Shell的文件模式(能够保含路径和通配符)

   在include前面能够有一些空字符,可是毫不能是[Tab]键开始。include和<filename>;能够用一个或多个空格隔开。举个例子,你有这样几个Makefile:a.mk、b.mk、c.mk,还有一个文件叫foo.make,以及一个变量$(bar),其包含了e.mk和f.mk,那么,下面的语句:

    include foo.make *.mk $(bar)

    等价于:

    include foo.make a.mk b.mk c.mk e.mk f.mk

  make命令开始时,会把找寻include所指出的其它Makefile,并把其内容安置在当前的位置。就好像C/C++的#include指令同样。若是文件都没有指定绝对路径或是相对路径的话,make会在当前目录下首先寻找,若是当前目录下没有找到,那么,make还会在下面的几个目录下找:

    一、若是make执行时,有“-I”或“--include-dir”参数,那么make就会在这个参数所指定的目录下去寻找。
    二、若是目录<prefix>;/include(通常是:/usr/local/bin或/usr/include)存在的话,make也会去找。

若是有文件没有找到的话,make会生成一条警告信息,但不会立刻出现致命错误。它会继续载入其它的文件,一旦完成makefile的读取,make会再重试这些没有找到,或是不能读取的文件,若是仍是不行,make才会出现一条致命信息。若是你想让make不理那些没法读取的文件,而继续执行,你能够在include前加一个减号“-”。如:

    -include <filename>;
    其表示,不管include过程当中出现什么错误,都不要报错继续执行。和其它版本make兼容的相关命令是sinclude,其做用和这一个是同样的。

  4.2 几种赋值符号差别

  Makefile中赋值符号中四种(=  :=  ?= +=),区别以下:

  1. = 是最基本的赋值,他会将整个Makefile展开后,将最后一个值付给变量。

  x = foo
    y = $(x) bar
    x = xyz

    在上例中,y的值将会是 xyz bar ,而不是 foo bar 。

  2. := 是覆盖以前的值,无需所有展开,在赋值处直接覆盖。

   x := foo
   y := $(x) bar
   x := xyz

   在上例中,y的值将会是 foo bar ,而不是 xyz bar 了。

  3. ?= 是若是没有被赋值过就赋予等号后面的值(容易理解)
  4. += 是添加等号后面的值(字面含义,直接追加)

  4.3 编译参数整理

  4.3.1: GCC参数整理:  

  在Makefile编译命令行中(即commend),第一个例子的${CCFLAG}. 变量中定义了一些编译器须要处理的工做(如:自动导出头文件包含 -M; 添加GDB调试:-g; 优化等级:-o ~ -o2), gcc选项以下

  用法:gcc [选项] 文件...
  选项:
    -pass-exit-codes         在某一阶段退出时返回最高的错误码
    --help                   显示此帮助说明
    --target-help            显示目标机器特定的命令行选项
    (使用‘-v --help’显示子进程的命令行参数)
    -dumpspecs               显示全部内建 spec 字符串
    -dumpversion             显示编译器的版本号
    -dumpmachine             显示编译器的目标处理器
    -print-search-dirs       显示编译器的搜索路径
    -print-libgcc-file-name  显示编译器伴随库的名称
    -print-file-name=<库>    显示 <库> 的完整路径
    -print-prog-name=<程序>  显示编译器组件 <程序> 的完整路径
    -print-multi-directory   显示不一样版本 libgcc 的根目录
    -print-multi-lib         显示命令行选项和多个版本库搜索路径间的映射
    -print-multi-os-directory 显示操做系统库的相对路径
    -Wa,<选项>               将逗号分隔的 <选项> 传递给汇编器
   -Wp,<选项>               将逗号分隔的 <选项> 传递给预处理器
    -Wl,<选项>               将逗号分隔的 <选项> 传递给连接器

    -Wall         打开全部编译警告
    -Xassembler <参数>       将 <参数> 传递给汇编器
    -Xpreprocessor <参数>    将 <参数> 传递给预处理器
    -Xlinker <参数>          将 <参数> 传递给连接器
    -combine                 将多个源文件一次性传递给汇编器
    -save-temps              不删除中间文件
    -pipe                    使用管道代替临时文件
    -time                    为每一个子进程计时
    -specs=<文件>            用 <文件> 的内容覆盖内建的 specs 文件
    -std=<标准>              指定输入源文件遵循的标准
    --sysroot=<目录>         将 <目录> 做为头文件和库文件的根目录
    -B <目录>                将 <目录> 添加到编译器的搜索路径中
    -b <机器>                为 gcc 指定目标机器(若是有安装)
    -V <版本>                运行指定版本的 gcc(若是有安装)
    -v                       显示编译器调用的程序
    -###                     与 -v 相似,但选项被引号括住,而且不执行命令
    -E                       仅做预处理,不进行编译、汇编和连接
    -S                       编译到汇编语言,不进行汇编和连接
    -c                       编译、汇编到目标代码,不进行连接
    -o <文件>                输出到 <文件>
    -x <语言>                指定其后输入文件的语言容许的语言包括:c c++ assembler none
                           ‘none’意味着恢复默认行为,即根据文件的扩展名猜想
                           源文件的语言

  以 -g、-f、-m、-O、-W 或 --param 开头的选项将由 gcc 自动传递给其调用的不一样子进程。若要向这些进程传递其余选项,必须使用 -W<字母> 选项。

  4.3.2: make参数整理(摘自跟我一块儿写Makefile)

下面列举了全部GNU make 3.80版的参数定义。其它版本和产商的make大同小异,不过其它产商的make的具体参数仍是请参考各自的产品文档。

“-b”
“-m”
这两个参数的做用是忽略和其它版本make的兼容性。

“-B”
“--always-make”
认为全部的目标都须要更新(重编译)。

“-C <dir>;”
“--directory=<dir>;”
指定读取makefile的目录。若是有多个“-C”参数,make的解释是后面的路径之前面的做为相对路径,并以最后的目录做为被指定目录。如:“make –C ~hchen/test –C prog”等价于“make –C ~hchen/test/prog”。

“—debug[=<options>;]”
输出make的调试信息。它有几种不一样的级别可供选择,若是没有参数,那就是输出最简单的调试信息。下面是<options>;的取值:
    a —— 也就是all,输出全部的调试信息。(会很是的多)
    b —— 也就是basic,只输出简单的调试信息。即输出不须要重编译的目标。
    v —— 也就是verbose,在b选项的级别之上。输出的信息包括哪一个makefile被解析,不须要被重编译的依赖文件(或是依赖目标)等。
    i —— 也就是implicit,输出因此的隐含规则。
    j —— 也就是jobs,输出执行规则中命令的详细信息,如命令的PID、返回码等。
    m —— 也就是makefile,输出make读取makefile,更新makefile,执行makefile的信息。

“-d”
至关于“--debug=a”。

“-e”
“--environment-overrides”
指明环境变量的值覆盖makefile中定义的变量的值。

“-f=<file>;”
“--file=<file>;”
“--makefile=<file>;”
指定须要执行的makefile。

“-h”
“--help”
显示帮助信息。

“-i”
“--ignore-errors”
在执行时忽略全部的错误。

“-I <dir>;”
“--include-dir=<dir>;”
指定一个被包含makefile的搜索目标。可使用多个“-I”参数来指定多个目录。

“-j [<jobsnum>;]”
“--jobs[=<jobsnum>;]”
指同时运行命令的个数。若是没有这个参数,make运行命令时能运行多少就运行多少。若是有一个以上的“-j”参数,那么仅最后一个“-j”才是有效的。(注意这个参数在MS-DOS中是无用的)

“-k”
“--keep-going”
出错也不中止运行。若是生成一个目标失败了,那么依赖于其上的目标就不会被执行了。

“-l <load>;”
“--load-average[=<load]”
“—max-load[=<load>;]”
指定make运行命令的负载。

“-n”
“--just-print”
“--dry-run”
“--recon”
仅输出执行过程当中的命令序列,但并不执行。

“-o <file>;”
“--old-file=<file>;”
“--assume-old=<file>;”
不从新生成的指定的<file>;,即便这个目标的依赖文件新于它。

“-p”
“--print-data-base”
输出makefile中的全部数据,包括全部的规则和变量。这个参数会让一个简单的makefile都会输出一堆信息。若是你只是想输出信息而不想执行makefile,你可使用“make -qp”命令。若是你想查看执行makefile前的预设变量和规则,你可使用“make –p –f /dev/null”。这个参数输出的信息会包含着你的makefile文件的文件名和行号,因此,用这个参数来调试你的makefile会是颇有用的,特别是当你的环境变量很复杂的时候。

“-q”
“--question”
不运行命令,也不输出。仅仅是检查所指定的目标是否须要更新。若是是0则说明要更新,若是是2则说明有错误发生。

“-r”
“--no-builtin-rules”
禁止make使用任何隐含规则。

“-R”
“--no-builtin-variabes”
禁止make使用任何做用于变量上的隐含规则。

“-s”
“--silent”
“--quiet”
在命令运行时不输出命令的输出。

“-S”
“--no-keep-going”
“--stop”
取消“-k”选项的做用。由于有些时候,make的选项是从环境变量“MAKEFLAGS”中继承下来的。因此你能够在命令行中使用这个参数来让环境变量中的“-k”选项失效。

“-t”
“--touch”
至关于UNIX的touch命令,只是把目标的修改日期变成最新的,也就是阻止生成目标的命令运行。

“-v”
“--version”
输出make程序的版本、版权等关于make的信息。

“-w”
“--print-directory”
输出运行makefile以前和以后的信息。这个参数对于跟踪嵌套式调用make时颇有用。

“--no-print-directory”
禁止“-w”选项。

“-W <file>;”
“--what-if=<file>;”
“--new-file=<file>;”
“--assume-file=<file>;”
假定目标<file>;须要更新,若是和“-n”选项使用,那么这个参数会输出该目标更新时的运行动做。若是没有“-n”那么就像运行UNIX的“touch”命令同样,使得<file>;的修改时间为当前时间。

“--warn-undefined-variables”
只要make发现有未定义的变量,那么就输出警告信息。

  4.4 Makefile规则检查

  有时候,咱们不想让咱们的makefile中的规则执行起来,咱们只想检查一下咱们的命令,或是执行的序列。因而咱们可使用make命令的下述参数:     “-n”     “--just-print”     “--dry-run”     “--recon”     不执行参数,这些参数只是打印命令,无论目标是否更新,把规则和连带规则下的命令打印出来,但不执行,这些参数对于咱们调试makefile颇有用处。     “-t”     “--touch”     这个参数的意思就是把目标文件的时间更新,但不更改目标文件。也就是说,make伪装编译目标,但不是真正的编译目标,只是把目标变成已编译过的状态。     “-q”     “--question”     这个参数的行为是找目标的意思,也就是说,若是目标存在,那么其什么也不会输出,固然也不会执行编译,若是目标不存在,其会打印出一条出错信息。     “-W <file>;”     “--what-if=<file>;”     “--assume-new=<file>;”     “--new-file=<file>;”     这个参数须要指定一个文件。通常是是源文件(或依赖文件),Make会根据规则推导来运行依赖于这个文件的命令,通常来讲,能够和“-n”参数一同使用,来查看这个依赖文件所发生的规则命令。

相关文章
相关标签/搜索