Linux:动态库加载、编址

2024-04-12 03:04
文章标签 动态 linux 加载 编址

本文主要是介绍Linux:动态库加载、编址,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

目录

一、库的概念

二、动静态库的加载

2.1绝对编址与相对编址

2.1一般程序的加载

三、动态库的加载


一、库的概念

库默认就是一个磁盘级文件,所以在执行代码时,库和可执行程序都会被加载到内存中,从原理上,库函数的调用依旧是在进程的地址空间中进行的。动态库/共享库只需要加载一份,然后通过页表映射到不同的进程地址空间中,所以就可以做到让多个进程使用同一份代码。

而库的加载由操作系统来决定,并且对已经加载的库进行管理,也就是先描述再组织。

和管理进程时类似,对加载进来的库的管理也可以转变为对链表的增删查改。

每个可执行程序都是有自己的格式信息的,即使可执行程序没有加载到内存中,我们的可执行程序中就有地址 ,可执行程序在没有被加载之前,也已经基本按照类别(比如权限,访问属性等)已经将可执行程序划分为各个区域了。

我们在创建一个进程,执行一个进程时,我们是如何对进程地址空间进行划分处理,去加载代码和数据的呢?操作系统是怎么知道该开辟多大空间怎么划分的呢?

我们进程地址空间里面的很多地址数据,是从要执行的可执行程序中来的。

二、动静态库的加载

2.1绝对编址与相对编址

可执行代码与数据如果从上到下从前到后进行连续的编址,这种方式称为绝对编址,也就是所谓的平坦模式。

如果我们将数据区与代码区分开,代码区从0开始,数据区从0开始,使用这种偏移量的方式,我们一般叫做相对地址也叫逻辑地址。

 虚拟地址空间不仅仅是操作系统要遵守,编译器编译程序时也需要遵守。

2.1一般程序的加载

即使没有加载程序本身也有地址,任何一个函数经过编制就没有变量名了,每个函数都是有入口的,所以程序形成可执行程序存到磁盘中加载到内存之前,本身就有编址了,每个程序还要ELF格式的表头,表头中也会记录类似于入口地址,整个可执行程序的区域划分,和各个区域的起始地址。

最终这个东西才是一个完整的可执行程序,而它的编址方式也采用绝对编址,从上到下挨着进行编,当然我们也可以将其看作其在磁盘中的逻辑地址也就是0+0X0000。

要将可执行程序加载到内存首先要创建一个进程,首先创建task_struct即PCB进程控制块,然后加载进程地址空间,而进程地址空间mm_strucr是从可执行程序的ELF表头来加载的。所以不同的程序就有不同的正文段,已初始化未初始化的范围大小。

而cpu中存在一个寄存器CR3保存页表的起始地址,而进程地址空间可以用可执行程序本身来进行初始化,所以可执行程序的大小也决定了进程地址空间的大小,然后将可执行程序代码加载到物理内存中,此时加载到物理内存的地址就是所谓的真实物理地址,而此时可执行程序内部的绝对编址就直接作为虚拟地址和物理地址进行映射建立页表。

而cpu寄存器中有个pc指针,也叫程序计数器,保存正在执行的下一条指令的地址,将ELF中存储的入口地址加载到pc指针,cpu中还有指令寄存器,经过MMU加页表将虚拟地址转换为物理地址,将指令读取然后去内存中查找执行。

所谓的地址空间本质是由操作系统+编译器+计算机体系结构(CPU)三者配合完成的。

所以逻辑地址和虚拟地址以及线性地址都是一个概念,物理地址则是在内存中的地址。

三、动态库的加载

 而可执行程序如何和库进行链接呢?

而库在磁盘中是按相对编制的方法来编址的,库有个起始地址,通过库的起始地址加上里面的各种函数指令所对应的偏移量来找到并执行该指令,而链接时只需要将调用的指令变成库的起始地点加上偏移量的方式存放在可执行程序中,并在前面包含头文件。

库加载到内存后,操作系统为了管理这个库,就会进行先描述再组织,会记录下来这个库被加载到了哪个位置。当cpu去执行和该库相关的指令时,系统只需要通过动态链接知道要执行哪个库,然后找到库的起始地址加上偏移量映射到共享区中库的虚拟地址,然后再通过页表找到库在内存中的位置,然后将指令加载到cpu去执行。

