Linux下的进程地址空间

news2025/1/22 6:50:25

Linux下的进程地址空间

  • 程序地址空间回顾
  • 从代码结果推结论
  • 引入进程地址空间
    • 页表
  • 为什么要有进程地址空间
  • 重新理解进程地址空间

程序地址空间回顾

我们在初学C/C++的时候,我们会经常看见老师们画这样的内存布局图:
在这里插入图片描述
可是这真的是内存吗?
如果不是它内存,那它是什么呢?

从代码结果推结论

在回答上面的问题之前我们先看一段代码:
在这里插入图片描述
运行结果:
在这里插入图片描述
通过运行结果我们可以看出,父进程的g_val一直都是保持为10,但是子进程的g_val却是一直在变化!这还不是最恐怖的,最恐怖的是父子进程的g_val是一样的地址,那么说明父子进程的g_val是同一块空间啊,那么是同一块空间的话,子进程去修改了g_val,父进程再去读取g_val应该是子进程修改过后的,可是父进程似乎还是读取的以前的值(10);
首先一块空间是不可能保存两份值的!父子进程能从统一个变量里读取两份值出来,那么说明父子进程的g_val绝对不是表示的同一块空间,&g_val出来的地址也绝对不可能是真实的物理地址!
如果&g_val出来的是物理地址的话,那么父子进程的g_val就表示的同一块空间,那么从同一块空间读取出来的值就应该是一样的,但是事实却不是这样的!
那么就说明我们在语言级别上的取地址,取出来的绝对不是真实的物理地址,相对的我们把这种地址叫做虚拟地址!

引入进程地址空间

通过上面的例子我们知道了我们在语言层面上取出来的地址绝对不是真实的物理地址,这个我们知道了,可是这与我们的进程地址空间有什么关系?
在讲解进程地址空间之前,我们先讲一个故事:

在遥远的美国,有一个大富豪,这个大富豪有100亿美金,同时他有4个私生子:A、B、C、D这4个私生子都不知道彼此的存在,都认为自己是大富豪唯一的孩子:
在这里插入图片描述
有一天呢大富豪对分别A、B、C、D单独说:你好好干,以后我的这100亿家产都是你的!用我们现在的话说,大富豪的承诺相当于在给A、B、C、D画饼!A、B、C、D都认为自己拥有100亿美金,因为他们都认为自己是大富豪的唯一继承人!在某一天A对大富豪说:“爹,给我1000美金,我需要买个表”,大富豪说没问题,大富豪就从100亿美金中拿出了1000美金给了A,B这时候也想大富豪申请了900美金,大富豪毫不犹豫的就答应了,但是C对大富豪说:“爹,快给我50亿美金,我这出了点事,需要摆一下”,大富豪说:“滚!”,大富豪无情的拒绝了C的请求!但是C还是认为自己拥有100亿美金,因为它认为自己是大富豪的唯一继承人,等到大富豪驾鹤西去之时就是100亿美金到账之日!
在上面的故事中呢:A、B、C、D相当于我们的进程;
大富豪给A、B、C、D画的“饼”就是进程地址空间
100亿美金就是物理内存;
大富豪就是OS;
其中A、B、C、D向大富豪借钱的动作就是向OS申请物理内存!申请的太多,OS是会拒绝我们的!
OS最为我们计算机中的管理者,那么它要不要把给进程们画的“饼”管理起来呢?
答案是要的!为什么呢?如果不管理起来的画,在进程多起来的时候OS也不知到他给进程们到底画的什么饼!
那么如何管理这些“饼”呢?
先描述,再组织
在Linux中,OS利用了一个struct mm_struct{}的结构体将这个饼管理了起来,每个进程都有属于自己的专属大饼,其中进程的pcb是中具有指向该大饼的指针!OS呢会将这些大饼做一个区域划分!比如规定大饼的这个区域是干嘛的,那个区域是干嘛的!这些区域,也就是我们看到的什么堆区、栈区、代码区等等!这个大饼也就是我们老师在日常中讲解的C/C++内存布局:
在这里插入图片描述

Linux下的mm_struct 结构体就是专门记录这张大饼的!

