首页 > 代码库 > [自制简单操作系统] 5、杂七杂八(高分辨率和键盘输入)
[自制简单操作系统] 5、杂七杂八(高分辨率和键盘输入)
前言:
>_<" 这几天正在研究一个好玩的,准备写《软硬结合第三篇——科班的还是可以修电脑的》,可是当前遇到个技术难点——WHDI,所以操作系统这里就慢了好大一节啦!但是以操作系统多任务的思路,感觉还是把这个优先级并不是太低的进程拿出来做一下吧!毕竟技术难点有时候需要灵感的哈~(不过有大神知道VGA接口的通信原理吗?求给个好一点的链接看看!)。回到正题上来,这次学的东西有点杂,主要就是高分辨率和键盘输入相关~对于前者和汇编及硬件联系较多(具体可以查VBE Video Electronics Standards Association BIOS extension),后者主要是映射关系,没多大难度~
效果展示:
>_<" 把分辨率调高了,可以看更宽的画面了。同时在WINDOW窗口中加入文字输入框,注意这里输入的英文字母必须是大写的,而且一些特殊的按键也会出现错误,今后还要优化~
部分程序说明:
>_<" 分辨率控制(asmhead.nas内部分)
1 ; haribote-os boot asm 2 ; TAB=4 3 4 [INSTRSET "i486p"] ;必须加!!! 5 6 VBEMODE EQU 0x101 ; 1024 x 768 x 8bit //显示模式 7 BOTPAK EQU 0x00280000 ; bootpack的内存首址 8 DSKCAC EQU 0x00100000 ; 9 DSKCAC0 EQU 0x00008000 ; 10 11 ; BOOT_INFO相关12 CYLS EQU 0x0ff0 ; 设定启动区13 LEDS EQU 0x0ff114 VMODE EQU 0x0ff2 ; 关于颜色的信息颜色的位数15 SCRNX EQU 0x0ff4 ; 分辨率X16 SCRNY EQU 0x0ff6 ; 分辨率Y17 VRAM EQU 0x0ff8 ; 图像缓冲区的开始地址18 19 ORG 0xc200 ; 这个程序要装在到什么位置,在ipl10读盘结束的地方转到该处20 ;-----------------------------------------------画面模式设置21 ;确认VBE是否存在[显示模式协议BOOS]22 ;给AX,ES,DI赋好值之后调用int如果AX没有变成0x004f就没有VBE就只能用320*200的分辨率了23 ;JMP scrn320 ;便于调试我们直接还是用老的显示模式,所以直接跳过高分辨率的部分24 MOV AX,0x900025 MOV ES,AX26 MOV DI,027 MOV AX,0x4f0028 INT 0x1029 CMP AX,0x004f30 JNE scrn32031 ;检查VBE的版本32 MOV AX,[ES:DI+4]33 CMP AX,0x020034 JB scrn320 ; if (AX < 0x0200) goto scrn320 35 ;取得画面模式36 MOV CX,VBEMODE37 MOV AX,0x4f0138 INT 0x1039 CMP AX,0x004f40 JNE scrn32041 ;画面模式信息确认42 CMP BYTE [ES:DI+0x19],843 JNE scrn32044 CMP BYTE [ES:DI+0x1b],445 JNE scrn32046 MOV AX,[ES:DI+0x00]47 AND AX,0x008048 JZ scrn320 49 ;画面模式切换50 MOV BX,VBEMODE+0x400051 MOV AX,0x4f0252 INT 0x1053 MOV BYTE [VMODE],8 ; 记下画面模式54 MOV AX,[ES:DI+0x12]55 MOV [SCRNX],AX56 MOV AX,[ES:DI+0x14]57 MOV [SCRNY],AX58 MOV EAX,[ES:DI+0x28]59 MOV [VRAM],EAX60 JMP keystatus 61 scrn320:62 MOV AL,0x13 ; VGA图、320x200x8bit彩色,如果高分辨率模式不行就只能用低分辨率模式了63 MOV AH,0x0064 INT 0x1065 MOV BYTE [VMODE],8 ; 记下画面模式66 MOV WORD [SCRNX],32067 MOV WORD [SCRNY],20068 MOV DWORD [VRAM],0x000a000069 70 ;高分辨率{给AX赋值0x4f02,BX赋值模式码{这里要把模式码+0x4000,QEMU中不能直接用下面的方法}71 ;0x101 640 480 8bit72 ;0x103 800 600 8bit73 ;0x105 1024 768 8bit74 ;0x107 1280 1024 8bit75 ; MOV BX,0x410176 ; MOV AX,0x4f0277 ; INT 0x1078 ; MOV BYTE[VMODE],879 ; MOV WORD[SCRNX],64080 ; MOV WORD[SCRNY],48081 ; MOV DWORD[VRAM],0xe000000082 ;-----------------------------------------------画面模式设置
- 第4行必须添加,否则就无法正常显示,采用高版本处理器的汇编语言
- 第6行VBEMODE是控制显示模式的,具体见71~74行
- 第23行,如果直接跳到scrn320就 执行老的画面显示模式,这里scrn32后的一段就是原来的老的显示模式部分,如果把老的显示模式设置部分换为70行后的部分也能显示高分辨率,但是在模 拟器上最后一种模式不能显示,其实这里上面那么多行是为了在真机上能够启用高分辨率而做的检查工作,由于VESA相关工作做的没能把所有的显示模式统一, 所以,只有遵守VBSA-BIOS extension协议的才能启用高分辨率~
>_<" 键盘输入
- 这里的键盘输入,根据映射规律我们建立一个映射表:
1 static char keytable[0x54] = {2 0, 0, ‘1‘, ‘2‘, ‘3‘, ‘4‘, ‘5‘, ‘6‘, ‘7‘, ‘8‘, ‘9‘, ‘0‘, ‘-‘, ‘^‘, 0, 0,3 ‘Q‘, ‘W‘, ‘E‘, ‘R‘, ‘T‘, ‘Y‘, ‘U‘, ‘I‘, ‘O‘, ‘P‘, ‘@‘, ‘[‘, 0, 0, ‘A‘, ‘S‘,4 ‘D‘, ‘F‘, ‘G‘, ‘H‘, ‘J‘, ‘K‘, ‘L‘, ‘;‘, ‘:‘, 0, 0, ‘]‘, ‘Z‘, ‘X‘, ‘C‘, ‘V‘,5 ‘B‘, ‘N‘, ‘M‘, ‘,‘, ‘.‘, ‘/‘, 0, ‘*‘, 0, ‘ ‘, 0, 0, 0, 0, 0, 0,6 0, 0, 0, 0, 0, 0, 0, ‘7‘, ‘8‘, ‘9‘, ‘-‘, ‘4‘, ‘5‘, ‘6‘, ‘+‘, ‘1‘,7 ‘2‘, ‘3‘, ‘0‘, ‘.‘8 };
- 然后在键盘消息处理部分直接利用映射表读取按键信息,注意这里函数putfonts8_asc_sht是显示字符串,所以s[1]=0;
- 这里sht_win是我们新建的一个(也是原来的那个显示计数的窗口,只是改了个名字而已)窗口,让输入的文字在上面显示,每次输入cursor_x即光标的位置要后移
1 if (i < 0x54 + 256) {2 if (keytable[i - 256] != 0 && cursor_x < 144) {//一般字符3 /* 显示一次就前移一次光标 */4 s[0] = keytable[i - 256];5 s[1] = 0;6 putfonts8_asc_sht(sht_win, cursor_x, 28, COL8_000000, COL8_FFFFFF, s, 1);7 cursor_x += 8;//记录光标位置8 }9 }
- 当然,退格键要特殊处理,如下:
1 if (i == 256 + 0x0e && cursor_x > 8) { /* 退格键 */2 /* 用空格键把光标消去后,后移1次光标 */3 putfonts8_asc_sht(sht_win, cursor_x, 28, COL8_000000, COL8_FFFFFF, " ", 1);4 cursor_x -= 8;5 }
- 此外,这里的文本编辑区,其实就是像画窗口一样画出来的界面:
1 void make_textbox8(struct SHEET *sht, int x0, int y0, int sx, int sy, int c) 2 { 3 int x1 = x0 + sx, y1 = y0 + sy; 4 boxfill8(sht->buf, sht->bxsize, COL8_848484, x0 - 2, y0 - 3, x1 + 1, y0 - 3); 5 boxfill8(sht->buf, sht->bxsize, COL8_848484, x0 - 3, y0 - 3, x0 - 3, y1 + 1); 6 boxfill8(sht->buf, sht->bxsize, COL8_FFFFFF, x0 - 3, y1 + 2, x1 + 1, y1 + 2); 7 boxfill8(sht->buf, sht->bxsize, COL8_FFFFFF, x1 + 2, y0 - 3, x1 + 2, y1 + 2); 8 boxfill8(sht->buf, sht->bxsize, COL8_000000, x0 - 1, y0 - 2, x1 + 0, y0 - 2); 9 boxfill8(sht->buf, sht->bxsize, COL8_000000, x0 - 2, y0 - 2, x0 - 2, y1 + 0);10 boxfill8(sht->buf, sht->bxsize, COL8_C6C6C6, x0 - 2, y1 + 1, x1 + 0, y1 + 1);11 boxfill8(sht->buf, sht->bxsize, COL8_C6C6C6, x1 + 1, y0 - 2, x1 + 1, y1 + 1);12 boxfill8(sht->buf, sht->bxsize, c, x0 - 1, y0 - 1, x1 + 0, y1 + 0);13 return;14 }
>_<" 鼠标移动窗口及光标闪烁
- 鼠标移动其实很简单,就是判断是否出屏,然后控制边界,最后调用窗口移动函数进行重新定位
1 /* 移动鼠标 */ 2 mx += mdec.x; 3 my += mdec.y; 4 if (mx < 0) { 5 mx = 0; 6 } 7 if (my < 0) { 8 my = 0; 9 }10 if (mx > binfo->scrnx - 1) {11 mx = binfo->scrnx - 1;12 }13 if (my > binfo->scrny - 1) {14 my = binfo->scrny - 1;15 }16 sprintf(s, "(%3d, %3d)", mx, my);17 putfonts8_asc_sht(sht_back, 0, 0, COL8_FFFFFF, COL8_008484, s, 10);//清,显,刷18 sheet_slide(sht_mouse, mx, my);19 //移动窗口计算20 if((mdec.btn & 0x01)!=0){21 sheet_slide(sht_win,mx-80,my-80);22 }//按下左键移动sht_win
- 光标闪烁就是利用定时器,直接贴代码,一看就知道
1 else if (i<=1) { //光标用定时器 2 if (i != 0) { 3 timer_init(timer3, &fifo, 0); /* 師偼0傪 */ 4 cursor_c = COL8_000000; 5 } else { 6 timer_init(timer3, &fifo, 1); /* 師偼1傪 */ 7 cursor_c = COL8_FFFFFF; 8 } 9 timer_settime(timer3, 50);10 boxfill8(sht_win->buf, sht_win->bxsize, cursor_c, cursor_x, 28, cursor_x + 7, 43);11 sheet_refresh(sht_win, cursor_x, 28, cursor_x + 8, 44);12 }
资料链接:http://pan.baidu.com/s/1m2a1C
[自制简单操作系统] 5、杂七杂八(高分辨率和键盘输入)
声明:以上内容来自用户投稿及互联网公开渠道收集整理发布,本网站不拥有所有权,未作人工编辑处理,也不承担相关法律责任,若内容有误或涉及侵权可进行投诉: 投诉/举报 工作人员会在5个工作日内联系你,一经查实,本站将立刻删除涉嫌侵权内容。