GCC,Makfile学习笔记

0.编译过程

C/C++源码文件---------编译----------》目标文件.o gcc -c main.c -o a.o
多个目标文件(.o)------------连接-----------》可执行文件 gcc main.o -o main.out
编译过程检查语法,需要知道依赖的头文件的声明,一般一个头文件对应一个目标文件
链接过程主要链接库和其他目标文件
完整过程如下所示

GCC,Makfile学习笔记_第1张图片

1.GCC

makefile里面最核心的还是编译。
GCC,Makfile学习笔记_第2张图片

2.makefile基础语法

参考:
https://seisman.github.io/how-to-write-makefile/introduction.html
参考了一下这位大神写的makefile的教程,水平有限依然很多地方没看懂,也参考做了很多笔记,一些查找到的别的内容也补充上去了。

2.1.makefile基础

target:prerequisites      
	command

target可以是一个或者多个目标文件(通配符也支持),也可以是一个标签
prerequisites 是生成该target的依赖项
command 是target要执行的指令,前面需要空出一个【tab】,一般来说会以/bin/sh来执行当没有target或者prerequistites中的依赖项有一个新于target。command中的命令就会被执行。

2.2.变量

makefile的变量定义赋值和shell脚本类似。但使用的时候makefile中变量需要使用 ( v a r ) 或 者 (var)或者 (var){var}。嵌套make的变量传递可以export veriable 或者unexport variable。这个原理就是和export作用是一致的。

2.2.1.操作符

:= :这种方法前面的变量不能使用后面的变量,只能使用前面已定义好了的变量。
?= :这个含义是如果这个变量没有被定义过怎进行赋值,否则不进行赋值
+= : 追加变量
$(var:a=b) 把变量中所有以“a”后面待空格或者结束符的替换成“b”

2.2.2.自动化变量

$@ : 表示目标的 集合,就像一个数组, @ 依 次 取 出 目 标 , 并 执 于 命 令 ‘ @ 依次取出目标,并执于命令 ` @<: 第一个依赖文件名。 ‘ : 目 标 文 件 在 本 规 则 中 的 所 有 依 赖 文 件 ^` :目标文件在本规则中的所有依赖文件 ? : 所有依赖文件中比目标文件新的文件列表
$&:目标文件在所有规则中的依赖文件

2.2.3特殊变量

VPATH = 路径1:路径2:路径3 make会在VPATH路径列出的目录和当前目录汇总寻找依赖文件和目标文件。
SHELL = /bin/sh指定命令运行的SHELL
MAKEEFLAGS 包含了make的参数信息,这个变量中会存放你所指定的终极目标的列表,如果在命令行上,你没有指定目标,则这个变量是空值。
CC = 编译器(gcc/g++)
CFLAGS= 编译器的参数(-MM)

2.3.常用关键字

设置目录vpath

vpath  为符合模式的文件指定搜索目录
vpath      清除符合模式的文件的搜索目录
vpath      清除所有已被设置好了的文件搜索目录。

vapth使用方法中的需要包含 % 字符。 % 的意思是匹配零或若干字符,(需引用 % ,使用 \ )例如, %.h 表示所有以 .h 结尾的文件。指定了要搜索 的文件集,而则指定了< pattern>的文件集的搜索的目录

2.4.命令

命令时使用SHELL环境变量的值来执行命令的。
命令的注释 #
显示命令 @echo
如果make的参数带有-n或者–just-print只是显示命令但不执行命令;如果参数-s或者–silent或者–quiet则是全面禁止命令的显示。
命令出错
命令前加一个减号“-”,某些文件如果出现问题可以忽略。
命令包
如果Makefile中出现一些相同命令序列,那么我们可以为这些相同的命令序列定义一个变量。定义这种命令序列的语法以 define 开始,以 endef 结束。这个作用类似于函数,如:

define run-yacc
yacc $(firstword $^)
mv y.tab.c $@
endef

2.5.条件判断

语法:

ifeq 条件关键字     
	Command1
else    
	 Command2
endif

2.6.函数

2.6.1.函数的调用

$( ,,)或者
${ ,,} 

2.6.2.常用函数

字符串处理函数

 # 字符串替换函数
 $(subst ,,)  
 # 模式字符串替换函数 
  $(patsubst ,,)
  # 去掉空格函数
  $(strip )
  # 查找字符串
  $(findstring ,)
  # 过滤函数
  $(filter , )
  # 反过滤函数
  $(filter-out , )
  # 排序
  $(sort )
  # 取单词函数
  $(word ,)
  # 取单词串函数
  $(wordlist ,,)

文件名操作函数

#取目录
$(dir )
#取文件函数
$(notdir )
# 取后缀函数
$(suffix )
# 取前缀函数
$(basename )
# 加后缀函数
$(addsuffix ,)
# 加前缀函数
$(addprfix ,)
# 连接函数
$(join ,)

shell函数
利用shell命令的输出结果

ret = $(shell cmd)

call函数
call函数可以用来创建新的参数化的函数。

$(call ,,,...,)

foreach函数
foreach用来做循环,和shell中的含义类似$(foreach ,,)

2.7.伪目标

伪目标定义的是一个标签,但是伪目标不能和文件重名,为避免这个情况,可以用.PHONY显示指明一个伪目标。
其实伪目标就是为了让make执行伪目标的那段的命令

.PHONY :clean
clean :       
	-rm *.o $(bin)

常见的伪目标:
GCC,Makfile学习笔记_第3张图片

2.8.引用其他makefile

语法:

include filename1 filename2 filename3

Note:不能以Tab开头
引用的头文件会在以下几个目录找:

  1. make命令中如果有 -I或者–include-dir参数时,在指定目录中寻找。
  2. 存在目录/include存在(/usr/local/bin或者/usr/include)

3.make参数

参见参数列表,具体可参考man文档。
GCC,Makfile学习笔记_第4张图片

5.makefile的工作原理

一般在目录中执行make
1.make会再当前目录下找到名字为makefile或者Makefile的文件
2.关于目标,如果make后不跟目标,则默认为第一个目标
3.确定目标后根据时间戳来判断是否重新生成,并递归检查依赖项,最后执行完目标里的command

6.自动生成makefile

像现有的CMake,Automake, qmake等都可以轻松实现makefile的功能。

6.1.CMake

CMake is an open-source, cross-platform family of tools designed to build, test and package software. CMake is used to control the software compilation process using simple platform and compiler independent configuration files, and generate native makefiles and workspaces that can be used in the compiler environment of your choice. The suite of CMake tools were created by Kitware in response to the need for a powerful, cross-platform build environment for open-source projects such as ITK and VTK.

opencv好像也是用这个工具生成makefile的,后面要拿opencv源码进行分析

  1. CMake生成Makefile并编译的流程
  2. 编写CMake配置文件CMakeList.txt
  3. 执行cmake PATH 或者ccmake PATH生成Makefile
  4. 使用make命令进行编译

入门参考:https://www.hahack.com/codes/cmake/

6.2.Automake

参考:https://www.gnu.org/doc/doc.htmlhttp://www.laruence.com/2009/11/18/1154.htmlhttps://www.ibm.com/developerworks/cn/linux/l-makefile/index.html

你可能感兴趣的:(GCC,Makfile学习笔记)