struct mm_struct{
long Code_start;
long Code_end;
long init_start;
long init_end;
……
long stack_start;
long stack_end;
}

当我们向堆区申请空间时heap_end就会变大,free时heap_end就会变小!
说白了进程地址空间就是OS欺骗进程的一种手段,让内存误以为自己拥有全部的物理内存;

页表

可是进程地址空间毕竟只是逻辑上的内存,并不是真正的物理内存,是不能存储数据,进程的数据和代码是只能存储在物理内存上的,但是进程使用的是虚拟内存!进程也只能访问虚拟地址,但是实际的数据是存储在物理内存上的,那么进程是如何通过虚拟地址拿到数据和代码的?
实际上在虚拟地址与物理地址之间是有一种映射关系的,这种映射关系被存储在页表中!每个进程都有自己的页表!
在这里插入图片描述
当进程需要访问虚拟地址上某一处的数据时,OS就会拿着进程提供的虚拟地址,根据该进程提供的页表转换成对于的物理地址,然后去对于的物理内存上取数据在交给进程!这个过程进程是看不到的,站在进程的角度就是,我(进程)需要访问虚拟地址为0x11223344处的数据,然后就直接拿到了数据,在进程看来它就认为自己的数据是存储在虚拟内存上的,只要自己需要,随时都可以拿到,殊不知其真实数据是存储在物理内存上的,进程之所以能随时拿到数据,都是由OS完成的!我们来画个图来理解:
在这里插入图片描述
只要理解了这一层,我们就能回答开头的问题了:
老师们经常给我们讲的内存分布实际上并不是真实(物理内存)的内存的分布,而是OS给进程画的一张“大饼”,就是让进程认为自己一个就拥有整个内存!说白了进程空间就是OS欺骗进程的一种手段!
每个进程都有属于自己的一张进程地址空间和对应的页表!

回答了开头的问题,我们再来解释一下,上面代码表现出的情况,父子进程对于g_val取地址取出的地址是一样的,但是父子进程从g_val取出来的值却不一样:
首先父进程会有自己的pcb、进程空间地址、页表,那么子进程也会拥有这些东西,但是子进程作为父进程的儿子,它会继承父进程的大部分属性,包括进程空间地址、页表等,画图表示就是:
在这里插入图片描述
那么根据上面图的表示的话,父子进程的g_val不就是同一块空间嘛,取出来的值也应该是一样的,可是为什么父子进程g_val取出来不同的值?
我们需要记得进程之间是具有独立性的!包括父子进程之间也是如此!当我们的子进程在尝试对g_val变量的值进行修改时,为了不影响父进程的正常读取g_val,OS会启动"写时拷贝"技术,当父子进程中的某个进程需要对父子进程共享的同一块空间进行修改时,OS会在物理内存重新开辟一块一摸一样大的空间,然后再将数据拷贝过来,修改需要修改数据的进程的页表映射关系!此时需要修改数据的进程就可以随意的修改了,同时不会影响另一个进程的数据!保持了进程之间的独立性;
画个图来表示:
在这里插入图片描述
这也就解释了为什么父子进程的g_val是同一个地址,但是却存着不同的值!
地址相同的原因就是:g_val都处于父子进程的进程地址空间的同一个位置(这里说的“处于”并不是真实的存储,而是逻辑上的存储!),取地址取出来的地址当然一样,但是由于子进程的g_val++造成了写实拷贝,就导致了父子进程的g_val映射到不同的物理地址空间,取出来的值自然不一样!
写时拷贝是发生在物理内存,对于虚拟内存没有影响!
注意:我们平时&地址,取出来的全是虚拟地址(也就是进程地址空间中的地址),我们用户没办法取到真实的物理地址,毕竟谁叫我们的进程被OS欺骗了,痴痴的认为自己享有全部内存!
明白了上面的例子,那么我们也就能很好的明白了使用fork函数时,利用变量接受fork返回值时,明明是同一个变量(虚拟地址相同),但是再父子进程中却输出了不同的值;
主要是应为再fork函数的内部也就是return的前一步的时候,子进程就已经被创建出来了,此时对于接受fork返回值的变量在父子进程中也还是映射的同一块物理空间,但是当return的时候,就会向这个接受返回值的变量中写入数据,此时就会触发写时拷贝,那么这时候父子进程中的某个进程就会为自己的这个接受fork返回值的变量重新映射一块新的物理空间!这也就是fork函数能返回两个返回值的秘密!实际上并不是真的能返回两个返回值,只是父子进程中的接受返回值的变量已经是两块独立的物理空间了,不在是同一块!在虚拟内存上他们也许是同一块,但是,真实情况并不是!!!
同时页表也不止是会映射虚拟地址的物理地址,页表同时也会记录一下映射的物理地址的读写权限!
比如:
在这里插入图片描述
这也是为什么我们平常所说的代码区的数据只能读!不可修改的原因!
因为当我们进程试图修改代码区的数据时,OS会拿着进程提供的虚拟地址(代码区的地址),然后根据页表映射到对应的物理内存上去,但是OS这时候发现这次映射的物理空间在页表中的权限也就只有可读,不可修改!我们的操作属于权限放大了,OS会直接拒绝我们的请求!并不是这块空间(物理内存)本身就只是可读的!而是我们通过一些手段从逻辑上限制了这块空间(物理内存)的权限!

为什么要有进程地址空间

上面我们大概讲解了什么是进程地址空间和怎么使用进程地址空间,但是为什么要有这个东西呢?

1、防止地址随意访问,保护我们进程的安全和独立;
假设我们不使用虚拟内存,就直接使用物理内存:
在这里插入图片描述
我们现在在物理内存中加载了两个程序,现在A程序是我们写的,但是我们的代码能力有问题,我们的A程序有bug,当我们cpu在处理进程A的时候,会从进程中读取到不属于进程A的地址,也就是说A进程存在野指针问题,但是刚好这个野指针被OS分配给了进程B使用;
要是这时候我们的进程A有个对该野指针解引用并修改数据的操作的话,那就完了因为这就造成了我们明明实在运行进程A但是由于野指针的问题间接的将B进程的数据修改了,如果进程B是个银行的账户信息的话,那么后果就会很严重!这也就破坏了进程之间的独立性!!同时也对进程的安全运行造成了威胁!但是我们使用虚拟内存时,我们如果造成了越界访问,OS会在映射该虚拟地址的时候检测出来,从而拒绝我们的访问!这也就让我们无法随意的根据地址访问其他空间了,同时进程的独立性和安全性也就增加了!
2、进程管理与内存管理解耦合了;
再此之前我们先来谈谈malloc的本质!
请问只要是我们已使用malloc或new申请空间OS就会立即给我们吗?
答案:显然不是!OS作为整个计算机最基础的软件,也是整个计算机中的管理者!它是不允许发生任何不高效和浪费的操作的!
如果有,那么一定是OS的bug;
如果OS在我们申请的时候就把空间给我们了,那么我们能保证我们申请了就一定使用吗?我们一定写过这样的代码:在程序的开头就先申请了一段空间,但是我们可能写了几十行代码才开始使用这块空间!那么在你从申请空间开始到你真正使用这块空间之间,这块空间就一直被我们占着,其他进程也用不到,实属有点“站着茅坑不拉屎”的感觉!你说一个进程这样!OS还能理解,但是如果每个进程都像这样了!这就会严重的造成内存资源使用不充分、不高效;
如果在我们并未真正使用这段空间的时间段内,OS将这块空间拿去给需要的进程使用,当我们真正需要使用这块空间的时候OS再给我们,这样的话内存使用率不就起来了!
那也有人会说,我们申请了立马使用就好了嘛,对不起!在你刚好申请完这块空间的时候CPU处理你的时间到了,该换下一个进程被CPU处理了!在你等待下一次CPU处理的时候,你又是单独站在这块空间,自己不用其他进程也用不了!又会造成内存资源的浪费!OS也不会允许!
为此在我们向OS申请空间的时候,OS不会立马给我们!而是当我们真正需要的时候才会给我们!
我们平常使用的malloc、new就是这样的原理;malloc、new是在虚拟内存上开辟空间(也就是逻辑上开辟的空间)返回的指针也自然是虚拟指针,虽然我们有了空间,但这些空间毕竟是逻辑上的,并不能真实的存储数据,也就是说这些虚拟空间还没有在页表中建立起与物理内存的映射关系,进程现在拿到的只是一张空头支票,具体的兑换,还是得靠OS!只有当我们真正需要使用这块空间的时候,OS才会将我们申请的虚拟空间映射到对应的物理内存!也就是为我们的虚拟空间在页表中建立起物理地址!只有完成映射关系,我们进程才能算是真正的拥有自己的空间(物理空间)!
同时我们进程也不必关心,OS到底给我们映射的那块空间,在物理内存中是否连续等!OS可以在物理内存的任何位置映射空间,物理内存并不一定是连续的,但是我们在虚拟内存上申请的空间一定是连续的!
我们作为进程是不关心我们的数据到底存储在物理内存的那一块空间的、申请的物理空间是否连续等等,在进程看来进程空间地址就是它的“内存”,只要在进程空间地址上连续就行了!至于映射到物理内存上是什么情况,我们进程压根不关心!
为此我们把就把内存管理与进程管理分开了!内存管理就处理内存的事!进程管理就专门管理进程!两个管理之间互不干扰!
在这里插入图片描述
如果没有进程空间地址的话,我们的进程在申请空间的时候,OS就会立马给它,这样导致内存资源浪费不说!OS会需要去刻意寻找一块物理内存,这时候就造成进程管理与内存管理耦合!也就是说我们我们在进行进程管理的时候就必须借助内存管理的力量!这是我们不希望看到的,我们希望进程管理能够单独完成自己的事情,内存管理也能单独完成自己的事情,两个进程耦合度不要太高!保证我们内存管理崩溃的时候不影响进程管理!进程管理崩溃的时候不影响内存管理!
有了进程地址空间,我们在申请空间的时候就只启用进程管理,先申请虚拟内存,当我们真正需要的时候,再启动内存管理来为我们分配物理空间!这样的话就算内存管理崩溃掉了也不影响进程管理!
3、让进程以统一的视角看待内存;
虚拟内存是OS欺骗进程的一种手段,进程在看待进程的时候都认为自己拥有整块内存,然后开始对着“这块内存”开始布局自己的代码和数据,但是实际上这些代码和数据到底存没存储起来,还得看OS,但是站在进程的角度,他是认为我们已经布局完整个内存了!进程是看不到真实物理内存的!进程只能看到进程地址空间!
4、可以充分的利用内存资源,让内存的利用率变的高效起来!
比如:两个进程可能都需要访问某个动态库;如果没有进程地址空间的话,OS就会将这个动态库加载内存两次,也就是内存中会有两份一模一样的数据!这是没必要的!但是有了进程地址空间过后,我们可以让两个进程的虚拟地址同时映射到这同一份数据!也就是说两个进程可以共享这份数据!这份数据也就只需要在内存中存在一份就行了!但是在进程看来他们都认为这份数据是自己独享的!
在这里插入图片描述

