写一个 MBR 程序
我们先来贴出程序,然后再来解释。如下这段代码的功能是在屏幕上打印字符串 “1 MBR”。背景色为黑色,前景色为绿色。
1 | ;主引导程序 |
1. section 相关
section 是伪指令,是 nasm 提供的。CPU 运行程序是不需要这个东西的,section 只是用来给程序员规划程序用的,逻辑上划分成段的好处就是方便开发人员梳理代码,方便管理。
如果没有定义 section,nasm 默认全部代码同为一个 section,起始位置为 0。
$
表示本行代码的地址,$$
表示本 section 的起始地址。这两个都是 nasm 预留的关键字,是编译器安排的。
vstart=0x7c00
表示在编译时,把起始地址编译为 0x7c00
。
使用 nasm 进行编译这个汇编文件:nasm -o mbr.bin mbr.s
使用 bin/bximage
来创建一块虚拟硬盘。然后使用 dd
工具将 mbr.bin
写入这块硬盘中。
bximage 命令创建硬盘:bximage -hd -mode="flat" -size=60 -q a.img
如下是 dd 命令的选项
1 | if=FILE // 指定要读取的文件 |
vstart=xxx
和 org xxx
都是告诉编译器将程序编译到 xxx 地址。
cpu 的工作原理总结如下:
控制单元要取下一条待运行的指令,该指令的地址在程序计数器 PC 中,在 x86 CPU 上,程序计数器就是 cs:ip
。于是读取 ip 寄存器后,将此地址送上地址总线,CPU 根据此地址便得到了指令,并将其存入到指令寄存器 IR 中。这时候轮到指令译码器,根据指令格式检查指令寄存器中的指令,先确定操作码是什么,再检查操作数类型,若是在内存中,就将相应操作数从内存中取回放入自己的存储单元,若操作数是在寄存器中就直接用了。此时,操作码有了,操作数也齐了,操作控制器给运算单元下令,开工,于是运算单元便真正开始执行指令了。ip 寄存器的值被加上当前指令的大小,于是 ip 又指向了下一条指令的地址。接着控制单元又要取下一条指令了,流程回到了本段开头,CPU 便开始了日复一日的循环,由于 CPU 特别不容易坏,所以唯一它停下来的条件就是断电。
