编译等底层知识

目录

一. GCC命令语句大全

二. GCC编译4个阶段

三. makefile的使用

四. CMake

五. GNU工具链开发流程图 

六. Keil中的地址段

七. 静态库和动态库


一. GCC命令语句大全

-c只编译源文件,生成目标文件(.o 文件),不进行链接。
-o <file>指定输出文件的名称。
-g生成调试信息,用于调试程序
-Wall打开所有警告提示。
-I <dir>添加头文件搜索路径。
-L <dir>添加库文件搜索路径。
-l <library>链接指定的库文件。
-std=<standard>指定使用的语言标准,如 -std=c99
-0<level>优化级别,如 -O0(无优化)、-O1(基本优化)、-O2(更高级别优化)。
-D<macro>定义预处理宏。
-E只进行预处理,输出预处理结果。
-S只进行编译,生成汇编代码。
-shared生成共享库(动态链接库)。
-static生成静态可执行文件,使用静态链接。
-pthread链接 POSIX 线程库。
-lm链接数学库。
-fopenmp启用 OpenMP 并行编程支持。

二. GCC编译4个阶段

GCC不仅仅是一个编译器,它还包括预处理器、汇编器以及链接器,可以处理从代码编写到可执行程序生成的整个流程。

三. makefile的使用

先新建一个main.c

#include "stdio.h"int add(int a, int b);
int main()
{printf("num:%d\r\n",add(1,2));return 0;}

再新建一个math.c 

int add(int a, int b)
{return (a + b);
}

编译永远都是以单个源文件为单位的。这里我们先编译一下mian.c文件。编译生成的.o文件是一个二进制文件,文件格式是ELF(Linux下所有可执行文件的通用格式),Windows使用的是PE格式,(都是对二进制代码的封装)

我们可以在文件头部找到可执行文件的基本信息比如支持的操作系统,机器类型等

查看一系列的区块,.text是代码区.data是数据区(里面保存了我们初始化的全局变量,局部静态变量等等),

查看main.o这个目标文件中的内容,这里call指令是之前调用的printf和add函数,但是它们的跳转地址都被设为了0,而这里的0会在后面链接的时候被修正。

另外为了让编译器能够定位到这些需要被修正的地址,在代码块中我们还可以找到一个重定位表,比如在.text区块中,找到被重定位的两个函数printf和add。

接下来编译math.c且连同main.o一起链接生成一个独立的可执行文件

gcc main.o math.o -o main

链接其实就是将编译之后的所有目标文件,连同用到的一些静态库,运行时库,组合拼装成一个独立的可执行文件,其中就包括之前说到的地址修正,在这个时候,连接器会根据我们的目标文件或者静态库中的重定位表,找到那些需要被重定位的函数,全局变量,从而修正它们的地址

但如果我们每次都要手动编译和链接就不高效,所以我们就得使用makefile,而makefile的核心是对"依赖"的管理,所以makefile其实就是在定义一颗依赖树。

新建一个makefile文件,具体makefile可以查看二. MakeFile-CSDN博客

堆makefile文件进行一定优化

1. 使用makefile变量

2. “%”表示长度任意的非空字符串,比如“%.c”就是所有的以.c 结尾的 文件,类似与通配符,a.%.c 就表示以 a.开头,以.c 结束的所有文件。如 %.o : %.c

3. makefile自动化变量

4. Makefile 伪目标

自动化变量描述
$@规则中的目标集合,在模式规则中,如果有多个目标的话,“$@”表示匹配模 式中定义的目标集合。
$%当目标是函数库的时候表示规则中的目标成员名,如果目标不是函数库文件, 那么其值为空。
$<依赖文件集合中的第一个文件,如果依赖文件是以模式(即“%”)定义的,那么 “$<”就是符合模式的一系列的文件集合。
$? 所有比目标新的依赖目标集合,以空格分开。
$^所有依赖文件的集合,使用空格分开,如果在依赖文件中有多个重复的文件, “$^”会去除重复的依赖文件,值保留一份。
$+和“$^”类似,但是当依赖文件存在重复的话不会去除重复的依赖文件。
$*这个变量表示目标模式中"%"及其之前的部分,如果目标是 test/a.test.c,目标模 式为 a.%.c,那么“$*”就是 test/a.test。
all:main#我们最终需要的mian文件
objects =  main.o math.o
main:$(objects) #main文件是由main.o math.o而来gcc -o main $(objects)%.o : %.cgcc -c $<#main.o:main.c #main.o由main.c而来
#	gcc -c main.c
#math.o:math.c #math.math.c而来
#	gcc -c math.c.PHONY : cleanclean:rm *.orm main