重新理解进程地址空间

请问我们的程序在编译完毕,但是还没有加载进内存的时候,我们的程序内部是否有地址呢?
答案是当然有的!
我们可以来看看一段代码的汇编文件:
在这里插入图片描述
我们将这段程序先编译成可执行程序,然后利用命令objdump -S对其进行反汇编:
在这里插入图片描述
我们会发现,在我们的程序在未加载进内存的时候,编译器就已经确定好了各条指令的地址!这是为什么??
答:进程地址空间不止是欺骗进程的,也会连同编译器也一起欺骗!当然这都是非常不标准的描述,严格意义上来说,源代码在被编译的时候,就已经按照虚拟地址空间的方式对代码和数据进行了地址的编制,只不过只些代码和数据的地址都是虚拟地址,并不是真实的物理地址!
只有当我们的程序被加载进内存了,才会真正的拥有物理地址!
:现在我们来理一理整个程序的运行过程:
1、将我们的程序加载进内存(注意并不是一次性全部加载进去,而是先加载一些比较重要的代码和数据);
2、OS为该程序建立pcb,来管理该进程;
3、OS为该进程创建地址空间地址和页表;
4、cpu从特定的进程空间地址处读取数据!然后OS在根据cpu提供的虚拟地址,映射到对应物理地址,获取对应的数据给cpu,cpu开始处理!如果OS在根据cpu提供虚拟地址没有建立起对应的物理地址时,OS会暂停cpu对于该进程的处理,然后重新加载一部分数据进入内存,然后再建立映射关系,出现这种情况:叫做缺页中断
我们画个图来理解:
在这里插入图片描述
注意:在CPU上读取到的地址,全是进程空间上的地址,也就是虚拟地址!CPU也不会直接去物理内存上读取数据!

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

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

