编译等底层知识

news2024/11/24 7:02:27

目录

一. 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 : %.c
    gcc -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 : clean

clean:
    rm *.o
    rm 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.coloradmin.cn/o/1796300.html

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

相关文章

学习使用Opentelemetry python SDK

前言 &#x1f4e2;博客主页&#xff1a;程序源⠀-CSDN博客 &#x1f4e2;欢迎点赞&#x1f44d;收藏⭐留言&#x1f4dd;如有错误敬请指正&#xff01; 一、什么是 OpenTelemetry OpenTelemetry 由 OpenTracing 和 OpenCensus 项目合并而成&#xff0c;是一组规范、工具、API…

DxO ViewPoint v4.8 解锁版安装教程 (校正几何和透视的图像处理)

前言 DxO ViewPoint中文版是一款能够校正几何和透视的图像处理软件,摄影师通过ViewPoint破解版修复构图和光学缺陷并恢复拍摄对象平衡,重新调整如弯曲架构和扭曲图案等细节,让图像具备更强冲击力和更优平衡性。 一、下载地址 下载链接&#xff1a;http://dygod/source 点击搜…

VMware Workstation虚拟机固定IP配置(主机互通、外网可访问)

VMware Workstation虚拟机固定IP配置 环境问题配置过程配置虚拟机网络适配器配置虚拟机网络配置虚拟网卡网络适配器配置虚拟机固定IP 结果验证结束语参考 环境 主机&#xff1a;Windows 11 VMware Workstation: 17.5.2 虚拟机&#xff1a;Ubuntu 24.02 LTS 注&#xff1a; 主…

VRRP----虚拟路由器冗余协议(技术专题)

目录 一、VRRP的基本原理 1.1 技术背景 1.2 VRRP带来了什么 1.2.1 VRRP的作用 1.2.2 VRRP工作的过程 1.2.3 VRRP报文: 1.3 VRRP术语 1.3.1 虚拟IP地址、MAC地址 1.3.2 Master、Backup路由器 二、VRRP的基础配置 实例一 需求 配置 一、VRRP的基本原理 1.1 技术背景…

Spring Cloud工程添加子模块打包后文件为war包而非jar包

Spring Cloud工程添加子模块打包后文件为war包而非jar包 Spring Cloud子模块打出的包通常是JAR包而非WAR包&#xff0c;这是因为Spring Cloud主要基于Spring Boot构建&#xff0c;而Spring Boot默认打包为可执行JAR包。然而&#xff0c;如果遇到了Spring Cloud子模块打成了WAR…

计算机毕业设计Spark+Flink+Hive地铁客流量预测 交通大数据 地铁客流量大数据 交通可视化 大数据毕业设计 深度学习 机器学习

项目说明​ ​ 1该项目主要分析通刷卡数据&#xff0c;通过大数据技术来研究地铁客运能力及探索优化服务的方向​ 2主要讲解Flink流处理实时分析部分&#xff0c;离线部分较简单&#xff0c;暂时略过​ ​ 技术架构​ ​项目流程&#xff1a;​ 采用python请求深圳地铁数…

每天五分钟深度学习:逻辑回归算法的单样本的梯度下降计算

本文重点 上节课我们已经知道了如何利用计算图通过链式法则来求解输出J对变量的梯度或者导数。本节课程我们将通过逻辑回归这一个具体的例子,来演示如何使用计算图完成逻辑回归的梯度下降算法。 逻辑回归 逻辑回归算法的目标函数,损失函数,代价函数,以及参数更新的方式如…

Ffmpeg安装和简单使用

