上次已经学到了 CPU为访问内存做好了准备,把一些重要寄存器的值都设置好了。
今天我们主要一起学习,操作系统是如何把自己从硬盘给弄到内存里的?
现在的状态已经把ds,es,cs设置好了,从CPU的角度看,已经知道如何访问内存了。
我们接着看下面的代码
load_setup:
mov $0x0000, %dx # drive 0, head 0
mov $0x0002, %cx # sector 2, track 0
mov $0x0200, %bx # address = 512, in INITSEG
.equ AX, 0x0200+SETUPLEN
mov $AX, %ax # service 2, nr of sectors
int $0x13 # read it
jnc ok_load_setup # ok - continue
mov $0x0000, %dx
mov $0x0000, %ax # reset the diskette
int $0x13
jmp load_setup
这里除了mov 和 jmp,还有一种新的指令:int。 这个int 指令是代表中断, int 0x13 就是 发起0x13号中断。
中断之前 先把ax,bx,cx,dx 这四个寄存器都赋上值了。 因为中断程序需要这四个寄存器作为参数。 还有另一种传参方式,通过栈来传递参数。
所谓中断,就是打断原有的指令执行的顺序,让我们去处理这个中断信息,上面这个案例是13号中断,那13号中断程序的段地址和偏移量怎么获得呢?又需要到中断向量表里来查找。
具体可以查看 《汇编语言》
好,简单介绍了中断后,我们继续回到代码上来看。 发起13号中断后,就会去中断向量表找到13号中断程序的地址,然后跳转过去执行,感觉有点像执行一个函数。
13号中断程序是BIOS写好的,就是读取硬盘的函数。随着我们阅读越来越多的代码,也会接触到更多的中断程序,各模块会注册自己的中断处理程序。
结合注释看出这段代码的作用: 从硬盘的第2个扇区开始,把数据加载到内存的0x90200处(正好跳过 上一节说的那512个字节),加载4个扇区。
大致如下图所示:
这里可能是加载古老的软盘的逻辑,因为软盘早已消失在历史的长河里了,所以就当作硬盘来看吧!
再往下看就是2个跳转指令jnc和jmp。 成功了就跳转到 ok_load_setup 去执行。 失败的话就继续跳转到 load_setup 标签执行,就是失败重试。
那我们接着看成功的逻辑
.equ SETUPSEG, 0x9020 # setup starts here
.equ SYSSEG, 0x1000 # system loaded at 0x10000 (65536).
ok_load_setup:
……
mov $SYSSEG, %ax
mov %ax, %es # segment of 0x010000
call read_it
call kill_motor
……
ljmp $SETUPSEG, $0
这里只留下了几行核心逻辑的代码。 就是从 把硬盘的第6个扇区到240个扇区,加载到内存0x10000处。 和上面从硬盘复制是一样的。
等操作系统的代码都加载到内存中后, 又通过ljmp 0x9020,0 跳转到0x90200处,也就是第二扇区开始的地方。
这里的内容就是 setup.s 这个文件了。
在看setup.s之前,先回顾下操作系统的编译过程: 通过Makefile和build.c达到这样的目的: