首页 > 代码库 > [自制简单操作系统] 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、杂七杂八(高分辨率和键盘输入)