Ffmpeg安装 下载并解压 进入官网 (https://ffmpeg.org/download.html)&#xff0c;选择 Window 然后再打开的页面中下滑找到 release builds&#xff0c;点击 zip 文件下载 环境变量配置 下载好之后解压&#xff0c;找到 bin 文件夹&#xff0c;里面有3个 .exe 文件 然后复制…

度安讲 | 第二期「安全左移·业务护航」技术沙龙成功举办

当下&#xff0c;“安全左移”作为落地DevSecOps的重要实践之一&#xff0c;已在业界达成共识。DevSecOps作为一种集开发、安全、运维于一体的软件开发和运营模式&#xff0c;强调在敏捷交付下&#xff0c;“安全”在软件开发生命周期的全覆盖贯穿和核心位置。所谓“安全左移”…

vue 文件预览mp4、txt、pptx、xls、xlsx、docx、pdf、html、xml

vue 文件预览 图片、mp4、txt、pptx、xls、xlsx、docx、pdf、html、xml 最近公司要做一个类似电脑文件夹的功能&#xff0c;支持文件夹操作&#xff0c;文件操作,这里就不说文件夹操作了&#xff0c;说说文件预览操作&#xff0c;本人是后端java开发&#xff0c;前端vue&#…

gRPC实战 | 实现Python 和 Go 之间的 gRPC 交互

前言 &#x1f4e2;博客主页&#xff1a;程序源⠀-CSDN博客 &#x1f4e2;欢迎点赞&#x1f44d;收藏⭐留言&#x1f4dd;如有错误敬请指正&#xff01; 一、gRPC 简介 gRPC是一个高性能、通用的开源RPC框架&#xff0c;其由Google主要面向移动应用开发并基于HTTP/2协议标准而设…

eNSP学习——RIP路由协议的汇总

目录 主要命令 原理概述 实验目的 实验内容 实验拓扑 实验编址 实验步骤 1、基本配置 2、配置RIPv1协议 3、配置RIPv2自动汇总 4、配置RIPv2手动汇总 需要eNSP各种配置命令的点击链接自取&#xff1a;华为&#xff45;NSP各种设备配置命令大全PDF版_ensp配置命令大全…

Acwing 786.第K个数

Acwing 786.第K个数 题目描述 786. 第k个数 - AcWing题库 运行代码 #include <iostream> #include <algorithm> using namespace std; const int N 100010; int q[N];int main() {int n, k;scanf("%d%d", &n, &k);for (int i 0; i < n; …

网络通讯协议UDP转发TCP工具_UdpToTcpRelay

网络通讯协议UDP转发TCP工具_UdpToTcpRelay 本程序旨在提供一个灵活的、可配置的服务&#xff0c;它处理特定的UDP端口以接收命令&#xff0c;然后将这些命令转换为TCP命令并通过网络发送到指定的TCP服务器【TCP支持十六进制和ASCII】。 此设计特别适用于需要远程控制或自动化…

百度/迅雷/夸克,网盘免费加速,已破!

哈喽&#xff0c;各位小伙伴们好&#xff0c;我是给大家带来各类黑科技与前沿资讯的小武。 之前给大家安利了百度网盘及迅雷的加速方法&#xff0c;详细方法及获取参考之前文章&#xff1a; 刚刚&#xff01;度盘、某雷已破&#xff01;速度50M/s&#xff01; 本次主要介绍夸…

Day23 自定义对话框服务

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

[个人总结]-java常用方法

1.获取项目根路径 user.dir是一个系统属性&#xff0c;表示用户当前的工作目录&#xff0c;大多数情况下&#xff0c;用户的当前工作目录就是java项目的根目录&#xff08;src文件的同级路径&#xff09; System.getProperty("user.dir") 结果&#xff1a;D:\code…

什么是IDE?– 集成开发环境

IDE &#xff08;集成开发环境&#xff09;是将常用的开发人员工具组合到紧凑的 GUI&#xff08;图形用户界面&#xff09;应用程序中的软件。它是代码编辑器、代码编译器和代码调试器等工具与集成终端的组合。 为什么 IDE 很重要&#xff1f; 人们当然不需要 IDE来编码或开发…

【动手学深度学习】softmax回归从零开始实现的研究详情

目录 &#x1f30a;1. 研究目的 &#x1f30a;2. 研究准备 &#x1f30a;3. 研究内容 &#x1f30d;3.1 softmax回归的从零开始实现 &#x1f30d;3.2 基础练习 &#x1f30a;4. 研究体会 &#x1f30a;1. 研究目的 理解softmax回归的原理和基本实现方式&#xff1b;学习…

Python SQLAlchemy库详解

大家好&#xff0c;在Python生态系统中&#xff0c;SQLAlchemy库是一个强大的工具&#xff0c;为开发人员提供了便捷的方式来处理与数据库的交互。无论是开发一个小型的Web应用程序&#xff0c;还是构建一个大型的企业级系统&#xff0c;SQLAlchemy都能满足你的需求&#xff0c…