接下来我们直接make就能生成最后的可执行文件.

四. CMake

CMake主要功能

1.配置和生成各大平台的工程

2.生成makefile文件

因为makefile的配置与当前系统有关,如果要开发一个跨平台的软件所以我们使用cmake。

CMakeList.txt文件

cmake_minimum required(VERSION 3.10)
project(hello)
add_executable(hello main.cpp factorial.cpp printhello.cpp)

cmake . 之后,也能生成makefile文件这时候我们再make,就能生成可执行文件了。

五. GNU工具链开发流程图 

GNU工具链包括了一系列重要的组件,如GCC(GNU编译器集合)、GNU Binutils、GNU Make和GDB等。

编译:编译器是armcc和armasm ,每个c/c++和汇编源文件编译成对应的以“.o”为后缀名的对象文件。

链接:链接器armlink把各个.o文件及库文件链接成一个映像文件“.axf”或“.elf”

格式转换:一般来说Windows或Linux系统使用链接器直接生成可执行映像文件elf后,内核根据该文件的信息加载后,就可以运行程序了,但在单片机平台上,需要把该文件的内容加载到芯片上,所以还需要对链接器生成的elf映像文件利用格式转换器fromelf转换成“.bin”或“.hex”文件,交给下载器下载到芯片的FLASH或ROM中。

六. Keil中的地址段

程序组件所属类别
机器代码指令Code
常量RO-data(Read Only data)
初值非0的全局变量RW-data(Read Write data)
初值为0的全局变量ZI-data(Zero Initialie data)
局部变量ZI-data的栈空间
使用malloc动态分配的空间ZI-data的堆空间

初值非0的全局变量和初值为0的全局变量其实就是数据区,而值为0的全局变量区又称为bss段。

七. 静态库和动态库

简单的来说二者的区别:

  • 静态库:就是在编译的时候直接将需要的代码连接进可执行程序中去;

  • 动态库:就是在需要调用其中的函数时,根据函数映射表找到该函数然后调入堆栈执行。当动态库被加载到内存之后,一旦它的内存地址被确定,我们就可以去修正动态库中的那些函数跳转地址,也就是重定位,这些地址在程序加载之前不过只是一堆占位符而已,如果我们直接去修改代码段中的跳转地址,由于代码段的内容被修改,自然就不能被其他进程所共享了,因为我们需要在内存中保存多个不同的副本,这刚好和节约内存的目标就背道而驰了,为了解决这个问题,动态链接采用了一种聪明的做法,不再修改代码段,而是在数据段中专门预留一片区域用来存放函数的跳转地址,它也被叫做全局偏移表

查看全局偏移表readelf -S ./libmath.so

got里面专门用来存放全局变量和函数的跳转地址,于是我们在调用函数的时候,会首先查表,然后根据表中的地址来进行跳转,这些地址会在动态库加载的时候,被修改成为真正的地址,而查表的过程也很容易实现,由于全局偏移表与代码段的相对位置是固定的,我们完全可以利用CPU的相对寻址来实现,有了全局偏移表,我们不再需要修改代码段,因此代码可以被所有进程共享,而全局偏移表虽然在每一个进程中保留一份副本,但由于占用空间很小,所以完全没有问题,采用这种方式实现的动态链接也被叫做 PIC(地址无关代码),换句话说我们的动态库不需要做任何修改,被加载到任意内存地址都能够正常运行,并且能够被所有进程共享,这也是为什么之前我们给编译器指定 -fPIC 参数的原因,另一方面由于动态链接在程序加载的时候需要对大量函数进行重定位,这一步是非常耗时的,为了进一步降低开销,我们的操作系统还做了一些其他的优化,比如延迟绑定或者也叫PLT,与其在一开始就对所有函数进行重定位,不如将这个过程推迟到函数第一次被调用的时候,因为绝大多数动态库中的函数可能在程序运行期间一次都不会被使用到。大概思路是,GOT中的跳转地址默认会指向一段辅助代码,它也被叫做桩代码(Stub),在我们第一次调用函数的时候,这段代码会负责查询真正函数的跳转地址,并且去更新GOT表,于是我们再次调用函数的时候,就会直接跳转到动态库中真正的函数实现。动态链接实际上将链接的整个过程,比如符号查询、地址的重定位从编译时推迟到了程序的运行时,更关键的是,它实现了二进制级别的代码复用。

从上面的描述可以知道,静态库是我们MCU开发者常用的一种,而动态库常用于Linux、Windows等开发场合

