使用Bootloader加载操作系统内核1:i386分段的机制
cac55 2025-03-19 10:46 25 浏览 0 评论
第1节先看一下bootloader的加载执行,bootloader用来加载操作系统内核,我们的操作系统运行在(i386)x86-32位CPU上面。
因为i386运行的时候,也是先运行在8086模式,然后才切换到i386模式去运行,所以我们对8086跟i386的寄存器,寻址的方式,以及分段机制要有一定的了解,8086是16位的CPU,它有8个16位的通用寄存器,4个16位的段寄存器,受当时工业的限制,它的数据线只能做到16位,但是地址线可以做到20位,这样它寻址的范围就是2的20次方1M,
但是指令在访问地址的时候,通常是把地址放在寄存器里面的,那么这时候一个16位的寄存器,就无法放一个20位的地址,所以8086它的寻址方式是结合着段寄存器来操作,它把段寄存器里面存储的16位的地址左移4位,得到一个20位的基地址,再加上一个偏移地址就可以获取到实际的一个20位的物理地址,
后来的i386就是32位的CPU,它的寄存器数据线跟地址线都是32位的,那么这时候一个寄存器里面,就可以存放一个32位地址,这时候其实就不需要段寄存器了,但是i386为了跟8086进行兼容,它依然保留了段寄存器,并且使用段寄存器来实现保护机制,这时候段寄存器里面存放的是段选择子,根据段选择子跟偏移地址来获取到实际的物理地址,我们一会会看,
i386又称为保护模式,它使用的是虚拟地址,通过虚拟地址来实现一些保护的机制,而8086又称为实模式它使用的是物理地址,使用物理地址的话就无法实现保护的机制,比如在物理地址里面,代码段跟数据段都是一样的都可以读写,这样我们就无法对代码进行保护,而使用虚拟地址的话,我们就可以针对代码段跟数据段设置不同的权限,只有权限满足的话,我们才把这个虚拟地址转换成它实际的物理地址,
在i386中可以使用分段机制来实现保护跟虚拟地址,在分段机制中,每一个进程会有一个自己的虚拟地址空间,虚拟地址空间的大小是0-4G,然后物理地址空间的大小也是0-4G如下图,我们在虚拟地址空间跟物理地址空间中划分成多个段,比如可以是数据段代码段,每个段的大小可以是不一样的,
然后每一个进程会有一个描述符表,用来描述怎么把一个虚拟地址映射到实际的物理地址,描述符表中的每一行就对应一个段的内容,比如代码段对应一行,数据段对应一行,每一行中包含这个段的段基址,就是在物理地址空间的首地址,我们在访问一个虚拟地址的时候,就把这个虚拟地址作为偏移地址,通过它所对应的段基址加上这个偏移地址,从而获取到实际的物理地址,
段界限是用来描述这个段它的范围,如果你的段基址加上偏移地址超过这个段界限的话,那么它访问的时候就会出错,段的权限是设置读写或者是只读,
我们在访问一个虚拟地址的时候,首先找到这个段它所对应的段选择子,前面我们说过段选择子是存放在段寄存器里面的,代码段使用的就是CS,数据段使用的是DS,段选择子就是在描述符表中的一个索引,就是描述符表这个数组的下标,
比如代码段我们就从CS寄存器里面获取到它的段选择子比如上图就是0,以0作为数组的下标,去获取到它所对应的段描述符,然后获取到它的段基址,我们用段基址加上虚拟地址作为偏移,看获取的实际物理地址有没有超过段界限,如果没有超过的话再看它的权限,权限是不是满足,如果都满足的话,我们就把它虚拟地址转换成实际物理地址去访问,
我们可以看到使用分段机制,我们可以把不同进程的相同地址映射到物理地址空间中的不同的地方,这样它们访问就不会冲突,然后之前的进程使用的内存块不用的时候,我们就可以把当前进程的虚拟地址映射到这个内存块,去使用这块内存,
如果一个进程使用内存比较多,我们也可以把一些不用的内存先换到磁盘上,然后当前的程序就可以使用这块内存,这样每一个进程就可以认为自己是有4G的内存空间,这样也体现了一种虚拟跟抽象的思想。
除了分段机制还可以使用分页来实现保护机制,在i386中有三个地址,程序里面使用的地址是虚拟地址也叫做逻辑地址,逻辑地址经过分段机制以后转换成的是线性地址,线性地址经过分页机制后转换的是物理地址,因为分页机制它跟分段机制相比有很多的优点,比如权限的粒度可以设置的更细,所以现在CPU一般使用分页机制,像后来的ARM,Risc-V使用的就是分页机制,
但X86系统为了跟之前的处理器进行兼容,它既有分段的机制也有分页的机制,而且分段的机制只能打开不能关闭,而分页机制可以打开也可以关闭,而像linux操作系统为了在X86系统上只使用分页机制,而不使用分段的机制,它把分段做了透明处理,就是它会把虚拟地址跟线性地址进行一一映射,就是每一个虚拟地址是等于线性地址的,我们在描述符表中把每个段的段基址设成0,把段界限设置成4G,这样就可以实现一一映射,
比如如果虚拟地址是1,000,那么通过描述符表得到它的物理地址就是0+1,000,是等于线性地址的,我们的实现跟Linux一样采用的是一一映射,在bootloader阶段我们的分页机制是没有打开的,所以这时候虚拟地址就等于线性地址等于物理地址的,
然后我们就看i386它具体是怎么实现分段机制的,首先需要一个描述符表,这个描述符表是存放在内存中的,我们看下面bootloader的代码,gdt就是在内存中一块区域用来存放描述符表,就类似一个数组,数组每一项对应的就是一个段描述符,比如第一个SEG_NULL描述的是一个空段是没有意义的用来表示描述符表的开始,第二项SEG对应的是代码段,第三项SEG对应的是数据段,
我们看下面的图,每个段描述符是64位的,从0到23位跟24到31位用来设置段基址,然后段限长是0-15位跟属性部分的16-19位,在属性部分可以去设置一些读写的权限。可以看到段限长跟段基址位置都不是连续的,这也是为了跟之前的处理器进行兼容,之前的80286它实现分段机制时就只使用前面的BYTE0到BYTE4,
上面的SEG宏和SEG_NULL宏就是根据段描述符每个字段的描述,去设置一个64位的段描述符的值,比如SEG_NULL宏里的.word就是两个字节,两个word就是4个字节,后面还有4个byte总共就是8个字节,SEG宏也是一样有8个字节,去根据段描述符的描述去设置权限跟段限长这些,比如代码段SEG(STA_X|STA_R, 0x0, 0xffffffff)里的参数STA_X|STA_R就设置权限是可读可运行,参数0x0设置段基址是0,0xffffffff是设置段限长是4G,然后数据段SEG(STA_W, 0x0, 0xffffffff)设置权限是可读写,段基址是0,段限长是4G,这样就可以实现虚拟地址跟物理地址的一一映射,
描述符表设置完成以后,CPU怎么知道它在内存中的什么位置呢,CPU通过一个专门的GDTR寄存器来存放描述符表的地址,GDTR是48位的如上图,32位用来存放描述符表gdt的起始地址,后面16位用来存放描述符表的段限长,就是描述符表的大小size-1,这样知道了描述符表的起始地址跟它的段限长,我们就知道描述符表的范围,
对于GDTR寄存器里面的内容也是放在内存中的,在内存有一块区域gdtdesc如下图,前面.word 0x17就是描述符表gdt的段限长size-1,后面.long gdt就是描述符表的地址,这个gdtdesc后面会使用lgdt指令把这个地址里面内容加载到GDTR寄存器,这样CPU就可以通过GDTR寄存器找到描述符表,
描述符表设置完成以后,我们就看段选择子,段选择子就是每个段在描述符表中的偏移,gdt表里第1个段的偏移是0,那么它的段选择子就是0,第二个段是代码段它的偏移是8个字节,所以它的段选择子是8,第三段是数据段它的偏移是16,所以它的段选择子是16,代码段的段选择子是放在CS寄存器里面的,数据段是放在DS寄存器里面的,下图就会设置相应段的段选择子,代码段是8,数据段是0x10就是16,
后面我们就会把这两个值设置到CS跟DS寄存器里面,这样CPU在访问一个地址的时候,它就首先根据CS寄存器获取到段选择子,然后再以段选择子作为描述符表中的偏移,去获取到对应的段描述符,获取到段基址,再从IP寄存器里面获取到偏移地址,把段基址加上偏移地址就可以获取到指定的地址,
访问数据也是一样,从DS寄存器里面获取到对应的段描述符,然后再去描述符表中获取到段基址,段基址再加上这个数据所对应的地址作为偏移地址,最终可以获取到实际的物理地址,
这就是i386分段的机制,下一节我们节省bootloader的启动流程。
相关推荐
- unetbootin中文版:能够将Linux系统装进U盘的U盘启动盘制作工具
-
unetbootin中文版是一款能够将Linux操作系统装进U盘或移动硬盘的U盘启动盘制作工具,制作好的U盘启动盘能够用于电脑的维护和系统还原等操作,使用起来非常地不错。该软件不会基于操作系统使用特定...
- 实用之选,实用之改:DELL 戴尔 灵越14CR-4528B 小改作业
-
昨天发布了一篇三脚架,今天有时间也写写早就准备写的DELL戴尔灵越14CR-4528B作业吧。话说上个笔记本还是2006年底买的华硕A6JE,电脑挺不错的,在家上上网也够用了,就是转轴设计缺陷,容...
- 教你如何制作一个启动U盘,从此电脑不用找专人做系统
-
在电脑使用中,老是遇到卡顿,蓝屏,重启等很多故障,大多都是因为自己日常使用习惯而造成的,很多用户在下载软件的时候不知不觉中都被安装许多乱七八糟的软件,当电脑乱七八糟的东西过多的时候我们就重新来装一个系...
- 8、Deepin操作系统启动盘(系统盘)制作
-
1、在Deepin官网https://www.deepin.org/zh/download/下载原版Deepin操作系统2、同时在Deepin官网https://www.deepin.org/zh/d...
- 电脑死机怎么办,电脑如何使用U盘重装系统
-
电脑死机是我们最常遇到的系统故障,遇到死机时通常重启就可以解决,不过系统损坏引起的死机就只能重装系统,那么电脑死机如何重装系统呢?下面来看看电脑死机怎么办如何使用U盘重装系统_小白一键重装系统官网。 ...
- bootmgr is compressed无法启动系统
-
bootmgriscompressedPressCtrlAltDeltorestart,电脑启动后无法正常开机出现了这样的字样,就是说明你的C盘驱动被压缩解决方法:1、使用系统光盘或者...
- 新手教程!如何分辨BIOS启动列表(菜单)中的各种启动项
-
在BIOS启动菜单中识别各类启动项,是新手安装系统或调整启动顺序的必备技能。下面用最直观的方式,为你梳理常见启动项及其含义,帮助你快速上手:一、传统存储设备启动项1.Floppy(软盘驱动器)对应...
- 带回家的MINI客厅电脑,自学成才,分享U盘装系统教程
-
刚好老家新装修了房子,客厅买了个大电视,本来是想在客厅弄台主机,接电视玩,大屏幕玩的才爽,但是台式机箱太占地方了。网上逛了一圈,发现有专门的客厅电脑,就搞了一个,外形不错,放客厅很有档次,主要是主机太...
- 电脑基础知识:BIOS简介及其与Windows操作系统的关系
-
什么是BIOS?BIOS,全称BasicInputOutputSystem,即“基本输入输出系统”,是一段固化在电脑主板芯片上的底层固件程序。它类似于一款极简化的操作系统,负责电脑开机时的硬件初...
- win 7 系统注册表文件丢失或损坏,求不重做系统的解决办法!
-
粉丝问题解答:win7系统注册表文件丢失或损坏,求不重做系统的解决办法!解决方法:你只需要有启动盘即可,不需要其他的。之所以要求启动盘,是因为下面要对系统文件进行还原覆盖,所以不能用原系统启动。用...
- UEFI怎么装Win7 小编呕血解难点!
-
自从广开言路之后,小编就被你们害苦了,这不,一条评论又让小编彻夜难眠。另外某些小伙伴坐不上沙发后提出要上墙的需求,其实呢只要大家提出的问题具有普遍性、有难度、而且适合小编做微信内容的话,都有机会将你们...
- 固态攻坚战——ASUS 华硕k45v换固态、拆机清灰教程
-
作者:蘑菇爱上我现在固态白菜价固态对于电脑体验的提升还是很大的对于固态存储芯片的问题没什么好说的有钱mlc,没钱tlc,不需要考虑什么寿命的问题,我用了一年多的m600,写入才3TB品牌很重要,主控...
- MBR启动报错?Win10不重装一样能好!
-
Win10一遇到启动故障,很多小伙伴可能就会抓瞎,这可怎么弄,我不会修复啊!其实大可不必惊慌,就像这种最常见的Winload启动错误,多半都是MBR分区表丢失造成的(UEFI分区模式的几乎没有这种故障...
- 从零开始:硬盘手动装系统全攻略
-
手动安装操作系统是计算机技术必备的基本技能。对于初学者来说,可能会感到有些挑战。但通过掌握硬盘手动装系统方法,你可以亲身体验整个安装过程,进而更好地理解操作系统的工作原理。本文将详细介绍硬盘手动装系统...
- 电脑开机后显示File:BCD错误0xc000000f
-
WIN7\WIN8\WIN101、一个win864位PE。这个64位PE的相关文件,路径在boot\BOOT.WIM实机测试,开机后显示File:\EFI\Microsoft\Boot\BCD,...
你 发表评论:
欢迎- 一周热门
- 最近发表
- 标签列表
-
- 如何绘制折线图 (52)
- javaabstract (48)
- 新浪微博头像 (53)
- grub4dos (66)
- s扫描器 (51)
- httpfile dll (48)
- ps实例教程 (55)
- taskmgr (51)
- s spline (61)
- vnc远程控制 (47)
- 数据丢失 (47)
- wbem (57)
- flac文件 (72)
- 网页制作基础教程 (53)
- 镜像文件刻录 (61)
- ug5 0软件免费下载 (78)
- debian下载 (53)
- ubuntu10 04 (60)
- web qq登录 (59)
- 笔记本变成无线路由 (52)
- flash player 11 4 (50)
- 右键菜单清理 (78)
- cuteftp 注册码 (57)
- ospf协议 (53)
- ms17 010 下载 (60)