本文主要是介绍X86架构(六)——光标控制,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!
光标(Cursor)是在屏幕上有规律地闪动的一条小橫线,通常用于指示下一个要显示的字符位置。
光标在屏幕上的位置保存在显卡内部的两个光标寄存器中,每个寄存器是8位的,合起来形成一个16位的数值。标准VGA文本模式是25 行,每行80个字符。所以,0表示光标在屏幕上第0行第0列,80表示它在第1行第0列,这样算来,当光标在屏幕右下角时,该值为25×80-1=1999
光标寄存器是可读可写的
取当前光标位置
显卡的操作非常复杂,内部的寄存器很多。为了不过多占用主机的I/O空间,很多寄存器只能通过索引寄存器间接访问
索引寄存器的端口号是0x3d4,向它写入一个值,用来指定目标内部寄存器
两个8位的光标寄存器,其索引值分别是0x0e和0x0f,分别用于提供光标位置的高8位和低8位
光标寄存器的读写控制通过数据端口0x3d5指定
;程序清单_1
mov dx, 0x3d4 ;索引寄存器端口
mov al, 0x0e
out dx, al ;指定操作寄存器索引
mov dx, 0x3d5 ;数据端口
in al, dx ;读出0x0e寄存器的值
mov ah, al ;高8位传送到ah寄存器mov dx, 0x3d4
mov al, 0x0f
out dx, al
mov dx, 0x3d5
in al, dx ;ax寄存器中的数值此时是光标数据位置 ah+al
处理回车和换行字符
ASCII码0x0d
是回车,0x0a
是换行
回车和换行的概念最早起源于老式打字机。那种打字机上有滚筒,用于使纸张上下卷动,每敲击一个按键,字车往右移动一格,位于下一个可打印的位置。在这种设备上,将字车推到最左边,也就是一行的开始,叫做回车(Carriage Return);而拧一下滚筒,将纸上卷一行,叫做换行(Line Feed)。
如果是回车符0x0d,应将光标移动到当前行的行首。每行有80个字符,用当前光标位置除以80,余数不要,就可以得到当前行的行号。接着,再乘以80,就是当前行行首的光标数值。
;程序清单_2
;该程序接程序清单_1
mov bx, ax ;ax寄存器中存储着当前光标的位置
mov ax, bx ;16位除法,被除数在ax寄存器中
mov bl, 80 ;除数
div bl ;商在al中余数在ah中
mul bl ;结果在ax中
;mul 是乘法指令
;mul r/m8 ax = al * r/m8 8位乘法,结果16位
;mul r/m16 dx:ax = ax * r/m16 16位乘法,结果32位
mov bx, ax
jmp __set_cursor
如果是换行符0x0a,应将光标向下移动一行
;换行处理
;接程序清单_2
cmp cl, 0x0a ;字符存储在cl寄存器中
add bx, 80 ;cl = 0x0a bx存储的当前光标位置
jmp __roll_screen
设置光标位置
;光标设置
__roll_screen:cmp bx, 2000jl __set_cursor ;bx小于2000不需要滚屏mov ax, 0xb800 ;0xb800显存映射的内存地址;滚屏;将屏幕上第2~25 行的内容整体往上提一行;最后用黑底白字的空白字符填充第25行mov ds, axmov es, axcldmov si, 0xa0 ;第二行第一列的显存偏移地址(一行80个字符,用两个字节控制)[ds:si]mov di, 0x00 ;目的偏移地址 [es:di]mov cx, 1920 ;传送字节数 24 * 80 * 2 / 2rep movsw ;使用movsw传送所以除以2mov bx, 3840 ;25行显存偏移地址 24 * 80 * 2mov cx, 80 ;传送字数
__cls:mov word[ex:bx], 0x0720add bx, 2loop clsmov bx 1920 ;滚屏后光标位置对应的数值(最后一行第一列)
__set_cursor:mov dx, 0x3d4mov al, 0x0eout dx, al ;指定操作寄存器mov dx, 0x3d5mov al, bh ;光标位置高8位out dx, al ;写入数据端口mov dx, 0x3d4mov al, 0x0fout dx, almov dx, 0x03d5mov al, bl ;光标位置低8位out dx, al ;写入数据端口
这篇关于X86架构(六)——光标控制的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!