接下来我们创建一个动态链接库。还是先创建一个main.c和math.c.

 gcc -shared  -fPIC math.c -o libmath.so//-shared 表明这是一个共享对象(共享对象),libmath.so是linux下动态库的扩展名,windows下的动态库就是我们熟悉的各种.dll文件,
 gcc main.c -lmath -L. -o main我们在编译主程序的时候,我们需要指定一个-l参数,它告诉编译器与之前建的libmath.so进行动态链接,这里在指定动态库的时候,我们省略了前缀lib和扩展名.so,最后我们通过-L来指定动态库所在的路径,
在运行main时,会提示找不到libmath这个动态库,这是因为linux默认只会去系统路径下搜索动态库
./main: error while loading shared libraries: libmath.so: cannot open shared object file: No such file or directory
解决办法有
第一种,将动态库拷贝到系统路径下,但之后还得删除这个文件所以不方便
另外一种是使用环境变量,将当前目录添加到 LD_LIBRARY_PATH 环境变量中,这样操作系统会先到我们指定的路径下搜索,找不到的话再去搜索系统路径。

下图可以看到它成功调用了动态库中的add函数并返回了正确的结果,那之前我们提到动态链接的一大优势是允许我们单独更新动态库本身。 

这样我们更新一下动态库,对主程序不需要做任何修改,直接运行我们就可以看到刚才我们对动态库的更新 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/web/23537.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

CC++内存管理【new和delete操作符的详细分析】【常见面试题】

C/C内存管理 1.C/C内存分布 我们先来看一段代码&#xff0c;来了解一下C/C中的数据内存分布。 # include <stdlib.h>int globalVar 1; static int staticGlobalVar 1; // 比globalVar还要先销毁,同一个文件下后定义的先析构 // 全局变量存在 数据段&#xff08;静态…

VSCode+Vite+Vue3断点调试

目录 lunch.json创建 vite.config.ts 打断点运行 lunch.json创建 首先&#xff0c;点击VSCode左上角&#xff0c;甲壳虫运行的按钮&#xff0c;然后点击运行与调试&#xff0c;选择chrome浏览器&#xff0c;修改成一下配置。 { // 使用 IntelliSense 了解相关属性。 // 悬停…

codeforces round 949 div2

A Turtle and Piggy Are Playing a Game 题目&#xff1a; 思路&#xff1a;输出2的幂次b使得2^b为最大的不超过x的数 代码&#xff1a; #include <iostream>using namespace std;const int N 2e5 10;void solve() {int l, r;cin >> l >> r;if(r % 2) …

vscode 运行和调试

vscode使用断点 1.安装并激活扩展 Debugger for Chrome (弃用 --> JavaScript Debugger)Debugger for Firefox 2. 配置config文件 打开 config/index.js 并找到 devtool property。将其更新为&#xff1a; 如果你使用的是 Vue CLI 2&#xff0c;请设置并更新 config/in…

SpringBoot Redis读写与数据序列化 RedisTemplate 与 StringRedisTemplate 防转字节

介绍 RedisTemplate 对象在底层默认会转成字节&#xff0c;造成了内存的开销很大&#xff0c;这是他底层进行处理的,造成可读性差&#xff0c;如需要转成简单的字符串存储需要进行序列化的配置。 RedisTemplate 配置类 Configuration public class RedisConfig {Beanpublic …

OpenGL系列(五)纹理贴图

概述 OpenGL纹理是一种在三维图形中应用纹理映射的技术。纹理是一张图像&#xff0c;可以应用到三维模型的表面上&#xff0c;从而使得模型看起来更加真实和具有细节。通过纹理映射&#xff0c;可以将图像的像素值与三维模型的顶点进行匹配&#xff0c;从而为模型的表面增加细节…

Java并发编程之由于静态变量错误使用可能导致的并发问题

Java并发编程之由于静态变量错误使用可能导致的并发问题 1.1 前言1.2 业务背景1.3 问题分析1.4 为什么呢&#xff1f;1.5 修复方案2 演示示例源码下载 1.1 前言 我们知道在 Java 后端服务开发中&#xff0c;如果出现并发问题一般都是由于在多个线程中使用了共享的变量导致的。…

JVM相关:Java内存区域

Java 虚拟机&#xff08;JVM)在执行 Java 程序的过程中会把它管理的内存划分成若干个不同的数据区域。 Java运行时数据区域是指Java虚拟机&#xff08;JVM&#xff09;在执行Java程序时&#xff0c;为了管理内存而划分的几个不同作用域。这些区域各自承担特定的任务&#xff0c…

Day23 自定义对话框服务

​本章节实现了,自定义对话框服务的功能 当现有的对话框服务无法满足特定需求时,我们可以采用自定义对话框的解决方案,以更好地满足一些特殊需求。 一.自定义对话框主机服务步骤 在Models 文件夹中,再建立一个 IDialogHostService 接口类,继承自 IDialogService 对话框服…

