我已经规范了系统代码风格,类似于按照linux分包,把各部分功能区分开了

Antz系统更新地址

Linux内核源码分析地址

Github项目地址

在之前的任务中,我们已经通过直接操作显卡驱动完成了简单的图形化。

需要了解之前的部分:

直接操作显卡请参考day03

简单图形化的实现请参考day09

Makefile

项目目录

console的图形化实现与规则均在main/bootpack.c中完成

interrupt/int.c 中实现了键盘中断处理,按键会中断两次,一次按下,一次弹起,在响应处理中,只需要处理第一次按下即可。

一 . 键盘按键

如何来判断中断来自于键盘?(代码如下)

    // gdt初始化操作...
// fifo加载操作...
if (fifo8_status(&keyfifo) != 0) { // True则说明中断来自于键盘
i = fifo8_get(&keyfifo);
io_sti();
// i 就是中断返回的值,分析他即可得到按键信息, 在下面我把它转换为了16进制存储在了一个char array s中
sprintf(s, "%02X", i);
// 把两次中断变为一次,看下文
}

得到了s,就是得到了键盘按键的信息。

开头说了,按下一次,会有两次中断发生,那么我们是否可以使用一个flag来区分按下和弹起呢?

	if (flag){
keyshow(); // 显示这次按键,把按下的中断当作一个键位的信息,把弹起的中断用下面flag的方法屏蔽掉
}
// 屏蔽
if(flag==1){
flag = 0 ;
}else {
flag = 1 ;
}

这是一个很拙略的实现方法,而且我测试了几次之后发现有一个bug,就是同时按下两个键位时,屏蔽的方法就会变成另一种。

比如开始是用按下识别一个键位,那么同时按下两个键位之后就是以弹起的方法来识别键位了。

这个情况留在之后再考虑。

二 . 按键识别

上文中已经将按键返回的数据存储到了char数组s,只需要在屏幕上显示s的数据就可以了。

int write_x = 55 ; //按键显示位置的x,y坐标
int write_y = 57 ; void key(struct BOOTINFO *binfo,char s[40]){
//在指定位置显示数据
showkeys(binfo->vram, binfo->scrnx, write_x, write_y, COL8_FFFFFF, s);
// 显示之后光标右移
write_x += 19 ;
// 如果超出右边界,换行
if(write_x>155){
write_x = 55 ;
write_y += 19 ;
}
// 如果超出下边界,刷新清理本页,开启新的一页
if(write_y>180){
new_pe(binfo);
} }

结果:

很明显,我们需要编写一种转换机制,将表示16进制的数据对应成为键盘按键。

键盘上需要显示的有字母和特殊符号,还有一些功能性的按键shift,backspace等。

测试记录了几个按键的按下数据

键盘 按下

F1 3B

F2 3C

F3 3D

F4 3E

A 1E

B 30

Backspace OE

空格 39

既然已经知道了对应关系,那么很容易就可以建立一种对应。

先来实现这几个特殊按键功能

我打算将 F1 实现为 clear 功能,实现页面刷新 。 Backspace 实现回退功能。Enter实现确定以及回车功能。

void showkey(struct BOOTINFO *binfo,char s[40]){
// 回车键
if(strcmp(s,"1C")==0){
write_x = 55 ; // 光标移动至下一行起始位置。
write_y += 19 ;
showkeys(binfo->vram, binfo->scrnx, 0, write_y, COL8_FFFFFF, "AntzOS>");
}
// F1 刷新本页
else if(strcmp(s,"3B")==0){
new_pe(binfo);
}
// 空格 光标后移一位
else if(strcmp(s,"39")==0){
showkeys(binfo->vram, binfo->scrnx, write_x, write_y, COL8_FFFFFF, " ");
write_x += 19 ;
}
// Backspace 删除退格
else if(strcmp(s,"0E")==0){
// 回退
write_x -= 19 ;
//重新覆盖这片区域
area_flash(binfo->vram, binfo->scrnx , COL8_000000, write_x, write_y, write_x+19, write_y+19);
}
// 其他按键
else {
showkeys(binfo->vram, binfo->scrnx, write_x, write_y, COL8_FFFFFF, s);
write_x += 19 ;
}
if(write_x>155){
write_x = 55 ;
write_y += 19 ;
//putfonts8_asc(binfo->vram, binfo->scrnx, 4, 57, COL8_FFFFFF, "AntzOS>");
}
if(write_y>180){
new_pe(binfo);
} }

字母识别同理,当然可以比上面实现的更加完善更加简洁,但我仓促之下就只能先做到这一步。

三 . Bug引发的思考

这里开始就和AntzOs实现没有多少联系了,不过在我测试按键中断时候发现了很多奇怪的小问题。

Caps Lock(大小写键) 是否开启并不会影响中断对你一个按键的返回信息,也就是所谓的大小写中断其实是无法区分的,那么现代系统如何区分呢? 同理于上面我们区分按下和弹起两次中断,我们可以将Caps Lock键的状态获取到,从而对当前按键进行所谓的大小写区分。

