计算机从加电到系统(Linux)启动完成
0x0 背景
在我参加的面试和我面试别人、或者参加别人对别人的面试的事后经常遇到的一个问题就是:请从计算机加电开始描述一下计算机启动到操作系统正式启动起来的全过程。这是一个考验对计算机体系结构和基本知识了解程度的问题。今天也就特别针对这个问题做一个回答,答案是基于80x86结构Linux 2.6及更高版本内核的为基准操作系统为例来回答的。
0x1 从加电到BIOS启动
STEP 1 加电引导寄存器置位
这一过程指的是,计算机加电后,一个特殊电路会在CPU对应的一阵针脚产生一个逻辑电平,这个电平的值从针脚进入CPU后,会引发将寄存器(cs、eip等等)设置成特定值。
STEP 2 引导BIOS启动
这一过程指的是系统从物理地址0xfffffff0处加载一段程序到只读内存(ROM-> Read Only Memory),这个程序在80x86体系架构中一般称为BIOS。
相关知识学习
- MS-DOS的很多系统调用依赖BIOS
- Linux进入保护模式后不再依赖BIOS,BIOS只能以实模式运行。
实模式的寻址是20位总线寻址,支持的寻址空间为2^20,也就是1MB,保护模式目前在x86结构下,支持4GB寻址;
实际区别主要是EIP中的虚地址到实地址转化的区别:
实模式是seg(eip地址)*16+offset(4为偏移量);
保护模式实EIP的16位地址代表页面位置,一个页在操作系统中都学习过是4KB,1M*4K = 4G,我相信很多人就此理解了为啥页的大小要设计成4K;
0x2 BIOS引导操作系统镜像的加载
STEP 1 检查硬件
没啥可说的,一般可认为是开机加电自检,这个阶段会显示一些信息,包括BIOS版本这一类的信息。
STEP 2 初始化硬件
主要是避免IRQ先与I/O冲突,本阶段最后会显示所有PCI(总线--内部硬件通信线路)设备信息。
STEP 3 搜索操作系统
从软盘、网络、磁盘、CD-ROM的主引导扇区上搜索。找到后加载注意到扇区的内容到0x00007c00的位置(RAM中),跳转到这个地址,开始执行这段代码,这段程序叫做bootloader。
由于大小限制,linux的启动程序GRUB(GRand Unified BootLoader)或者是LILO(LInux LOader)被分为两部分。
第一部分就是加载到0x00007c00的这一段,他会把自己移动到0x00096a00的位置,建立实模式栈(0x00098000~0x000969ff)
第一部分吧第二部分加载到0x00096c00开始的位置中。
以上的位置都是在RAM中。
第二部分搜索磁盘上的OS景象,,把对应的扇区拷贝到RAM中执行:
1、首先把内核景象的第一个512B的部分从0x00090000处装入RAM中;
2、把setup()函数代码段装入0x00090200位置(RAM);
3、加载其他内核部分从高(0x00100000)或低(0x00010000)两个位置任选其一加载到RAM中,分别称为大映像内核和小映像内核;
4、跳转到setup函数执行;
0x3 setup函数引导内核
这个过程主要是检查和初始化硬件、虽然BIOS完成了相似的大部分工作,但是因为不依赖与BIOS,所以,还是重新初始化了硬件方面的事情;重要的过程有:
1、移动低装载小映像内核的位置到0x00001000去,如果是高装载则不移动;
2、建立IDT(临时中断描述符表)和GDT(临时全局描述符表);
3、如果需要,重置浮点单元(FPU);
4、重新编写可编程终端控制器(PIC),屏蔽除IRQ2外的所有终端;
5、设置cr0寄存器到PE位,设置PG位为0,切换到保护模式,暂未启用分页;
6、跳转到startup_32()函数;
0x4 内核建立阶段
STEP 1 startup_32()函数
主要做的事情如下:
1、初始化段寄存器和一个临时堆栈,并清零eflags寄存器所有为;
2、用0填充_edata 和_end符号标识的内核未初始化数据区;
3、调用decompress_kernel函数解压内核映像;
【低装载的情况解压内容放在0x00100000位置开始的RAM中,高装载的放在这后面的一个临时缓冲区内,解压后的内核就被移动到0x00100000位置】
4、跳转到0x00100000位置开始执行,新的执行点事arch/i386/kernerlhead.s中的另一个startup_32函数。
STEP 2 再战startup_32()函数
这个函数就是init进程(也是pid=0的0号进程)主要做了一下工作:
1、段寄存器初始化为最终值,内核的bss段填写为0;
2、初始化临时内核页表,初始化pg0,使得线性地址一律映射到统一的物理地址上;
3、cr3寄存器保存了页全局目录,并设置cr0的pg位启用分页;
4、清零eflags,使用setup_idt函数用空的终端处理程序填充IDT;
5、从bios获取的数据(系统参数和传递给os的参数)放入页框1;
6、识别处理器、用GDT和IDT填充gdtr和idtr寄存器;
7、跳转到start_kernel函数
0x5 内核完善阶段start_kernel函数
这一阶段最终完善了内核的初始化的后续工作,启动了程序调度、内存管理等操作系统的功能,其中就涉及到了著名的函数sched_init函数,至此,系统完全启动成功。
计算机从加电到系统(Linux)启动完成的更多相关文章
- 第1章 从开机加电到main函数之前的过程
主要讲解了80x86cpu在启动的时候时bios如何工作,以及如何最终转换到保护模式. 1.1 启动bios 80x86作为冯诺依曼结构下的cpu,工作模式也是取指执行,即cpu根据cs:ip寄存器的 ...
- 深入理解Linux启动过程
深入理解Linux启动过程 本文详细分析了Linux桌面操作系统的启动过程,涉及到BIOS系统.LILO 和GRUB引导装载程序,以及bootsect.setup.vmlinux等映像文件 ...
- Linux启动流程详解
在BIOS阶段,计算机的行为基本上被写死了,可以做的事情并不多:一般就是通电.BIOS.主引导记录.操作系统这四步.所以我们一般认为加载内核是linux启动流程的第一步. 第一步.加载内核 操作系统接 ...
- Linux启动流程详解【转载】
在BIOS阶段,计算机的行为基本上被写死了,可以做的事情并不多:一般就是通电.BIOS.主引导记录.操作系统这四步.所以我们一般认为加载内核是linux启动流程的第一步. 第一步.加载内核 操作系统接 ...
- lesson - 2 yum /单用户/救援模式/Linux 启动
课程大纲:1. yum使用yum 是一个在线安装软件包的工具,它可以帮我们解决软件包的依赖,这个日后会详细介绍.我们介绍了以下几个用法:yum list 这个命令可以列出所有安装过和未安装的软 ...
- 详解linux系统的启动过程及系统初始化
一.linux系统的启动流程 关于linux系统的启动流程我们可以按步进行划分为如下: POST加电自检 -->BIOS(Boot Sequence)-->加载对应引导上的MBR(boot ...
- linux内核剖析(零)linux系统启动过程详解-开机加电后发生了什么
本文参考了如下文章 深入理解linux启动过程 mbr (主引导记录(Master Boot Record)) 电脑从开机加电到操作系统main函数之前执行的过程 详解linux系统的启动过程及系统初 ...
- linux系统启动过程具体解释-开机加电后发生了什么 --linux内核剖析(零)
本文參考了例如以下文章 深入理解linux启动过程 mbr (主引导记录(Master Boot Record)) 电脑从开机加电到操作系统main函数之前执行的过程 详细解释linux系统的启动过程 ...
- 计算机/linux启动过程
开机过程指的是从打开计算机电源直到LINUX显示用户登录画面的全过程. 分析LINUX开机过程也是深入了解LINUX核心工作原理的一个很好的途径. 计算机开机过程是一个非常复杂的过程,想真正理解透彻并 ...
随机推荐
- shell脚本重启tomcat
1. 新建shell空脚本文件,如 /home/tr.sh,并设置权限 # chmod 750 /home/tr.sh 2. 设置文件形式: # sed -e 's/\^M//g' /home/tr. ...
- Global.asax的Application_BeginRequest实现url重写无后缀的代码
本文为大家详细介绍下利用Global.asax的Application_BeginRequest 实现url重写其无后缀,具体核心代码如下,有需求的朋友可以参考下,希望对大家有所帮助 利用Global ...
- Java开发者需要学习的移动开发编程语言
编程语言(programming language),是用来定义计算机程序的形式语言.它是一种被标准化的交流技巧,用来向计算机发出指令.一种计算机语言让程序员能够准确地定义计算机所需要使用的数据,并精 ...
- JS去除字符串左右两端的空格
去除字符串左右两端的空格,在vbscript里面可以轻松地使用 trim.ltrim 或 rtrim,但在js中却没有这3个内置方法,需要手工编写.下面的实现方法是用到了正则表达式,效率不错,并把这三 ...
- IE每次关闭都提示IE已停止工作
方法一:打开IE浏览器,点击工具,选择下拉菜单中的internet选项,切换至高级选项卡标签,找到“启用第三方浏览器扩展”选项.把前面的打勾去掉: 方法二:打开IE浏览器,不管是6.0还是更高的版本这 ...
- Python 和C#的交互
IronPython是一个.NET平台上的Python实现,包括了完整的编译器.执行引擎与运行时支持,能够与.NET已有的库无缝整合到一起. IronPython已经很好的集成到了.NET frame ...
- tomcat启动时出现了Failed to start component [StandardEngine[Catalina].StandardHost[localhost]]
https://blog.csdn.net/imjcoder/article/details/78725267 <dependency> <groupId>org.spring ...
- Android Studio 视图解析
AS一共同拥有三种视图.我们来分别分析每一种视图的作用. 一.Project视图.(白色字体的文件夹/文件可不关注) 图片中的链接 Gralde介绍:http://stormzhang.com/dev ...
- struts开发<struts中的參数传递.三>
不说废话,直接上干货 1.通过set和get传递參数 添加username 和password两个属性并添加set和get方法 package fzl.user.struts.demo; import ...
- Bypass X-WAF SQL注入防御(多姿势)
0x00 前言 X-WAF是一款适用中.小企业的云WAF系统,让中.小企业也可以非常方便地拥有自己的免费云WAF. 本文从代码出发,一步步理解WAF的工作原理,多姿势进行WAF Bypass. ...