手写操作系统 -- 微内核

2023-12-29 12:12

本文主要是介绍手写操作系统 -- 微内核,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

搭建操作系统框架,输出hello world

知识点:

  • CPU的两种运行模式:实模式、保护模式(普遍)
    • 操作系统或应用程序运行中用到的都是真实的物理地址,这种场景叫做实模式运行;我们的操作系统刚开机的时候就是运行在实模式下;我们自己写的操作系统也是,先在实模式下运行,载入内核,然后进入保护模式
  • 如何向屏幕输出内容?
  • BIOS中断?
    • 在实模式下,我们写的OS内核想要访问硬件,就必须借助BIOS中断

实现微内核

运行起来一个简单的框架

重点理解:

  • 汇编脚本怎么转换成内核?
  • 内核存放在硬盘什么位置?
    • 0柱面0磁道1扇区(512字节 0x200
    • 硬盘哪里来的?
    • 怎么写进硬盘?
  • 内核如何运行起来?
    • BIOS例程读取0柱面0磁道1扇区
    • 读到0x7c00(为什么?)
    • 执行权限如何给微内核?

在这里插入图片描述

具体实现:

  1. 创建一个内核项目编写boot.asm脚本(使用clion)
;基本框架
[ORG  0x7c00]   ;指定想对位置 物理地址[SECTION .text]  ;段
[BITS 16]     	;指定16位寄存器
global _start   ;声明全全局的开始 _start可以换成别的,但一般这么写
_start:xchg bx, bx  ;断点调试jmp $times 510 - ($ - $$) db 0   ;times指定重复次数 $: 表示当前地址 $$: 表示当前节(section)的开始地址 
db 0x55, 0xaa   ;启动扇区的标准签名,表示这是一个有效的启动扇区

(1)使用bximage -q -hd=16 -func=create -sectsize=512 -imgmode=flat hd.img 命令生成hd.img硬盘

`bximage` 是一个用于创建硬盘或软盘映像文件的工具,该工具常用于`Bochs`模拟器。这个模拟器主要用于开发操作系统或运行旧的操作系统。让我们分解你给出的命令:1. **bximage**: 这是调用该工具的命令名。2. **-q**: 这个选项使`bximage`在“快速模式”下运行,意思是它不会询问任何交互式问题,只会基于提供的参数来执行。3. **-hd=16**: 指定创建的映像文件的大小为16MB。这里的`-hd`表示硬盘映像。4. **-func=create**: 指定你要执行的功能是创建一个新的映像文件。5. **-sectsize=512**: 设置每个扇区的大小为512字节。这是一个常见的扇区大小。6. **-imgmode=flat**: 这指定映像文件的模式为`flat`。在这种模式下,映像文件就是一个简单的连续字节流,没有任何特殊的结构或格式。7. **hd.img**: 这是你要创建的映像文件的文件名。总结一下:这个命令告诉`bximage`快速地创建一个16MB的硬盘映像文件,名为`hd.img`,每个扇区大小为512字节,映像模式为`flat`。

(2) 使用nasm boot.asm 命令生成boot.o文件;内存分布如下:

在这里插入图片描述

(3)使用 "dd if=boot of=hd.img bs=512 seek=0 count=1 conv=notrunc"内核程序(boot)写入硬盘(hd.img)当中;内存图如下:

在这里插入图片描述

![在这里插入图片描述](https://img-blog.csdnimg.cn/af037a79425b4ba8aeaa596eb549af04.png)`dd` 是一个 Unix/Linux 下的命令行工具,用于在文件之间或者从文件和设备之间复制数据。它可以用于各种数据转换和复制任务,经常用于制作启动盘或拷贝整个硬盘或分区的数据。 让我们详细解释你给出的命令:1. **dd**: 调用这个命令。2. **if=boot**: 设置输入文件为`boot`。`if`代表 "input file"。3. **of=hd.img**: 设置输出文件为`hd.img`。`of`代表 "output file"。4. **bs=512**: 设置块大小为512字节。`bs`代表 "block size"。这意味着`dd`每次读取和写入512字节。5. **seek=0**: 开始从输出文件的开始位置写入。`seek`用于指定输出文件中的起始位置。6. **count=1**: 只复制一个块。因为`bs`是512字节,所以这个命令只会复制512字节。7. **conv=notrunc**: 不截断输出文件。`conv`用于指定一些转换操作。`notrunc`意味着它不会截断(或者缩短)输出文件。如果输出文件已经存在并且比要复制的数据大,它不会被缩短,只会在指定的位置被新数据覆盖。综上所述,这个命令的作用是将`boot`文件的前512字节复制到`hd.img`文件的开始位置,而不会截断或者改变`hd.img`文件的其他部分。

(4)利用 bochsqemu 来运行

将命令汇总位makefile文件:

# 汇总命令执行
BUILD:=./buildHD_IMG_NAME:= "hd.img"all: ${BUILD}/boot/boot.o$(shell rm -rf $(HD_IMG_NAME))bximage -q -hd=16 -func=create -sectsize=512 -imgmode=flat $(HD_IMG_NAME)dd if=${BUILD}/boot/boot.o of=hd.img bs=512 seek=0 count=1 conv=notrunc${BUILD}/boot/%.o: %.asm$(shell mkdir -p ${BUILD}/boot)nasm $< -o $@clean:$(shell rm -rf ${BUILD})bochs: allbochs -q -f bochsrcqemu: allqemu-system-x86_64 -hda hd.img

运行起来的样子:

在这里插入图片描述

实战BIOS中断之屏幕输出

;基本框架
[ORG  0x7c00]   ;指定想对位置 物理地址[SECTION .text]  ;段
[BITS 16]     	;指定16位寄存器
global _start   ;声明全全局的开始 _start可以换成别的,但一般这么写
_start:xchg bx, bx  ;断点调试jmp $times 510 - ($ - $$) db 0   ;times指定重复次数 $: 表示当前地址 $$: 表示当前节(section)的开始地址 
db 0x55, 0xaa   ;启动扇区的标准签名,表示这是一个有效的启动扇区

输出hello world

[ORG  0x7c00][SECTION .text]
[BITS 16]
global _start
_start:; 设置屏幕模式为文本模式,清除屏幕mov ax, 3int 0x10mov     si, msgcall    printjmp $; 如何调用
; mov     si, msg   ; 1 传入字符串
; call    print     ; 2 调用
print:mov ah, 0x0emov bh, 0mov bl, 0x01
.loop:mov al, [si]cmp al, 0jz .doneint 0x10  ;利用中断的方式打印出来inc sijmp .loop
.done:retmsg:db "hello, world", 10, 13, 0times 510 - ($ - $$) db 0
db 0x55, 0xaa;

详细理解:

  • 内核如何运行起来?
    • BIOS例程读取0柱面0磁道1扇区
    • 读到0x7c00(为什么?)
    • 执行权限如何给微内核?
      在这里插入图片描述
  1. 实时模式下,Intel的早期微处理器,例如8086,它确实有20位的地址总线,这允许它访问1MB的物理内存(2^20 = 1,048,576字节或1MB);
  1. 考虑哪些内存可以用哪些不能用,参考实时模式下内存布局图。
    在这里插入图片描述

3.BIOS启动过程
在实时模式下的内存布局中可以看到在最后16字节的内容时跳转指令jmp f000:e05b
实时模式下寻址方式为 段:段内偏移,所有f000 << 4 + e05b = fe05b (真实物理地址)
跳转到fe05b去执行BIOS启动过程
在这里插入图片描述
在这里插入图片描述

BIOS(Basic Input/Output System)是计算机启动时首先运行的固件。BIOS启动过程是相对固定的,负责初始化和测试系统硬件,然后加载并执行引导加载程序来启动操作系统。以下是BIOS启动过程的简要概述:1. **上电自检(POST,Power-On Self Test)**:当计算机上电或重启时,BIOS首先执行POST。这是一系列的硬件测试,确保关键组件(如RAM、CPU、键盘和其他设备)功能正常。2. **硬件初始化**:BIOS将初始化系统的所有硬件设备,包括内存控制器、显示卡、硬盘、光驱等。3. **设备检查**:BIOS检查连接到计算机的所有设备,例如键盘、鼠标、硬盘、CD/DVD驱动器等。4. **启动顺序**:根据BIOS设置中定义的启动顺序,BIOS会尝试从各种设备上加载操作系统。例如,通常的启动顺序可能是:光驱、USB驱动器、硬盘。但这可以在BIOS设置中自定义。5. **引导扇区加载**:当BIOS找到一个有效的启动设备后,它会加载该设备的第一个扇区——称为引导扇区。引导扇区包含引导加载程序的代码,它的任务是开始加载操作系统。6.**转交控制权限**:一旦引导扇区的内容被加载到内存中,BIOS就将控制权转交给这段代码。这通常发生在物理内存的0x7C00地址处,因为这是BIOS加载引导扇区内容的标准位置。7. **引导加载程序执行**:引导加载程序接管后,它会继续加载操作系统的其余部分。在某些系统中,例如使用GRUB引导加载程序的Linux系统,用户可能会看到一个菜单,允许他们选择不同的操作系统或配置。8. **操作系统加载**:引导加载程序加载并执行操作系统。操作系统接管后,它将完成其自己的初始化并最终显示用户界面,等待用户输入。这只是BIOS启动过程的高级概述。具体的步骤和细节可能因计算机硬件和配置的不同而异。此外,现代计算机上也有一个BIOS的替代者,称为UEFI(统一的可扩展固件接口),它提供了更多的功能和安全性,并支持更大的硬盘驱动器和现代操作系统。

总结

  • 创建一个微内核项目,编写boot.asm
    • 使用nasm boot.asm 命令生成boot.o文件。
  • 如何生成一个硬盘?
    • 使用bximage -q -hd=16 -func=create -sectsize=512 -imgmode=flat hd.img 命令生成。hd.img硬盘
  • 如何将内核写入硬盘?
    • 使用 "dd if=boot of=hd.img bs=512 seek=0 count=1 conv=notrunc"内核程序(boot)写入硬盘(hd.img)当中。
  • 如何运行起来?
    • 利用 bochsqemu 来运行 。
  • 为什么要读到0x7c00位置?
    • 查看实时实时模式下内的存布局图可以发现0x7c00BIOS例程的入口。
  • 为什么要调到0xf000:e05b处?
    • BIOS例程的内存地址

这篇关于手写操作系统 -- 微内核的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/549408

相关文章

高效管理你的Linux系统: Debian操作系统常用命令指南

《高效管理你的Linux系统:Debian操作系统常用命令指南》在Debian操作系统中,了解和掌握常用命令对于提高工作效率和系统管理至关重要,本文将详细介绍Debian的常用命令,帮助读者更好地使... Debian是一个流行的linux发行版,它以其稳定性、强大的软件包管理和丰富的社区资源而闻名。在使用

龙蜥操作系统Anolis OS-23.x安装配置图解教程(保姆级)

《龙蜥操作系统AnolisOS-23.x安装配置图解教程(保姆级)》:本文主要介绍了安装和配置AnolisOS23.2系统,包括分区、软件选择、设置root密码、网络配置、主机名设置和禁用SELinux的步骤,详细内容请阅读本文,希望能对你有所帮助... ‌AnolisOS‌是由阿里云推出的开源操作系统,旨

五大特性引领创新! 深度操作系统 deepin 25 Preview预览版发布

《五大特性引领创新!深度操作系统deepin25Preview预览版发布》今日,深度操作系统正式推出deepin25Preview版本,该版本集成了五大核心特性:磐石系统、全新DDE、Tr... 深度操作系统今日发布了 deepin 25 Preview,新版本囊括五大特性:磐石系统、全新 DDE、Tree

Linux操作系统 初识

在认识操作系统之前,我们首先来了解一下计算机的发展: 计算机的发展 世界上第一台计算机名叫埃尼阿克,诞生在1945年2月14日,用于军事用途。 后来因为计算机的优势和潜力巨大,计算机开始飞速发展,并产生了一个当时一直有效的定律:摩尔定律--当价格不变时,集成电路上可容纳的元器件的数目,约每隔18-24个月便会增加一倍,性能也将提升一倍。 那么相应的,计算机就会变得越来越快,越来越小型化。

stl的sort和手写快排的运行效率哪个比较高?

STL的sort必然要比你自己写的快排要快,因为你自己手写一个这么复杂的sort,那就太闲了。STL的sort是尽量让复杂度维持在O(N log N)的,因此就有了各种的Hybrid sort algorithm。 题主你提到的先quicksort到一定深度之后就转为heapsort,这种是introsort。 每种STL实现使用的算法各有不同,GNU Standard C++ Lib

1、简述linux操作系统启动流程

1、简述linux操作系统启动流程 启动第一步--加载BIOS 当你打开计算机电源,计算机会首先加载BIOS信息,BIOS信息是如此的重要,以至于计算机必须在最开始就找到它。这是因为BIOS中包含了CPU的相关信息、设备启动顺序信息、硬盘信息、内存信息、时钟信息、PnP特性等等。开机时将ROM中的指令映射到RAM的低地址空间,CPU读取到这些指令,硬件的健康状况进行检查,按照BIOS中设置的启

JS手写实现深拷贝

手写深拷贝 一、通过JSON.stringify二、函数库lodash三、递归实现深拷贝基础递归升级版递归---解决环引用爆栈问题最终版递归---解决其余类型拷贝结果 一、通过JSON.stringify JSON.parse(JSON.stringify(obj))是比较常用的深拷贝方法之一 原理:利用JSON.stringify 将JavaScript对象序列化成为JSO

操作系统是怎么为不同的程序分配所需的内存空间的

操作系统为不同的程序分配内存空间的过程涉及多个关键步骤,确保每个程序都有其所需的内存资源,同时避免程序之间的冲突。以下是操作系统如何为程序分配内存空间的详细过程: 1. 内存管理的基础概念 虚拟内存:现代操作系统使用虚拟内存机制来为程序提供隔离的内存空间。每个程序运行在其独立的虚拟地址空间中,这使得程序间的内存互不干扰。物理内存:实际的 RAM(随机存取存储器),由操作系统和硬件共同管理。虚拟

T1打卡——mnist手写数字识别

🍨 本文为🔗365天深度学习训练营中的学习记录博客🍖 原作者:K同学啊 1.定义GPU import tensorflow as tfgpus=tf.config.list_physical_devices("GPU")if gpus:gpu0=gpus[0]tf.config.experimental.set_memort_groth(gpu0,True) #设置GPU现存用量按需

操作系统安全保护

操作系统安全概述 概念:满足安全策略要求,具有响应安全机制及安全功符合特定安全标准,在一定约束条件下 能抵御常见网络安全威胁,保障自身安全运行及资源安全 安全等级:根据安全功能和安全保障要求分为 用户自主保护级  系统审计保护级 安全标记保护级 结构化保护级 访问验证保护级 操作系统作用: 负责计算系统的资源管理、支撑和控制各种应用程序运行,为用户提供计算机系统管理接口 是构成网络信息