李忠老师的《x86汇编语言:从实模式到保护模式》中第五章到第七章的部分,每一章在讲述知识点的同时,分别使用了三种不同的显示字符的方法,加上调用BIOS的10h中 断的方法,这里做出一次简单梳理:
一:第五章,最基础的直接用mov 的方法
代码如下:
1 ;代码清单5-1 2 ;文件名:c05_mbr.asm 3 ;文件说明:硬盘主引导扇区代码 4 ;创建日期:2011-3-31 21:15 5 6 mov ax,0xb800 ;指向文本模式的显示缓冲区 7 mov es,ax 8 9 ;以下显示字符串"Label offset:" 10 mov byte [es:0x00],'L' 11 mov byte [es:0x01],0x07 12 mov byte [es:0x02],'a' 13 mov byte [es:0x03],0x07 14 mov byte [es:0x04],'b' 15 mov byte [es:0x05],0x07 16 mov byte [es:0x06],'e' 17 mov byte [es:0x07],0x07 18 mov byte [es:0x08],'l' 19 mov byte [es:0x09],0x07 20 mov byte [es:0x0a],' ' 21 mov byte [es:0x0b],0x07 22 mov byte [es:0x0c],"o" 23 mov byte [es:0x0d],0x07 24 mov byte [es:0x0e],'f' 25 mov byte [es:0x0f],0x07 26 mov byte [es:0x10],'f' 27 mov byte [es:0x11],0x07 28 mov byte [es:0x12],'s' 29 mov byte [es:0x13],0x07 30 mov byte [es:0x14],'e' 31 mov byte [es:0x15],0x07 32 mov byte [es:0x16],'t' 33 mov byte [es:0x17],0x07 34 mov byte [es:0x18],':' 35 mov byte [es:0x19],0x07 36 37 mov ax,number ;取得标号number的偏移地址 38 mov bx,10 39 40 ;设置数据段的基地址 41 mov cx,cs 42 mov ds,cx 43 44 ;求个位上的数字 45 mov dx,0 46 div bx 47 mov [0x7c00+number+0x00],dl ;保存个位上的数字 48 49 ;求十位上的数字 50 xor dx,dx 51 div bx 52 mov [0x7c00+number+0x01],dl ;保存十位上的数字 53 54 ;求百位上的数字 55 xor dx,dx 56 div bx 57 mov [0x7c00+number+0x02],dl ;保存百位上的数字 58 59 ;求千位上的数字 60 xor dx,dx 61 div bx 62 mov [0x7c00+number+0x03],dl ;保存千位上的数字 63 64 ;求万位上的数字 65 xor dx,dx 66 div bx 67 mov [0x7c00+number+0x04],dl ;保存万位上的数字 68 69 ;以下用十进制显示标号的偏移地址 70 mov al,[0x7c00+number+0x04] 71 add al,0x30 72 mov [es:0x1a],al 73 mov byte [es:0x1b],0x04 74 75 mov al,[0x7c00+number+0x03] 76 add al,0x30 77 mov [es:0x1c],al 78 mov byte [es:0x1d],0x04 79 80 mov al,[0x7c00+number+0x02] 81 add al,0x30 82 mov [es:0x1e],al 83 mov byte [es:0x1f],0x04 84 85 mov al,[0x7c00+number+0x01] 86 add al,0x30 87 mov [es:0x20],al 88 mov byte [es:0x21],0x04 89 90 mov al,[0x7c00+number+0x00] 91 add al,0x30 92 mov [es:0x22],al 93 mov byte [es:0x23],0x04 94 95 mov byte [es:0x24],'D' 96 mov byte [es:0x25],0x07 97 98 infi: jmp near infi ;无限循环 99 100 number db 0,0,0,0,0 101 102 times 203 db 0 103 db 0x55,0xaa
这里采用的最基础的做法,就是对字符进行一个一个的处理。先将显示缓存区的地址0xb800赋给es寄存器,然后通过 mov byte[es:0x00],'L' 的形式,来处理后续的字符。这种方法较为简单,这里不再赘述。
二:第六章,采用了批量处理的方法
代码如下:
1 ;代码清单6-1 2 ;文件名:c06_mbr.asm 3 ;文件说明:硬盘主引导扇区代码 4 ;创建日期:2011-4-12 22:12 5 6 jmp near start 7 8 mytext db 'L',0x07,'a',0x07,'b',0x07,'e',0x07,'l',0x07,' ',0x07,'o',0x07, 9 'f',0x07,'f',0x07,'s',0x07,'e',0x07,'t',0x07,':',0x07 10 number db 0,0,0,0,0 11 12 start: 13 mov ax,0x7c0 ;设置数据段基地址 14 mov ds,ax 15 16 mov ax,0xb800 ;设置附加段基地址 17 mov es,ax 18 19 cld 20 mov si,mytext 21 mov di,0 22 mov cx,(number-mytext)/2 ;实际上等于 13 23 rep movsw 24 25 ;得到标号所代表的偏移地址 26 mov ax,number 27 28 ;计算各个数位 29 mov bx,ax 30 mov cx,5 ;循环次数 31 mov si,10 ;除数 32 digit: 33 xor dx,dx 34 div si 35 mov [bx],dl ;保存数位 36 inc bx 37 loop digit 38 39 ;显示各个数位 40 mov bx,number 41 mov si,4 42 show: 43 mov al,[bx+si] 44 add al,0x30 45 mov ah,0x04 46 mov [es:di],ax 47 add di,2 48 dec si 49 jns show 50 51 mov word [es:di],0x0744 52 53 jmp near $ 54 55 times 510-($-$$) db 0 56 db 0x55,0xaa
这里采用的办法是批量传送,后续用loop循环挨个处理。这样的写法明显比上一种写法要高明一些,减少了工作量。这段代码中值得注意的地方是 mov si,mytext (其中mytext是声明的字符的地址),这里值得留意的原因之一是在做显示时间的编码中,有过下列这样的写法,所以会格外的留心。
1 org 7c00h 2 start1: 3 4 5 mov ax, cs ; 置其他段寄存器值与CS相同 6 mov ds, ax ; 数据段 7 mov es, ax 8 9 mov bl, 10h 10 mov bp, Message1 11 12 mov ah, 02h 13 int 1ah 14 15 xor ax, ax 16 mov al, ch 17 div bl 18 add al, 0x30 19 mov [es:bp+2], al 20 add ah, 0x30 21 mov [es:bp+3], ah 22 23 xor ax, ax 24 mov al, cl 25 div bl 26 add al, 0x30 27 mov [es:bp+5], al 28 add ah, 0x30 29 mov [es:bp+6], ah 30 31 xor ax, ax 32 mov al, dh 33 div bl 34 add al, 0x30 35 mov [es:bp+8], al 36 add ah, 0x30 37 mov [es:bp+9], ah 38 39 mov dh, 3 40 mov dl, 0 41 mov ax, 1301h ; 功能号 42 mov bp, Message1 43 mov cx, MessageLength1 44 mov bx, 0007h 45 int 10h 46 47 ; ret 48 49 Message1: 50 db ' 00:00:00' 51 MessageLength1 equ ($-Message1) 52 53 times 510-($-$$) db 0 ; 用0填充引导扇区剩下的空间 54 db 55h, 0aah ; 引导扇区结束标志
(上面的那段代码的功能是调用BIOS中断显示系统时间)这段代码中对于“00:00:00”的处理方法,代码二中批量处理si处的mytext字段有异曲同工之妙,这里mark一下。
关于代码二中显示数字的方法,是用到了loop循环。先将数字按照“除以10”的方法得到每一位的值,然后将其加上0x30(有关ASCII的知识可解释这一点是为什么),然后将最终值赋予 依次递增的显存地址对应的内容,直到将之前处理的每一位数字都显示出来,over.
三:第七章,使用栈来操作
这一章的代码的特殊之处在于通过将字符串按照一个一个的顺序分别取到之后,将其按照顺序压栈,然后再依次出栈再处理而显示。
1 ;代码清单7-1 2 jmp near start 3 4 message db '1+2+3+...+100=' 5 6 start: 7 mov ax,0x7c0 ;设置数据段的段基地址 8 mov ds,ax 9 10 mov ax,0xb800 ;设置附加段基址到显示缓冲区 11 mov es,ax 12 13 ;以下显示字符串 14 mov si,message 15 mov di,0 16 mov cx,start-message 17 @g: 18 mov al,[si] 19 mov [es:di],al 20 inc di 21 mov byte [es:di],0x07 22 inc di 23 inc si 24 loop @g 25 26 ;以下计算1到100的和 27 xor ax,ax 28 mov cx,1 29 @f: 30 add ax,cx 31 inc cx 32 cmp cx,100 33 jle @f 34 35 ;以下计算累加和的每个数位 36 xor cx,cx ;设置堆栈段的段基地址 37 mov ss,cx 38 mov sp,cx 39 40 mov bx,10 41 xor cx,cx 42 @d: 43 inc cx 44 xor dx,dx 45 div bx 46 or dl,0x30 47 push dx 48 cmp ax,0 49 jne @d 50 51 ;以下显示各个数位 52 @a: 53 pop dx 54 mov [es:di],dl 55 inc di 56 mov byte [es:di],0x07 57 inc di 58 loop @a 59 60 jmp near $ 61 62 63 times 510-($-$$) db 0 64 db 0x55,0xaa
对于代码段四,第一部分显示“1+2+3+4+...+100=”的部分是沿用了上面的代码二中的做法,使用loop循环处理。
而下面处理数字的部分,是一种新的处理方式。这里是将数字依次“除以10”得到每一位的数之后,将其加上0x00(原因:ASCII显示字符需要)压入栈中,然后在下一个循环中,依次出栈并且处理使得其能够显示出来。
四:调用BIOS的10h中断来显示字符
以上,无论是最简单的mov的做法,还是movbw的做法,异或压栈出栈的做法,都难免分别处理每一个字符的圈子。这里介绍一种调用BIOS中断的做法,直接处理一串字符串,较为简单,可参考性高。
1 org 07c00h ; 告诉编译器程序加载到 7c00处 2 mov ax, cs 3 mov ds, ax 4 mov es, ax 5 call DispStr ; 调用显示字符串例程 6 jmp $ ; 无限循环 7 8 DispStr: 9 mov ax, BootMessage 10 mov bp, ax ; es:bp = 串地址 11 mov cx, 16 ; cx = 串长度 12 mov ax, 01301h ; ah = 13, al = 01h 13 mov bx, 000ch ; 页号为 0(bh = 0) 黑底红字(bl = 0Ch,高亮) 14 mov dl, 0 15 int 10h ; 10h 号中断 16 ret 17 18 BootMessage: 19 db "Hello, OS world!" 20 times 510-($-$$) db 0 ; 填充剩下的空间,使生成的二进制代码恰好为 21 dw 0xaa55 ; 结束标志
这里的做法是调用BIOS的10h中断来显示“Hello,OS world!”,其中bp为字符串地址,cx为串长度,ah为功能号,al指示光标置于串尾,bx指示页号为0然后字符显示属性为黑底红字,dh为行号,dl为列号(如果不做处理的话,默认dh,dl皆为0,即在第0行第0列显示),参数设置完之后则调用10h中断显示字符串。
总结:以上的四种方法,通过学习不仅了解显示的方法,更重要的是对汇编语言有了更多的认识。以上方法在实际操作中介于方便与否,大多采用的直接调用BIOS的10h 中断来操作。