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. 第1章 从开机加电到main函数之前的过程

    主要讲解了80x86cpu在启动的时候时bios如何工作,以及如何最终转换到保护模式. 1.1 启动bios 80x86作为冯诺依曼结构下的cpu,工作模式也是取指执行,即cpu根据cs:ip寄存器的 ...

  2. 深入理解Linux启动过程

    深入理解Linux启动过程       本文详细分析了Linux桌面操作系统的启动过程,涉及到BIOS系统.LILO 和GRUB引导装载程序,以及bootsect.setup.vmlinux等映像文件 ...

  3. Linux启动流程详解

    在BIOS阶段,计算机的行为基本上被写死了,可以做的事情并不多:一般就是通电.BIOS.主引导记录.操作系统这四步.所以我们一般认为加载内核是linux启动流程的第一步. 第一步.加载内核 操作系统接 ...

  4. Linux启动流程详解【转载】

    在BIOS阶段,计算机的行为基本上被写死了,可以做的事情并不多:一般就是通电.BIOS.主引导记录.操作系统这四步.所以我们一般认为加载内核是linux启动流程的第一步. 第一步.加载内核 操作系统接 ...

  5. lesson - 2 yum /单用户/救援模式/Linux 启动

    课程大纲:1.  yum使用yum 是一个在线安装软件包的工具,它可以帮我们解决软件包的依赖,这个日后会详细介绍.我们介绍了以下几个用法:yum list    这个命令可以列出所有安装过和未安装的软 ...

  6. 详解linux系统的启动过程及系统初始化

    一.linux系统的启动流程 关于linux系统的启动流程我们可以按步进行划分为如下: POST加电自检 -->BIOS(Boot Sequence)-->加载对应引导上的MBR(boot ...

  7. linux内核剖析(零)linux系统启动过程详解-开机加电后发生了什么

    本文参考了如下文章 深入理解linux启动过程 mbr (主引导记录(Master Boot Record)) 电脑从开机加电到操作系统main函数之前执行的过程 详解linux系统的启动过程及系统初 ...

  8. linux系统启动过程具体解释-开机加电后发生了什么 --linux内核剖析(零)

    本文參考了例如以下文章 深入理解linux启动过程 mbr (主引导记录(Master Boot Record)) 电脑从开机加电到操作系统main函数之前执行的过程 详细解释linux系统的启动过程 ...

  9. 计算机/linux启动过程

    开机过程指的是从打开计算机电源直到LINUX显示用户登录画面的全过程. 分析LINUX开机过程也是深入了解LINUX核心工作原理的一个很好的途径. 计算机开机过程是一个非常复杂的过程,想真正理解透彻并 ...

随机推荐

  1. iOS :UIPickerView reloadAllComponets not work

    编辑信息页面用了很多选择栏,大部分都用 UIPickerView 来实现.在切换数据显示的时候, UIPickerView 不更新数据,不得其解.Google 无解,原因在于无法描述自己的问题,想想应 ...

  2. NFS介绍 NFS服务端安装配置 NFS配置选项

    NFS 介绍 • NFS是Network File System的缩写 • NFS最早由Sun公司开发,分2,,4三个版本,2和3由Sun起草开发,.0开始Netapp公司参与并主导开发,最新为4.1 ...

  3. 深入浅出:全面理解SQL Server权限体系

    转自IT168  好文转载存档! [IT168 技术]权限两个字,一个权力,一个限制.在软件领域通俗的解释就是哪些人可以对哪些资源做哪些操作.在SQL Server中,"哪些人", ...

  4. 安卓开发笔记——GridView组件

    1.什么是GridView? GridView(网格视图)是按照行列的方式来显示内容的,一般用于显示图片,图片等内容,比如实现九宫格图,用GridView是首选,也是最简单的. 2.正文 GridVi ...

  5. java用substring函数截取string中一段字符串

    在String中有两个substring()函数,如下: 一:String.substring(int start) 参数: start:要截取位置的索引 返回: 从start开始到结束的字符串 例如 ...

  6. linux环境中iostat命令的安装,解决-bash: iostat: command not found问题

    需求说明: 今天在测试环境的主机上,准备通过iostat来查看系统的io情况,发现没有该命令 [root@testvm Packages]# iostat -bash: iostat: command ...

  7. iOS 严重问题解释(crash)

    问题1:Exception Type: 00000020 Exception Codes: 0x000000008badf00d Exception Note: SIMULATED (this is  ...

  8. 转换python脚本为可执行程序的方式

    背景: 部分工具使用python脚本编写,而目标服务器,没有安装python包,导致使用工具不方便,还需要另外安装python. 目前主要有2个主流软件,可做此类转换,把对应工具脚本转换为exe: p ...

  9. SpringBoot(九)-- SpringBoot JDBC

    1.属性配置文件(application.properties) # type 可以修改连接池类型,默认采用Tomcat的连接池 # spring.datasource.type=com.alibab ...

  10. leetCode练习题

    1.求二叉树的最小深度: public class Solution { public int run(TreeNode root) { if(root==null) return 0; int l ...