ZYNQ - 嵌入式Linux开发 -05- Linux C编程和Makefile(二)

简介: ZYNQ - 嵌入式Linux开发 -05- Linux C编程和Makefile

Makefile变量


跟 C 语言一样 Makefile 也支持变量,对于重复输入的内容,可以进行定义变量进行简化Makefile的编写。和C语言不太一样的是,Makefile 中的变量都是

字符串。所以简化上面的Makefile文件后为下所示:

obj = main.o test.o test1.o
main: $(obj)
  gcc -o main $(obj)
#main:main.o test.o test1.o
# gcc -o main main.o test.o test1.o
main.o:main.c
  gcc -c main.c
test.o:test.c
  gcc -c test.c
test1.o:test1.c
  gcc -c test1.c
clean:
  rm *.o
  rm main

赋值符 “=”

使用“=”在给变量的赋值的时候,不一定要用已经定义好的值,也可以使用后面定义的值。可以将变量的真实值推到后面去定义,也就是变量的真实值取决于它所引用的变量的最后一次有效值。

赋值符 “:=”

赋值符“:=”和“=”类似,但是不同的是,“:=”不会使用后面定义的变量,只能使用前面已经定义好的,这就是“=”和“:=”两个的区别。

赋值符 “?=”

如果变量前面没有被赋值,那么此变量就是赋值符右边的值,如果前面已经赋过值了,那么就使用前面赋的值。

变量追加 “+=”

Makefile 中的变量是字符串,有时候我们需要给前面已经定义好的变量添加一些字符串进去,此时就要使用到符号“+=”。

模式规则与自动化变量


上述 Makefile 中第 3~8 行是将对应的.c 源文件编译为.o 文件,每一个 C 文件都要写一个对应的规则,如果工程中 C 文件很多,这样操作会增加冗余工作量。所以可以使用 Makefile 中的模式规则,通过模式规则就可以使用一条规则来将所有的.c 文件编译为对应的.o 文件。

模式规则中,至少在规则的目标定定义中要包涵“%”,否则就是一般规则,目标中的“%”表示对文件名的匹配,“%”表示长度任意的非空字符串,比如“%.c”就是所有的以.c 结尾的文件,类似与通配符,a.%.c 就表示以 a.开头,以.c 结束的所有文件。

当“%”出现在目标中的时候,目标中“%”所代表的值决定了依赖中的“%”值,使用方法如下:

%.o : %.c
  命令

对于规则中把c文件编译为o文件已经简化,为了简化命令操作,同时引入自动化变量的内容。目标和依赖都是一系列的文件,每一次对模式规则进行解析的时候都会是不同的目标和依赖文件,而命令只有一行。自动化变量可以把命令进行通用化处理。

自动化变量 就是这种变量会把模式中所定义的一系列的文件自动的挨个取

出,直至所有的符合模式的文件都取完,自动化变量只应该出现在规则的命令中。

image.png

使用自动化变量来完成Makefile,最终代码如下所示:

obj = main.o test.o test1.o
main: $(obj)
  gcc -o main $(obj)
%.o : %.c
  gcc -c $<
clean:
  rm *.o
  rm main

Makefile 伪目标


Makefile 有一种特殊的目标——伪目标,一般的目标名都是要生成的文件,而伪目标不代表真正的目标名,在执行 make 命令的时候通过指定这个伪目标来执行其所在规则的定义的命令。

使用伪目标的主要是为了避免 Makefile 中定义的只执行命令的目标和工作目录下的实际文件出现名字冲突,有时候我们需要编写一个规则用来执行一些命令,但是这个规则不是用来创建文件的。比如在该文件下创建了一个clean文件,而在Makefile中又创建了一个clean的命令,此时执行make clean时,无法进行正常执行Makefile中的清除指令。为了解决这个“共存”问题,可以对clean进行声明为伪目标。

.PHONY : clean

Makefile 条件判断


Makefile 也支持条件判断,语法有两种如下:

<条件关键字>
  <条件为真时执行的语句>
endif

以及:

<条件关键字>
  <条件为真时执行的语句>
else
  <条件为假时执行的语句>
endif

其中条件关键字有 4 个: ifeq、 ifneq、 ifdef 和 ifndef,这四个关键字其实分为两对、 ifeq 与 ifneq、 ifdef与 ifndef,先来看一下 ifeq 和 ifneq, ifeq 用来判断是否相等, ifneq 就是判断是否不相等, ifeq 用法如下:

ifeq (<参数 1>, <参数 2>)
ifeq ‘<参数 1 >’,‘ <参数 2>’
ifeq “<参数 1>”, “<参数 2>”
ifeq “<参数 1>”, ‘<参数 2>’
ifeq ‘<参数 1>’, “<参数 2>”

上述用法中都是用来比较“参数 1”和“参数 2”是否相同,如果相同则为真,“参数 1”和“参数 2”可以为函数返回值,ifneq 的用法类似。

Makefile 函数


Makefile 支持函数,不支持我们自定义函数,只能使用定义好的函数。函数的用法如下:

$(函数名 参数集合)   or   ${函数名 参数集合}