相关文章

【设计模式】 模板方法模式介绍及C代码实现

【设计模式】 模板方法模式介绍及C代码实现 背景 在软件构建过程中,对于某一项任务,它常常有稳定的整体操作结构,但各个子步骤却有很多改变的需求,或者由于固有的原因(比如框架与应用之间的关系)而无法和任…

2023年1月综合预订类APP用户洞察——旅游市场复苏明显,三年需求春节集中释放

2023年1月,随着国家对新型冠状病毒感染实施“乙类乙管”,不再对入境人员和货物等采取检疫传染病管理措施,并且取消入境后全员核酸检测和集中隔离,横亘在旅游者与旅游目的地之间的隔阂从此彻底消失。2023年1月恰逢春节假期&#xf…

SQL零基础入门学习(十一)

SQL零基础入门学习(十) SQL NOT NULL 约束 NOT NULL 约束强制列不接受 NULL 值。 NOT NULL 约束强制字段始终包含值。这意味着,如果不向字段添加值,就无法插入新记录或者更新记录。 下面的 SQL 强制 “ID” 列、 “LastName” …

Mac OSX下使用VMware Fusion 配置静态IP 图文教程指南

目录一. 前言二. Mac OSX下使用VMware Fusion 配置静态IP2.1 了解静态IP如何划分基础知识2.2 Centos7 安装操作系统时图形界面配置静态IP2.3 Centos7安装操作系统后修改动态IP为静态IP三参考文献一. 前言 Mac OSX 下使用VMware Fusion 创建的虚拟机,默认是通过DHCP…

雷达实战之射频前端配置说明

在无线通信领域,射频系统主要分为射频前端,以及基带。从发射通路来看,基带完成语音等原始信息通过AD转化等手段转化成基带信号,然后经过调制生成包含跟多有效信息,且适合信道传输的信号,最后通过射频前端将信号发射出去…

msys2+minGW方案编译ffmpeg的最佳实践

一、Win10 64bit编译环境的建立1)从http://www.msys2.org/下载 msys2-x86_64-xxx.exe2) 安装msys2到默认路径 C:\msys64\3) 运行MSYS2 w644)执行 pacman -Syu 更新系统当出现提示时,选择y5) 当窗口关闭时,重…

九龙证券|美股创年内最大周跌幅!美联储官员密集发声!波音重挫近5%

当地时刻2月24日,美股三大指数收盘明显跌落。道指跌1.02%,标普500指数跌1.05%,纳指跌1.69%。 大型科技股普跌,微软、亚马逊跌超2%。波音大跌4.8%,居道指跌幅榜首位,公司因机身部件有问题再次暂停向用户交付…

zabbix4.0-动作-邮件告警

目录 1、创建动作Actions 动作触发流程 创建一个动作 2、配置 Media types 媒介类型,添加一个发件邮箱来发送告警邮件 3、配置 Users Media,添加一个收件邮箱来接收告警邮件 4、更改一个触发器表达式来触发动作Action,最终发送告警邮…

【数据库】MongoDB数据库详解

目录 一,数据库管理系统 1, 什么是数据库 2,什么是数据库管理系统 二, NoSQL 是什么 1,NoSQL 简介 2,NoSQL数据库 3,NoSQL 与 RDBMS 对比 三,MongoDB简介 1, MongoDB 是什…

Python入门教程(非常详细)从零基础入门到精通,看完这一篇就够了

