[RTOS 学习记录] 工程管理工具make及makefile

2024-04-23 23:20

本文主要是介绍[RTOS 学习记录] 工程管理工具make及makefile,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

[RTOS 学习记录] 工程管理工具make及makefile

image-20240421150217202

这篇文章是我阅读《嵌入式实时操作系统μCOS-II原理及应用》后的读书笔记,记录目的是为了个人后续回顾复习使用。
前置内容:

开发工具 Borland C/C++ 3.1 精简版

文章目录

  • 1 make 工具
  • 2 makefile 的内容结构
  • 3 程序段标号的作用
    • 3.1 makefile 示例代码
    • 3.2 代码说明
    • 3.3 第一次运行
    • 3.4 第二次运行
    • 3.5 第三次运行
    • 3.6 结论
  • 4 makefile 实现编译、链接工作
    • 4.1 示例代码
    • 4.2 makefile 代码说明
    • 4.3 makefile 的执行
  • 5 程序段标号的目标作用
    • 5.1 makefile 示例代码
    • 5.2 伪目标
      • 5.2.1 第一次运行
  • 6 makefile 文件的命名
  • 7 makefile 中的变量

1 make 工具

一个开发平台提供给我们,用于管理工程或项目的实用程序,它可以按照我们用户编写的makefile脚本文件对工程项目进行管理。

2 makefile 的内容结构

makefile是一个脚本文件,文件内容中有许多我们在命令行中常常用到的各种命令。

makefile程序段的格式如下:

程序段的标号(target): 关联程序段1的标号 关联程序段2的标号 ...命令集
关联程序段1的标号:命令集
关联程序段2的标号:命令集
...

注意:命令集中的所有命令行必须缩进一个tab键。

一个makefile文件有若干个程序段,程序段的开头必须有一个target进行标注,区分各个程序段。不同的程序段之间可以进行关联,此时在target后面以空格为界罗列相关联程序段的target。每个程序段有一组实现工程项目管理的命令集。

3 程序段标号的作用

标号可以看作是对应程序段的名称,我们可以在make命令的后面使用标号来指定需要执行的程序段。

3.1 makefile 示例代码

按照makefile的内容格式编写一个makefile脚本文件,命名为makefile(makefile的默认名称):

mkdir1:md dir1
mkdir2:md dir2
rmdir:rd dir1rd dir2

3.2 代码说明

上面编写的makefile文件中一共有3个程序段:mkdir1、mkdir2 和 rmdir。作用分别是:

  1. mkdir1——在当前目录下创建一个名为dir1的文件夹
  2. mkdir2——在当前目录下创建一个名为dir2的文件夹
  3. rmdir——删除前面两个步骤创建的两个文件夹dir1和dir2

3.3 第一次运行

在命令行窗口中使用 cd EXP2_3 进入此次示例makefile文件所在的目录中,输入 make 命令并且回车执行。可以看到,执行完成后在当前目录新建了一个名为dir1的文件夹,如下图所示:

image-20240421154600345

根据执行结果,我们知道了make执行了makefile中的第一个程序段mkdir1,其余两个程序段mkdir2rmdir都没有被执行。

3.4 第二次运行

输入命令 make mkdir2 并且回车执行,可以看到,执行完毕后当前目录下又新建了一个名为dir2的文件夹,如下图所示:

image-20240421155052461

3.5 第三次运行

输入命令 make rmdir 并且回车执行,可以看到,执行后dir1和dir2这两个文件夹都被删除了,如下图所示:

image-20240421155302943

3.6 结论

当使用 make 命令时,makefile的第一个程序段会被执行,即makefile的首段程序段是make.exe的默认执行程序段,makefile的其他程序段需要执行时必须在make命令后面显式地指定标号。

4 makefile 实现编译、链接工作

由于makefile的程序段中的命令集中可以使用一切命令行命令,所以我们可以把源文件的编译和链接工作步骤编写到makefile中,然后通过执行makefile脚本文件“自动的”完成编译、链接工作。