按下两次导致规则置换,会不会是因为中断响应的时间导致的。

自制操作系统Antz(10)——实现shell(上)的更多相关文章

  1. 自制操作系统Antz(11)——实现shell(下)命令响应

    我已经规范了系统代码风格,类似于按照linux分包,把各部分功能区分开了 Antz系统更新地址 Linux内核源码分析地址 Github项目地址 在之前的任务中,我们已经通过直接操作显卡驱动完成了简单 ...

  2. 自制操作系统Antz(7)——实现内核 (上)

    Antz系统更新地址: https://www.cnblogs.com/LexMoon/category/1262287.html Linux内核源码分析地址:https://www.cnblogs. ...

  3. 自制操作系统Antz -- 系列文章

    自制操作系统Antz day10——实现shell(上) AntzUhl 2018-10-10 16:25 阅读:192 评论:0   Linux内核源码分析 day01——内存寻址 AntzUhl ...

  4. 自制操作系统Antz(13) 显示图片

    显示图片只是在多媒体课上看着bmp格式图片的突发奇想,然后就实现在了我自己的操作系统 Antz系统更新地址 Linux内核源码分析地址 Github项目地址 效果图: 显示图片的原理 在之前显卡操作时 ...

  5. 自制操作系统Antz(2)——进入保护模式 (上) jmp到保护模式

    Antz系统更新地址: https://www.cnblogs.com/LexMoon/category/1262287.htm Linux内核源码分析地址:https://www.cnblogs.c ...

  6. 自制操作系统Antz(1)——Boot Sector

    0.引子 最近在看操作系统底层方面的东西,最开始的为什么是07c00h这个问题就让我对操作系统有了很大的兴趣.所以准备在看书之余顺便写一个操作系统(Anz).至于为什么这个系统会被叫做Antz,可以参 ...

  7. 自制操作系统Antz(12)——承上启下

    我已经规范了系统代码风格,类似于按照linux分包,把各部分功能区分开了 Antz系统更新地址 Linux内核源码分析地址 Github项目地址 在之前的工作中,AntzOS已经从单调的界面,变得逐渐 ...

  8. 自制操作系统Antz(5)——深入理解保护模式与进入方法

    Antz系统更新地址: https://www.cnblogs.com/LexMoon/category/1262287.html Linux内核源码分析地址:https://www.cnblogs. ...

  9. 自制操作系统Antz(3)——进入保护模式 (中) 直接操作显存

    Antz系统更新地址: https://www.cnblogs.com/LexMoon/category/1262287.html Linux内核源码分析地址:https://www.cnblogs. ...

随机推荐

  1. 虚拟机设置IP

    方式NAT 版本信息VMware(14.1.1 build-7528167)和centos7 一,设置网络适配器: 二.设置虚拟网络 三.配置虚拟机(/etc/sysconfig/network-sc ...

  2. SpringBoot注解大全 转

    2019年3月17日22:30:10 一.注解(annotations)列表 @SpringBootApplication:包含了@ComponentScan.@Configuration和@Enab ...

  3. SQL之NULL值的几种处理方式

    1.创建测试表: drop table if exists tab_null_operator; create table tab_null_operator as select 1 as id,'c ...

  4. 抓包分析、多线程爬虫及xpath学习

    1.抓包分析 1.1 Fiddler安装及基本操作 由于很多网站采用的是HTTPS协议,而fiddler默认不支持HTTPS,先通过设置使fiddler能抓取HTTPS网站,过程可参考(https:/ ...

  5. python之for循环

    for循环,也称定循环,即一般用于循环次数确定的循环,通常可用于遍历序列,如字符串str,列表list,元组tuple等 格式: for 变量 in 序列: command1 command2 ... ...

  6. springBoot生成日志文件

    一.安装lombok 说明: 安装bomlok后model可以不用写get.set方法,slf4j日志直接使用log打印 1. Maven Repository中下载lombok.jar 2. 将lo ...

  7. 2017(4)数据库系统,分布式数据库,NoSQL,反规范化

    试题四(共 25 分) 阅读以下关于数据库分析与建模的叙述,在答题纸上回答问题 1至问题 3. [说明] 某电子商务企业随着业务不断发展,销售订单不断增加,每月订单超过了 50 万笔,急需开发一套新的 ...

  8. 多线程深入:乐观锁与悲观锁以及乐观锁的一种实现方式-CAS(转)

    原文:https://www.cnblogs.com/qjjazry/p/6581568.html 首先介绍一些乐观锁与悲观锁: 悲观锁:总是假设最坏的情况,每次去拿数据的时候都认为别人会修改,所以每 ...

  9. RAxML安装

    1.下载解压 $ wget https://codeload.github.com/stamatak/standard-RAxML/zip/master -O standard-RAxML-maste ...

  10. python爬虫简单的添加代理进行访问

    在使用python对网页进行多次快速爬取的时候,访问次数过于频繁,服务器不会考虑User-Agent的信息,会直接把你视为爬虫,从而过滤掉,拒绝你的访问,在这种时候就需要设置代理,我们可以给proxi ...