前言 本文罗列了了python零基础入门到精通的详细教程,内容均以知识目录的形式展开。 第一章:python基础之markdown Typora软件下载Typora基本使用Typora补充说明编程与编程语言计算机的本质计算机五大组成部分计算机三大核心硬件操作系统 第二章&…

【LeetCode】剑指 Offer 15. 二进制中1的个数 p100 -- Java Version

题目链接:https://leetcode.cn/problems/er-jin-zhi-zhong-1de-ge-shu-lcof/?favoritexb9nqhhg 1. 题目介绍(15. 二进制中1的个数) 编写一个函数,输入是一个无符号整数(以二进制串的形式),返回…

Systemverilog覆盖率的合并和计算方式

在systemverilog中,对于一个covergroup来说,可能会有多个instance,我们可能需要对这些instance覆盖率进行操作。 只保存covergroup type的覆盖率,不需要保存instance-specified的覆盖率coverage type和instance-specified的覆盖率…

SVM支持向量机理解_KKT条件_拉格朗日对偶_SMO算法

目录 一、支持向量机基本型(线性可分) 1.1 问题描述 1.2 参考资料 二、KKT条件 2.1 KKT条件的几个部分 2.1.1 原始条件 2.1.2 梯度条件 2.1.3 松弛互补条件 2.2 参考资料 三、对偶形式 四、SMO算法 五、线性不可分情形 六、核函数 一、支持…

TimeWheel时间轮算法原理及实现(附源码)

时间轮算法原理及实现前言1.时间轮核心2.简单定时器3.任务队列4.优化任务队列5.简单时间轮6.多层时间轮前言 在各种业务场景中,我们总是会需要一些定时进行一些操作,这些操作可能是需要在指定的某个时间点操作,也可能是每过一个固定的时间间隔后进行操作,这就要求我们需要有一个…

【蓝桥OJ—C语言】高斯日记、马虎的算式、第39级台阶

文章目录高斯日记马虎的算式第39级台阶总结高斯日记 题目: 大数学家高斯有个好习惯:无论如何都要记日记。 他的日记有个与众不同的地方,他从不注明年月日,而是用一个整数代替,比如:4210。 后来人们知道&am…

You Only Need 90K Parameters to Adapt Light 论文阅读笔记

这是BMVC2022的论文,提出了一个轻量化的局部全局双支路的低光照图像质量增强网络,有监督。 思路是先用encoder f(⋅)f(\cdot)f(⋅)转到raw-RGB域,再用decoder gt(⋅)g_t(\cdot)gt​(⋅)模拟ISP过程转到sRGB域。虽然文章好像没有明确指出&…

【蓝牙mesh】Network协议层介绍

【蓝牙mesh】Network协议层介绍 Network层简介 上一章节我们讲解了蓝牙Mesh中Lower层的功能和数据格式。 Lower层的数据往下传输就到了网络层(Network Layer)。网络层定义了收到Lower层的数据后,如何对其进行判断、封装、加密、认证&#xf…

学习(mianshi)必备-ClickHouse高性能查询/写入和常见注意事项(五)

目录 一、ClickHouse高性能查询原因-稀疏索引 二、ClickHouse高性能写入-LSM-Tree存储结构 什么是LSM-Tree 三、ClickHouse的常见注意事项和异常问题排查 一、ClickHouse高性能查询原因-稀疏索引 密集索引: 在密集索引中,数据库中的每个键值都有一个索引记录&…

Amazon S3 服务15岁生日快乐!

2021年3月14日,作为第一个发布的服务,Amazon S3 服务15周岁啦!在中国文化里,15岁是个临界点,是从“舞勺之年”到“舞象之年”的过渡。相信对于 Amazon S3 和其他的云服务15周岁也将是其迎接更加美好未来的全新起点。亚…

【论文解读】如何使用1B参数的小模型吊打GPT3.5

大型语言模型 (LLM) 通过利用思维链 (CoT) 提示生成中间推理链作为推断答案的基本原理,在复杂推理上表现出了令人印象深刻的性能。 然而现有的 CoT 研究主要集中在语言模态上。 我们提出 Multimodal-CoT(多模态思维链推理模型),它…