4.1 示例代码

一个具有3个源文件应用程序的示例如下:

头文件 printA.h

#ifndef _PRINTA_H_
#define _PRINTA_H_extern const char *msgA;#endif

源文件 printA.c

#include "printA.h"const char *msgA = "AAAAA";

头文件 printB.h

#ifndef _PRINTB_H_
#define _PRINTB_H_extern const char *msgB;#endif

源文件 printB.c

#include "printB.h"const char *msgB = "BBBBB";

源文件 test.c

#include <stdio.h>
#include "printA.h"
#include "printB.h"int main(void)
{unsigned char i = 0;for (i=0; i<5; i++){printf("%s\n", msgA);printf("   %s\n", msgB);}return 0;
}

链接文件 TESTLINK

C:\BC\LIB\C0L.OBJ +
PRINTA.OBJ +
PRINTB.OBJ +
TEST.OBJ
TEST
TEST
C:\BC\LIB\CL.LIB

接下来就编写一个具有4个程序段的makefile脚本文件,实现源文件的编译、中间目标文件的链接,最终生成可执行文件TEST.EXE。

make脚本文件 makefile

##############################################
#             创建可执行文件(exe)
TEST.EXE:TLINK   @TESTLINK##############################################
#           创建各个目标文件(obj)
PRINTA.OBJ:BCC -c -ml -IC:\BC\INCLUDE -LC:\BC\LIB PRINTA.C
PRINTB.OBJ:BCC -c -ml -IC:\BC\INCLUDE -LC:\BC\LIB PRINTB.C
TEST.OBJ:BCC -c -ml -IC:\BC\INCLUDE -LC:\BC\LIB TEST.C

4.2 makefile 代码说明

为了增强可读性,此次示例的makefile代码中使用了文件名称作为程序段的标号,而且该文件正是对应程序段的命令集中的命令所要实现的目标结果。

第2~4个程序段分别完成对3个源文件printA.c、printB.c和test.c的编译,然后得到3个中间目标文件printA.obj、printB.obj和test.obj。第1个程序段即首段程序段完成各个中间目标文件的链接,最终得到可执行文件TEST.EXE。

4.3 makefile 的执行

分别依次使用 make PRINTA.OBJmake PRINTB.OBJmake TEST.OBJmake 完成示例应用程序的编译、链接,如下图所示:

image-20240421165535513

image-20240421165621705

image-20240421165657126

image-20240421165732331

运行可执行程序 TEST.EXE,可以看到屏幕上重复5次打印了字符串,如下图所示:

image-20240421165907435

5 程序段标号的目标作用

前面的示例makefile文件中,我们使用了文件名:PRINTA.OBJPRINTB.OBJTEST.OBJ 作为它们各自程序段的标号,而该文件名对应的文件正是它的程序段命令集所要生成的文件,因此我们把满足这种关系的程序段标号又称作程序段的目标,例如:PRINTA.OBJ 是它对应程序段的目标,PRINTB.OBJ 是它对应程序段的目标,TEST.OBJ 是它对应程序段的目标等等。

