编写主引导扇区代码
主引导扇区
处理器加电或者复位之后,如果硬盘是首选的启动设备,那么,ROM-BIOS将试图读取硬盘的0面0道1扇区。传统上,这就是主引导扇区(Main Boot Sector,MBR)。
读取主引导扇区数据有512字节,ROM-BIOS程序将它加载到逻辑地址0x0000:0x7c00处,也就是物理地址0x07c00处,然后判断它是否有效。
一个有效的主引导扇区,其最后两个字节应当是0x55和0xAA。ROM-BIOS程序首先检测这两个标志,如果主引导扇区有效,则以一个段间转移指令jmp 0x0000:0x7c00跳到那里继续执行。
在屏幕上显示文字
如何在屏幕上显示字符。我们需要把数据写入到显存,用数据控制每个像素是亮,还是不亮。显存的数据的字节对应着屏幕上连续的8个像素,如果bit是0则不亮,如果是1则点亮对应的像素,要显示更多的颜色一个bit不够了。
如何在屏幕上显示出字符的形状。有这样一个标准,美国信息交换代码标准(American Standard Code for Information Interchange)。用特定的ASCII码代表字符,显卡在任何时候都认为你发送的是ASCII码。
如何在屏幕上某个位置显示出字符。计算机在加电自检之后都会把自己初始化到80×25的文本模式。在这种模式下,屏幕上可以显示25行,每行80个字符,每屏总共2000个字符。考虑到文本模式下现存的起始物理地址是0xB8000,这块内存可以看成是段地址为0xB800,偏移地址从0x0000延伸到0xFFFF的区域,因此我们可以把段地址定为0xB800,通过段偏移定位到想显示的位置。
如何在屏幕上显示带颜色的字符。屏幕上的每个字符对应着显存中的两个连续字节,前一个是字符的ASCII码,后面是字符的显示属性,包括字符颜色(前景色)和底色(背景色)。字符的显示属性(1字节)分为两个不部分,低4位定义的是前景色,高4位定义的是背景色。色彩主要由R,G,B 这三位决定。
代码解析
jmp near start ;跳转到标号代表并指示它所在位置处的汇编地址
text db 'H',0x07,'e',0x07,'l',0x07,'l',0x07,'o',0x07,\
' ',0x07,'W',0x07,'o',0x07,'r',0x07,'l',0x07,'d',0x07
;数据标号不仅仅表示内存单元的地址,还表示了内存单元的长度
;db声明字节,它后面的操作数都占一个字节长度
;\续行符表示下一行和当前行合并为一行
;0x07 黑底白字,无闪烁,无加亮
start: ;标号代表并指示它所在位置处的汇编地址
mov ax,0x7c0
mov ds,ax ;数据段逻辑地址0x7c0为主引导起始地址
mov ax,0xb800
mov es,ax ;显存的逻辑段地址0xb800
cld ;清除方向标志DF置0,以表示正方向传送
mov si,text ;设置数据的地址
mov di,0 ;目的段偏移地址为0
mov cx,(start-text)/2 ;设置批量传送的字节数到CX寄存器,因为数据串是在两个标号之间声明的,
;除以2是每个显示的字符实际占两个字节,而movsw每次传送一个字
rep movsw ;movsw,操作码是0xA5,该指令没有操作数。使用movsw而不是movsb的原因是每次需要传送一个字(ASCII码和属性)。
;单纯的movsb和movsw只能执行一次,如果希望处理器自动的反复执行,需要加上rep(repeat),意思是CX不为0则重复。
times 510-($-$$) db 0 ;伪指令times可用于重复它后面的指令若干次。
;里面的$表示当前指令的地址,$$表示程序的起始地址0x7c00,所以$-$$就等于本条指令前的字节数,
;times 510-($-$$) db 0 为在填充一些数据共为510个字节,
db 0x55,0xaa ;结束表示,整个段程序的大小正好为510字节,占满一个扇区。