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. 收集整理的240多个jQuey插件 .

    “jQuery 是继 prototype 之后又一个优秀的 Javascrīpt 框架.其宗旨是——WRITE LESS,DO MORE,写更少的代码,做更多的事情.它是轻量级的 js 库(压缩后只有 ...

  2. AsyncTask异步类的简单操作

    package com.example.day9; import org.apache.http.HttpEntity; import org.apache.http.HttpResponse; im ...

  3. 【WP8】扩展CM的INavigationService方法

    CM支持通过ViewModel进行导航,并通过支持参数传递,但是内部只是通过反射的方式构造Uri的参数进行导航,所以只支持简单类型的参数传递,下面对其进行扩展,在页面导航时支持复杂类型的参数传递,并扩 ...

  4. saltstack系列1之salt-api配置与使用

    salt-api salt-api是我们通过restful-api调用salt-master的接口,且调用的时候必须通过认证才能调用,认证的用户为系统用户,下面就说说如何配置salt-api. 不带S ...

  5. Laravel查询构造器简介

    数据表 CREATE TABLE IF NOT EXISTS students( `id` INT AUTO_INCREMENT PRIMARY KEY, `name` VARCHAR(255) NO ...

  6. SpringBoot------新建一个项目

    1.新建一个Maven Project 2.选择项目路劲 3.选择Web App 4.添加项目名称 5.右击“Properties”->“Java Build Path”->“Source ...

  7. Java使用for循环输出杨辉三角

    杨辉三角形由数字排列,可以把它看做一个数字表,其基本特性是两侧数值均为1,其他位置的数值是其正上方的数字与左上角数值之和.编写程序,使用for循环输出包括10行在内的杨辉三角形. 思路是创建一个整型二 ...

  8. 手机APP支付--整合支付宝支付控件

    长话短说,本文根据支付宝官方说明文档,简单总结下,并且说明下开发过程碰到的问题以及该如何解决. 整合步骤: 1 登录商家服务网站,下载开发包,地址:https://b.alipay.com/order ...

  9. Go之对象拷贝

    这里interface{}就相当于c#,java中的object, boy := util.Boy{util.Person{"Eric", 19, "boy"} ...

  10. unity2017.4.0f1使用AS3.0的AndroidSDK遇到的问题

    原因: Unity 在编译时会调用 Android SDK tools 中的 android 命令,而在新版本的 Android SDK tools 中,android这个命令已经废弃了,导致 Uni ...