我们前面已经提到了[makefile允许关联程序段](#makefile 的内容结构),即makefile允许我们把程序段编写成如下形式:

目标(标号): 生成目标所需的文件名(依赖文件,简称“依赖”)命令集

为了强调程序段目标与其所需文件之间的关系,我们把生成目标所需的文件称作依赖文件,简称依赖

因此,我们可以把一个工程的编译、链接工作所需的多个程序段关联起来,从而仅需要执行一次 make 命令即可完成所有的编译和链接工作。

对于上述示例的makefile来说,如果要把生成TEST.EXE文件的程序段和生成其依赖文件的程序段关联起来,那么按照上述格式,makefile的第一个程序段就为:

TEST.EXE: PRINTA.OBJ PRINTB.OBJ TEST.OBJTLINK @TESTLINK

该程序段的含义是:本程序段的目标(标号)为 TEST.EXE,该目标(标号)需要由 PRINTA.OBJPRINTB.OBJTEST.OBJ 三个文件来生成,其命令则为 TLINK @TESTLINK

如果目标所依赖的文件都存在,满足生成目标所需要的条件,则连接命令 TLINK 被执行,否则程序会以 PRINTA.OBJPRINTB.OBJTEST.OBJ 为转移目标转向以它们为标号的程序段。也就是说,目标:依赖文件名 的这种格式是一种多分支条件转移语句。当生成目标的条件不满足(依赖文件不存在)时,程序的执行将要发生转移,其转移目标就是以依赖文件名为标号或目标的程序段。

实际上,make工具在执行makefile的各个程序段时,首先会检查目标(target)文件是否已经存在,如果存在,则会进一步检查该目标所依赖文件的时间戳(文件属性中的“创建时间”、“修改时间”等时间信息),只有当依赖文件比现有目标新时,其命令集才会被执行。其目的就是:尽量不做不必要的重复编译工作。

5.1 makefile 示例代码

为了格式上的整齐,凡是以目标为标号的程序段都要写上目标的依赖。

make 文件 makefile

##############################################
#             创建可执行文件(exe)
TEST.EXE: PRINTA.OBJ PRINTB.OBJ TEST.OBJTLINK   @TESTLINK
##############################################
#           创建各个目标文件(obj)
PRINTA.OBJ: PRINTA.C PRINTA.HBCC -c -ml -IC:\BC\INCLUDE -LC:\BC\LIB PRINTA.C
PRINTB.OBJ: PRINTB.C PRINTB.HBCC -c -ml -IC:\BC\INCLUDE -LC:\BC\LIB PRINTB.C
TEST.OBJ: TEST.C PRINTA.H PRINTB.HBCC -c -ml -IC:\BC\INCLUDE -LC:\BC\LIB TEST.C

5.2 伪目标

由上可知,makefile的target有目标和标号两种作用:当它是文件名时,它既是标号也是目标;而当它只是一个标识时,它就是标号。听起来很混乱,所以为了明确起见,人们把makefile中的target全部叫做目标,把那种仅起标号作用的目标则叫做“伪目标”。
在makefile中,伪目标所对应的程序段是一个不与其他程序段相关联的程序段,所以在需要执行它们时,必须在make命令中显式地使用其标号,除非它是makefile的第一个程序段(几乎没人这样做)。它们通常被用来完成一些创建目录、删除目录、复制文件、移动文件及删除文件等项目管理任务。

例如,可以为示例makefile添加一个标号为 CLEAN 的伪目标代码段,该段的任务就是为了用户目录的整洁,在已生成了最终可执行文件后,删除那些中间目标文件 PRINTA.OBJPRINTB.OBJTEST.OBJ
修改后的makefile如下:

##############################################
#             创建可执行文件(exe)
TEST.EXE: PRINTA.OBJ PRINTB.OBJ TEST.OBJTLINK   @TESTLINK
##############################################
#           创建各个目标文件(obj)
PRINTA.OBJ: PRINTA.C PRINTA.HBCC -c -ml -IC:\BC\INCLUDE -LC:\BC\LIB PRINTA.C
PRINTB.OBJ: PRINTB.C PRINTB.HBCC -c -ml -IC:\BC\INCLUDE -LC:\BC\LIB PRINTB.C
TEST.OBJ: TEST.C PRINTA.H PRINTB.HBCC -c -ml -IC:\BC\INCLUDE -LC:\BC\LIB TEST.C
CLEAN:DEL PRINTA.OBJDEL PRINTB.OBJDEL TEST.OBJ

5.2.1 第一次运行

使用 make 命令执行 makefile 的第一个程序段,生成最终可执行文件 TEST.EXE,如下图所示:

image-20240422091935127

image-20240422092035300

执行 make clean 表示显示地使用 CLEAN 参数执行 makefile 文件中的 CLEAN 程序段,将中间目标文件删除,如下图所示:

image-20240422092319854

image-20240422092334231

6 makefile 文件的命名

makefile是make文件的默认名称,如果用户不喜欢该名称,则完全可以自行对其进行命名(包括扩展名),但在make命令中要使用参数f,即:

make -f 文件名

7 makefile 中的变量

通常,在一个makefile中会有很多经常要重复使用的元素,例如示例makefile中的编译命令 BCC、编译命令中的参数 -c -ml -IC:\BC\INCLUDE -LC:\BC\LIB 等等。显然,用一些比较简洁且语义清楚的符号变量来表示它们更好,因此makefile允许人们定义变量。

变量格式:

变量名 = 变量的值

引用变量格式:

$(变量名)

使用变量改写示例makefile后,代码如下:

其中,前面带有符号“#”的为注释行;如果依赖文件表示行过长,也可以反斜杠“\”为换行符分行书写。

##############################################
#             makefile
##############################################
#        用变量来表示所使用的开发工具
BORLAND = C:\BC
CC = $(BORLAND)\BIN\BCC
LINK = $(BORLAND)\BIN\TLINK
##############################################
#               编译选项说明
#
# -l    生成80286实模式代码
# -c    编译为.obj文件
# -I    指示包含头文件所在路径
# -k    采用标准栈帧
# -L    指示库文件所在路径
# -ml   Large memory内存模式
# -n    指示生成目标文件的位置
##############################################
#             C编译选项变量
C_FLAGS = -c -ml -l -n.\ -k -I$(BORLAND)\INCLUDE -L$(BORLAND)\LIB
##############################################
#             链接选项变量
LINK_FLAGS = 
##############################################
#             创建可执行文件(exe)
TEST.EXE:       \PRINTA.OBJ      \PRINTB.OBJ      \TEST.OBJ$(LINK) $(LINK_FLAGS) @TESTLINK
##############################################
#           创建各个目标文件(obj)
PRINTA.OBJ:         \PRINTA.c        \PRINTA.h$(CC) $(C_FLAGS) PRINTA.c
PRINTB.OBJ:         \PRINTB.C        \PRINTB.H$(CC) $(C_FLAGS) PRINTB.C
TEST.OBJ:       \TEST.C      \PRINTA.H        \PRINTB.H$(CC) $(C_FLAGS) TEST.C
# 以下为伪目标代码段
CLEAN:DEL PRINTA.OBJDEL PRINTB.OBJDEL TEST.OBJ

运行结果如下图:

make

image-20240422094447941

image-20240422094550617

make clean

image-20240422094621696

image-20240422094647703

可以看到,使用变量后,make执行时会自动将makefile中引用的变量替换成变量的值。

参考资料:

《嵌入式实时操作系统μCOS-II原理及应用》

这篇关于[RTOS 学习记录] 工程管理工具make及makefile的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/930168

相关文章

51单片机学习记录———定时器

文章目录 前言一、定时器介绍二、STC89C52定时器资源三、定时器框图四、定时器模式五、定时器相关寄存器六、定时器练习 前言 一个学习嵌入式的小白~ 有问题评论区或私信指出~ 提示:以下是本篇文章正文内容,下面案例可供参考 一、定时器介绍 定时器介绍:51单片机的定时器属于单片机的内部资源,其电路的连接和运转均在单片机内部完成。 定时器作用: 1.用于计数系统,可

问题:第一次世界大战的起止时间是 #其他#学习方法#微信

问题:第一次世界大战的起止时间是 A.1913 ~1918 年 B.1913 ~1918 年 C.1914 ~1918 年 D.1914 ~1919 年 参考答案如图所示

[word] word设置上标快捷键 #学习方法#其他#媒体

word设置上标快捷键 办公中,少不了使用word,这个是大家必备的软件,今天给大家分享word设置上标快捷键,希望在办公中能帮到您! 1、添加上标 在录入一些公式,或者是化学产品时,需要添加上标内容,按下快捷键Ctrl+shift++就能将需要的内容设置为上标符号。 word设置上标快捷键的方法就是以上内容了,需要的小伙伴都可以试一试呢!

AssetBundle学习笔记

AssetBundle是unity自定义的资源格式,通过调用引擎的资源打包接口对资源进行打包成.assetbundle格式的资源包。本文介绍了AssetBundle的生成,使用,加载,卸载以及Unity资源更新的一个基本步骤。 目录 1.定义: 2.AssetBundle的生成: 1)设置AssetBundle包的属性——通过编辑器界面 补充:分组策略 2)调用引擎接口API