调用函数和调用普通变量一样,使用符号“” 来 标 识 。 参 数 集 合 是 函 数 的 多 个 参 数 , 参 数 之 间 以 逗 号 “ , ” 隔 开 , 函 数 名 和 参 数 之 间 以 “ 空 格 ” 分 隔 开 , 函 数 的 调 用 以 “ ”来标识。参数集合是函数的多个参数,参数之间以逗号“,”隔开,函数名和参数之间以“空格”分隔开,函数的调用以“”来标识。参数集合是函数的多个参数,参数之间以逗号“,”隔开,函数名和参数之间以“空格”分隔开,函数的调用以“”开头。

函数 subst

函数 subst 用来完成字符串替换,调用形式如下:

$(subst <from>,<to>,<text>)

此函数的功能是将字符串中的内容替换为,函数返回被替换以后的字符串。

函数 patsubst

函数 patsubst 用来完成模式字符串替换,使用方法如下:

$(patsubst <pattern>,<replacement>,<text>)

此函数查找字符串中的单词是否符合模式,如果匹配就用来替换掉,可以使用包括通配符“ %”,表示任意长度的字符串,函数返回值就是替换后的字符串。如果中也包涵“%”,那么中的“%”将是中的那个“%”所代表的字符串。

比如:

$(patsubst %.c,%.o,a.c b.c c.c)

将字符串“a.c b.c c.c”中的所有符合“%.c”的字符串,替换为“%.o”,替换完成以后的字符串为“a.o

b.o c.o”。

函数 dir

函数 dir 用来获取目录,使用方法如下:

$(dir <names…>)

此函数用来从文件名序列中提取出目录部分,返回值是文件名序列的目录部分。

函数 notdir

函数 notdir 看名字就是知道去除文件中的目录部分,也就是提取文件名,用法如下:

$(notdir <names…>)

此函数用与从文件名序列中提取出文件名非目录部分。

函数 foreach

foreach 函数用来完成循环,用法如下:

$(foreach <var>, <list>,<text>)

此函数的意思就是把参数中的单词逐一取出来放到参数中,然后再执行所包含的表达

式。每次都会返回一个字符串,循环的过程中, 中所包含的每个字符串会以空格隔开,最后当整个循环结束时, 所返回的每个字符串所组成的整个字符串将会是函数 foreach 函数的返回值。

函数 wildcard

通配符“%”只能用在规则中,只有在规则中它才会展开,如果在变量定义和函数使用时,通配符不会自动展开,这个时候就要用到函数 wildcard,使用方法如下:

$(wildcard PATTERN…)

比如:

$(wildcard *.c)

上面的代码是用来获取当前目录下所有的.c 文件,类似“%”。

References


  1. 正点原子开发视频
  2. 正点原子ZYNQ之linux开发教程
  3. 跟我一起写 Makefile
目录
相关文章
|
20小时前
|
Linux Shell C语言
linux下configure,make(makefile),cmake命令详解-makefile和CMakeList
linux下configure,make(makefile),cmake命令详解-makefile和CMakeList
|
2天前
|
编解码 Linux 5G
FFmpeg开发笔记(二十)Linux环境给FFmpeg集成AVS3解码器
AVS3,中国制定的第三代音视频标准,是首个针对8K和5G的视频编码标准,相比AVS2和HEVC性能提升约30%。uavs3d是AVS3的解码器,支持8K/60P实时解码,且在各平台有优秀表现。要为FFmpeg集成AVS3解码器libuavs3d,需从GitHub下载最新源码,解压后配置、编译和安装。之后,重新配置FFmpeg,启用libuavs3d并编译安装,通过`ffmpeg -version`确认成功集成。
12 0
FFmpeg开发笔记(二十)Linux环境给FFmpeg集成AVS3解码器
|
3天前
|
运维 监控 关系型数据库
【Zabbix 6(1),Linux运维组件化开发教程
【Zabbix 6(1),Linux运维组件化开发教程
|
6天前
|
Linux C语言 调度
|
6天前
|
Linux API
Linux系统编程之文件编程常用API回顾和文件编程一般步骤
Linux系统编程之文件编程常用API回顾和文件编程一般步骤
Linux系统编程之文件编程常用API回顾和文件编程一般步骤
|
6天前
|
Linux C语言
|
6天前
|
存储 算法 网络协议
【探索Linux】P.26(网络编程套接字基本概念—— socket编程接口 | socket编程接口相关函数详细介绍 )
【探索Linux】P.26(网络编程套接字基本概念—— socket编程接口 | socket编程接口相关函数详细介绍 )
13 0
|
6天前
|
Shell Linux 编译器
C语言,Linux,静态库编写方法,makefile与shell脚本的关系。
总结:C语言在Linux上编写静态库时,通常会使用Makefile来管理编译和链接过程,以及Shell脚本来自动化构建任务。Makefile包含了编译规则和链接信息,而Shell脚本可以调用Makefile以及其他构建工具来构建项目。这种组合可以大大简化编译和构建过程,使代码更易于维护和分发。
30 5
|
6月前
|
消息中间件 缓存 Unix
[面试必备]嵌入式Linux内核开发必须了解的三十道题
[面试必备]嵌入式Linux内核开发必须了解的三十道题
|
8月前
|
Linux
嵌入式Linux QT开发之如何实现获取磁盘空间大小的应用逻辑
嵌入式Linux QT开发之如何实现获取磁盘空间大小的应用逻辑
156 0