绝对实用Linux命令行下的文件夹逐层创建术,从小白到大神的必学技能

哈喽&#xff0c;大家好&#xff0c;我是木头左&#xff01; 基础篇&#xff1a;初识Linux文件系统 在深入了解如何在Linux中逐层创建文件夹之前&#xff0c;需要对Linux的文件系统有一个基本的认识。Linux文件系统以其树状结构而著称&#xff0c;其中/&#xff08;根目录&…

SIMBA方法解读

目录 预处理scRNA-seqscATAC-seq 图构建&#xff08;5种场景&#xff09;scRNA-seq分析scATAC-seq分析多模态分析批次整合多模态整合 图学习SIMBA空间中查询实体识别TF-target genes 预处理 scRNA-seq 过滤掉在少于三个细胞中表达的基因。原始计数按文库大小标准化&#xff0…

DDS自动化测试落地方案 | 怿星科技携最新技术亮相是德科技年度盛会

5月28日&#xff0c;怿星科技作为是德科技的重要合作伙伴亮相Keysight World Tech Day 2024。在此次科技盛会上&#xff0c;怿星科技不仅展示了领先的DDS自动化测试解决方案等前沿技术&#xff0c;还分享了在“周期短、任务重”的情况下&#xff0c;如何做好软件开发和测试验证…

前端开发之性能优化

本文章 对各大学习技术论坛知识点&#xff0c;进行总结、归纳自用学习&#xff0c;共勉&#x1f64f; 文章目录 1. [CDN](https://www.bootcdn.cn/)2.懒加载3.缓存4.图片压缩5.图片分割6.sprite7.Code Splitting8.gzip9.GPU加速10.Ajax11.Tree Shaking12.Resource Hints 1. CD…

YOLO系列模型 pt文件转化为ONNX导出

文章目录 啥是onnx怎么导出导出之后 啥是onnx Microsoft 和合作伙伴社区创建了 ONNX 作为表示机器学习模型的开放标准。许多框架&#xff08;包括 TensorFlow、PyTorch、scikit-learn、Keras、Chainer、MXNet 和 MATLAB&#xff09;的模型都可以导出或转换为标准 ONNX 格式。 在…

C++笔试强训day40

目录 1.游游的字母串 2.体育课测验(二) 3.合唱队形 1.游游的字母串 链接https://ac.nowcoder.com/acm/problem/255195 英文字母一共就26个&#xff0c;因此可以直接暴力枚举以每个字母作为最后的转变字母。最后去最小值即可 #include <iostream> #include <cmath&…

项目实战系列——WebSocket——websock简介

最近项目中需要用到mes和本地客户端进行实时通讯&#xff0c;本来想用webapi进行交互的&#xff0c;但是考虑到高效和实时性&#xff0c;就采用这一项技术。 以往采用的方式——长轮询 客户端主动向服务器发送一个请求&#xff0c;如果服务器没有更新的数据&#xff0c;客户端…

五.应用层协议——HTTP协议

HTTP协议 在上一节中&#xff0c;我们提到了协议的本质&#xff0c;其实是双方约定好的某种格式的数据&#xff0c;常见的就是用结构体或者类来进行表达 而上层的业务逻辑决定了我们协议的定制&#xff0c;有了协议&#xff0c;双方就可以按照同样的角度&#xff0c;去解读数据…

一篇文章带你入门XXE

1.什么是XXE&#xff1f; XML External Entity&#xff08;XXE&#xff09;攻击是一种利用 XML 处理器的漏洞&#xff0c;通过引入恶意的外部实体来攻击应用程序的安全性。这种攻击通常发生在对用户提供的 XML 数据进行解析时&#xff0c;攻击者利用了 XML 规范允许引用外部实体…

kafka-集群搭建(在docker中搭建)

文章目录 1、kafka集群搭建1.1、下载镜像文件1.2、创建zookeeper容器并运行1.3、创建3个kafka容器并运行1.3.1、9095端口1.3.2、9096端口1.3.3、9097端口 1.4、重启kafka-eagle1.5、查看 efak1.5.1、查看 brokers1.5.2、查看 zookeeper 1、kafka集群搭建 1.1、下载镜像文件 d…

实时监控电脑屏幕软件有哪些?(珍藏篇)

在当今的数字化工作环境中&#xff0c;实时监控电脑屏幕软件是企业管理、远程协助、教育监控等领域不可或缺的工具。 这些软件能够帮助管理者了解员工的工作状态、提升团队协作效率、确保数据安全&#xff0c;同时在家庭教育和远程技术支持中也有广泛应用。 以下是精选的几款实…