C++工程编译链接错误汇总VisualStudio

目录 一些小的知识点 make工具 可以使用windows下的事件查看器崩溃的地方 dumpbin工具查看dll是32位还是64位的 _MSC_VER .cc 和.cpp 【VC++目录中的包含目录】 vs 【C/C++常规中的附加包含目录】——头文件所在目录如何怎么添加,添加了以后搜索头文件就会到这些个路径下搜索了 include<> 和 include"" WinMain 和

Javascript高级程序设计(第四版)--学习记录之变量、内存

原始值与引用值 原始值:简单的数据即基础数据类型,按值访问。 引用值:由多个值构成的对象即复杂数据类型,按引用访问。 动态属性 对于引用值而言,可以随时添加、修改和删除其属性和方法。 let person = new Object();person.name = 'Jason';person.age = 42;console.log(person.name,person.age);//'J

大学湖北中医药大学法医学试题及答案,分享几个实用搜题和学习工具 #微信#学习方法#职场发展

今天分享拥有拍照搜题、文字搜题、语音搜题、多重搜题等搜题模式,可以快速查找问题解析,加深对题目答案的理解。 1.快练题 这是一个网站 找题的网站海量题库,在线搜题,快速刷题~为您提供百万优质题库,直接搜索题库名称,支持多种刷题模式:顺序练习、语音听题、本地搜题、顺序阅读、模拟考试、组卷考试、赶快下载吧! 2.彩虹搜题 这是个老公众号了 支持手写输入,截图搜题,详细步骤,解题必备