而库的地址在加载到进程地址空间时,一般是通过编址的地址加上偏移量形成一个地址然后得到进程地址空间中的绝对地址,这叫做地址加载时的动态重定向方式。

这篇关于Linux:动态库加载、编址的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Linux join命令的使用及说明

《Linuxjoin命令的使用及说明》`join`命令用于在Linux中按字段将两个文件进行连接,类似于SQL的JOIN,它需要两个文件按用于匹配的字段排序,并且第一个文件的换行符必须是LF,`jo... 目录一. 基本语法二. 数据准备三. 指定文件的连接key四.-a输出指定文件的所有行五.-o指定输出

Linux jq命令的使用解读

《Linuxjq命令的使用解读》jq是一个强大的命令行工具,用于处理JSON数据,它可以用来查看、过滤、修改、格式化JSON数据,通过使用各种选项和过滤器,可以实现复杂的JSON处理任务... 目录一. 简介二. 选项2.1.2.2-c2.3-r2.4-R三. 字段提取3.1 普通字段3.2 数组字段四.

Linux kill正在执行的后台任务 kill进程组使用详解

《Linuxkill正在执行的后台任务kill进程组使用详解》文章介绍了两个脚本的功能和区别,以及执行这些脚本时遇到的进程管理问题,通过查看进程树、使用`kill`命令和`lsof`命令,分析了子... 目录零. 用到的命令一. 待执行的脚本二. 执行含子进程的脚本,并kill2.1 进程查看2.2 遇到的

Linux云服务器手动配置DNS的方法步骤

《Linux云服务器手动配置DNS的方法步骤》在Linux云服务器上手动配置DNS(域名系统)是确保服务器能够正常解析域名的重要步骤,以下是详细的配置方法,包括系统文件的修改和常见问题的解决方案,需要... 目录1. 为什么需要手动配置 DNS?2. 手动配置 DNS 的方法方法 1:修改 /etc/res

MyBatis延迟加载与多级缓存全解析

《MyBatis延迟加载与多级缓存全解析》文章介绍MyBatis的延迟加载与多级缓存机制,延迟加载按需加载关联数据提升性能,一级缓存会话级默认开启,二级缓存工厂级支持跨会话共享,增删改操作会清空对应缓... 目录MyBATis延迟加载策略一对多示例一对多示例MyBatis框架的缓存一级缓存二级缓存MyBat

Linux创建服务使用systemctl管理详解

《Linux创建服务使用systemctl管理详解》文章指导在Linux中创建systemd服务,设置文件权限为所有者读写、其他只读,重新加载配置,启动服务并检查状态,确保服务正常运行,关键步骤包括权... 目录创建服务 /usr/lib/systemd/system/设置服务文件权限:所有者读写js,其他

Linux下利用select实现串口数据读取过程

《Linux下利用select实现串口数据读取过程》文章介绍Linux中使用select、poll或epoll实现串口数据读取,通过I/O多路复用机制在数据到达时触发读取,避免持续轮询,示例代码展示设... 目录示例代码(使用select实现)代码解释总结在 linux 系统里,我们可以借助 select、

Linux挂载linux/Windows共享目录实现方式

《Linux挂载linux/Windows共享目录实现方式》:本文主要介绍Linux挂载linux/Windows共享目录实现方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地... 目录文件共享协议linux环境作为服务端(NFS)在服务器端安装 NFS创建要共享的目录修改 NFS 配

Spring Gateway动态路由实现方案

《SpringGateway动态路由实现方案》本文主要介绍了SpringGateway动态路由实现方案,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随... 目录前沿何为路由RouteDefinitionRouteLocator工作流程动态路由实现尾巴前沿S

linux系统中java的cacerts的优先级详解

《linux系统中java的cacerts的优先级详解》文章讲解了Java信任库(cacerts)的优先级与管理方式,指出JDK自带的cacerts默认优先级更高,系统级cacerts需手动同步或显式... 目录Java 默认使用哪个?如何检查当前使用的信任库?简要了解Java的信任库总结了解 Java 信