vcpkg安装opencv中的特殊问题记录(无法找到opencv_corexd.dll)

我是按照网上的vcpkg安装opencv方法进行的(比如这篇:从0开始在visual studio上安装opencv(超详细,针对小白)),但是中间出现了一些别人没有遇到的问题,虽然原因没有找到,但是本人给出一些暂时的解决办法: 问题1: 我在安装库命令行使用的是 .\vcpkg.exe install opencv 我的电脑是x64,vcpkg在这条命令后默认下载的也是opencv2:x6

《offer来了》第二章学习笔记

1.集合 Java四种集合:List、Queue、Set和Map 1.1.List:可重复 有序的Collection ArrayList: 基于数组实现,增删慢,查询快,线程不安全 Vector: 基于数组实现,增删慢,查询快,线程安全 LinkedList: 基于双向链实现,增删快,查询慢,线程不安全 1.2.Queue:队列 ArrayBlockingQueue:

硬件基础知识——自学习梳理

计算机存储分为闪存和永久性存储。 硬盘(永久存储)主要分为机械磁盘和固态硬盘。 机械磁盘主要靠磁颗粒的正负极方向来存储0或1,且机械磁盘没有使用寿命。 固态硬盘就有使用寿命了,大概支持30w次的读写操作。 闪存使用的是电容进行存储,断电数据就没了。 器件之间传输bit数据在总线上是一个一个传输的,因为通过电压传输(电流不稳定),但是电压属于电势能,所以可以叠加互相干扰,这也就是硬盘,U盘