30天开发操作系统 第 17 天 -- 命令行窗口

前言

今天一开始,请大家先回忆一下任务A的情形。在harib13e中,任务A下面的LEVEL中有任务因此FIFO为空时我们可以让任务A进入休眠状态。那么,如果我们并未启动任务B0~ B0~ B2, B2的话,任务A又将会如何呢? 首先,如果我们不对任务A进行任何改写,就按照它现在的样子进入休眠状态的话,大家想一想,会发生什么呢?一旦任务A休眠,mtask.c将自动寻找下层LEVEL中的任务,但由于这一次 我们没有启动任务B0~B2,因此程序就找不到其他的任务而导致运行出现异常。 如果这样不行,那我们把休眠的部分再改回io_hlt;不就好了吗?这样一来,不但程序运行不会出现异常,还能省电呢…话虽如此,但如果一个操作系统要根据下层LEVEL是否 存在任务来改写程序的话,大家觉得靠谱吗?即使不改写程序,也能自动在适当的 LEVEL运行适当的任务,这样的操作系统才是优秀的操作系统。 因此,一般情况下可以让任务休眠,但当所有LEVEL中都没有任务存在的时候,就需要 HTL 了。接下来我们就按照这个要求来改写mtask.c。

一、书接上文,闲置任务

我们确实可以这样来改, 那么我们从task_sleep开始改起吧。 但是改写之后task_sleep 将变得更加复杂,速度也会打折扣(说是会打折扣,其实应该也就是稍微慢一点点而已啦)其实我们还有更好的方法。 如果“所有LEVEL中都没有任务”就会出问题,那我们只要避免这种情况发生不就可以了吗? 这类似于我们写定时器的时候所采用的“卫兵”的思路。

void task_idle(void)
{for (;;) {io_hlt();}
}

我们创建这样一个任务,并把它一直放在最下层LEVEL中,大家觉得如何? “idle”(闲置) 就是代表“不工作,空闲”的意思,这个任务的功能只是执行HTL。
这样一来,即便任务A进入休眠状态, 系统也会自动切换到上面这个闲置任务,于是便开始执行HTL。当我们移动鼠标,FIFO中有数据写入的时候,任务A就会被唤醒,系统会自动切换到任务A继续工作。
我们完全不需要对task_sleep等代码进行任何改动,只需在task_init中将这个闲置 任务放在最下层LEVEL中就可以了。

struct TASK *task_init(struct MEMMAN *memman)
{int i;struct TASK *task, *idle;...idle = task_alloc();idle->tss.esp = memman_alloc_4k(memman, 64 * 1024) + 64 * 1024;idle->tss.eip = (int) &task_idle;idle->tss.es = 1 * 8;idle->tss.cs = 2 * 8;idle->tss.ss = 1 * 8;idle->tss.ds = 1 * 8;idle->tss.fs = 1 * 8;idle->tss.gs = 1 * 8;task_run(idle, MAX_TASKLEVELS - 1, 1);return task;
}

接着我们来测试一下, 将HariMain改成下面这样。

void HariMain(void)
{.../* sht_win_b */for (i = 0; i < 3; i++) {sht_win_b[i] = sheet_alloc(shtctl);buf_win_b = (unsigned char *) memman_alloc_4k(memman, 144 * 52);sheet_setbuf(sht_win_b[i], buf_win_b, 144, 52, -1); sprintf(s, "task_b%d", i);make_window8(buf_win_b, 144, 52, s, 0);task_b[i] = task_alloc();task_b[i]->tss.esp = memman_alloc_4k(memman, 64 * 1024) + 64 * 1024 - 8;task_b[i]->tss.eip = (int) &task_b_main;task_b[i]->tss.es = 1 * 8;task_b[i]->tss.cs = 2 * 8;task_b[i]->tss.ss = 1 * 8;task_b[i]->tss.ds = 1 * 8;task_b[i]->tss.fs = 1 * 8;task_b[i]->tss.gs = 1 * 8;*((int *) (task_b[i]->tss.esp + 4)) = (int) sht_win_b[i];/* task_run(task_b[i], 2, i + 1); 这里*/}...
}

这样就只剩下任务A和task_idle 了。也就是说, 我们不让系统运行任务B0~B2, 当然, 这样改只是不启动任务而已,窗口还是会照常显示的。 然后我们来 make run”,画面如下。
在这里插入图片描述

二、创建命令行窗口

1.雏形

所谓命令行窗口,就是大家在运行 “make run” 的时候所使用的那个黑底白字的,在里面输入文件名就可以运行程序的东西。接下来我们就做一个试试看。 “这玩意儿能算是命令行窗口吗?”我们一开始做出来的东西很可能带来这样的疑问,不过 没关系,我们会让它逐步发展壮大,最终实现可以启动应用程序的功能。你看,是不是越来越像个操作系统的样子了呢?大家一定迫不及待了吧。
我们并不打算将命令行窗口作为任务A的一部分,而是单独做成一个新的任务。这样一来,就像任务B0-B2-样,我们可以很容易地创建多个命令行窗口。
我们对任务B的程序进行一些修改,并将任务A程序的一部分融合进去,就写成了下面这个样子。计数我们已经玩够了,所以把计数的代码从任务B中删除。

void HariMain(void)
{/* sht_cons */sht_cons = sheet_alloc(shtctl);buf_cons = (unsigned char *) memman_alloc_4k(memman, 256 * 165);sheet_setbuf(sht_cons, buf_cons, 256, 165, -1); make_window8(buf_cons, 256, 165, "console", 0);make_textbox8(sht_cons, 8, 28, 240, 128, COL8_000000);task_cons = task_alloc();task_cons->tss.esp = memman_alloc_4k(memman, 64 * 1024) + 64 * 1024 - 8;task_cons->tss.eip = (int) &console_task;task_cons->tss.es = 1 * 8;task_cons->tss.cs = 2 * 8;task_cons->tss.ss = 1 * 8;task_cons->tss.ds = 1 * 8;task_cons->tss.fs = 1 * 8;task_cons->tss.gs = 1 * 8;*((int *) (task_cons->tss.esp + 4)) = (int) sht_cons;task_run(task_cons, 2, 2); /* level=2, priority=2 */...sheet_slide(sht_back,  0,  0);sheet_slide(sht_cons, 32,  4);sheet_slide(sht_win,  64, 56);sheet_slide(sht_mouse, mx, my);sheet_updown(sht_back,  0);sheet_updown(sht_cons,  1);sheet_updown(sht_win,   2);sheet_updown(sht_mouse, 3);sprintf(s, "(%3d, %3d)", mx, my);...
}void console_task(struct SHEET *sheet)
{struct FIFO32 fifo;struct TIMER *timer;struct TASK *task = task_now();int i, fifobuf[128], cursor_x = 8, cursor_c = COL8_000000;fifo32_init(&fifo, 128, fifobuf, task);timer = timer_alloc();timer_init(timer, &fifo, 1);timer_settime(timer, 50);for (;;) {io_cli();if (fifo32_status(&fifo) == 0) {task_sleep(task);io_sti();} else {i = fifo32_get(&fifo);io_sti();if (i <= 1) { /* 光标用计时器 */if (i != 0) {timer_init(timer, &fifo, 0); /* 下次置位 0 */cursor_c = COL8_FFFFFF;} else {timer_init(timer, &fifo, 1); /* 下次置位 1 */cursor_c = COL8_000000;}timer_settime(timer, 50);boxfill8(sheet->buf, sheet->bxsize, cursor_c, cursor_x, 28, cursor_x + 7, 43);sheet_refresh(sheet, cursor_x, 28, cursor_x + 8, 44);}}}
}

有一个地方需要解释一下,就是console_task中task=tasknow();这里。在console_task中需要执行休眠,因此必须要知道自己本身的TASK 结构所在的内存地址。由于HariMain中准备了可以像sheet那样从HariMain传入这个地址,但那样也有点过于繁琐了。忽然想到在 task_cons, mtask.c中不是有个task_now函数吗,就用它来获取TASK地址好了。

作为我们的命令行窗口雏形, 看上去还不赖。
在这里插入图片描述

2.切换输入窗口

命令行窗口这东西,如果不能在里面输人字符的话就毫无用处了,因此我们得让它能接受字符输入才行。不过现在无论我们输入什么字符,都会跑到任务A的窗口中去,所以为了能够往命令行窗口中输入字符,我们要让系统在按下“Tab”键的时候,将输入窗口切换到命令行窗口上去。
虽说是切换窗口,其实我们只是先将窗口标题栏的颜色改一改而已啦(苦笑)。真正负责输人切换的部分我们下一小节再写,改变先从表面工夫开始吧。

要改变窗口标题栏颜色,最好将makewindow8中描绘窗口标题栏的代码,和描绘窗口剩余部分的代码区分开来,我们将这个函数改写一下。

void make_window8(unsigned char *buf, int xsize, int ysize, char *title, char act)
{boxfill8(buf, xsize, COL8_C6C6C6, 0,         0,         xsize - 1, 0        );boxfill8(buf, xsize, COL8_FFFFFF, 1,         1,         xsize - 2, 1        );boxfill8(buf, xsize, COL8_C6C6C6, 0,         0,         0,         ysize - 1);boxfill8(buf, xsize, COL8_FFFFFF, 1,         1,         1,         ysize - 2);boxfill8(buf, xsize, COL8_848484, xsize - 2, 1,         xsize - 2, ysize - 2);boxfill8(buf, xsize, COL8_000000, xsize - 1, 0,         xsize - 1, ysize - 1);boxfill8(buf, xsize, COL8_C6C6C6, 2,         2,         xsize - 3, ysize - 3);boxfill8(buf, xsize, COL8_848484, 1,         ysize - 2, xsize - 2, ysize - 2);boxfill8(buf, xsize, COL8_000000, 0,         ysize - 1, xsize - 1, ysize - 1);make_wtitle8(buf, xsize, title, act);return;
}void make_wtitle8(unsigned char *buf, int xsize, char *title, char act)
{static char closebtn[14][16] = {"OOOOOOOOOOOOOOO@","OQQQQQQQQQQQQQ$@","OQQQQQQQQQQQQQ$@","OQQQ@@QQQQ@@QQ$@","OQQQQ@@QQ@@QQQ$@","OQQQQQ@@@@QQQQ$@","OQQQQQQ@@QQQQQ$@","OQQQQQ@@@@QQQQ$@","OQQQQ@@QQ@@QQQ$@","OQQQ@@QQQQ@@QQ$@","OQQQQQQQQQQQQQ$@","OQQQQQQQQQQQQQ$@","O$$$$$$$$$$$$$$@","@@@@@@@@@@@@@@@@"};int x, y;char c, tc, tbc;if (act != 0) {tc = COL8_FFFFFF;tbc = COL8_000084;} else {tc = COL8_C6C6C6;tbc = COL8_848484;}boxfill8(buf, xsize, tbc, 3, 3, xsize - 4, 20);putfonts8_asc(buf, xsize, 24, 4, tc, title);for (y = 0; y < 14; y++) {for (x = 0; x < 16; x++) {c = closebtn[y][x];if (c == '@') {c = COL8_000000;} else if (c == '$') {c = COL8_848484;} else if (c == 'Q') {c = COL8_C6C6C6;} else {c = COL8_FFFFFF;}buf[(5 + y) * xsize + (xsize - 21 + x)] = c;}}return;
}

这样就差不多了,接下来我们来改写HariMMain。

void HariMain(void)
{...int key_to = 0;for (;;) {io_cli();for (;;) {io_cli();if (fifo32_status(&fifo) == 0) {task_sleep(task_a);io_sti();} else {i = fifo32_get(&fifo);io_sti();if (256 <= i && i <= 511) { /* 键盘数据 */sprintf(s, "%02X", i - 256);putfonts8_asc_sht(sht_back, 0, 16, COL8_FFFFFF, COL8_008484, s, 2);if (i < 0x54 + 256) {...}if (i == 256 + 0x0e && cursor_x > 8) { /* 退格键 */				...}if (i == 256 + 0x0f) { /* Tab */...} else {key_to = 0;make_wtitle8(buf_win,  sht_win->bxsize,  "task_a",  1);make_wtitle8(buf_cons, sht_cons->bxsize, "console", 0);}sheet_refresh(sht_win,  0, 0, sht_win->bxsize,  21);sheet_refresh(sht_cons, 0, 0, sht_cons->bxsize, 21);}boxfill8(sht_win->buf, sht_win->bxsize, cursor_c, cursor_x, 28, cursor_x + 7, 43);sheet_refresh(sht_win, cursor_x, 28, cursor_x + 8, 44);} else if (512 <= i && i <= 767) { /* 鼠标数据 */...} else if (i <= 1) {}}}
}

这段代码的重点在于key to这个变量,用于记录键盘输入(key)应该发送到(to)哪里。为0则发送到任务A,为1则发送到命令行窗口任务。
还是一如既往地“make run”,然后按下Tab键试试看。
哇,颜色变了耶!真不错啊。心情好激动,先按个10次Tab键看看。嗒…
然后又很得意地输入了“abc”试试看,果然,结果是这个样子。
在这里插入图片描述
好,接下来我们就来实现键盘输人啦!

3.字符输入

要实现字符的输入,只要在键盘被按下的时候向console task的FIFO发送数据即可。但要发 送数据,必须要知道structFIFO的内存地址才行。唔,这可怎么办呢? 我们可以让任务A在创建taskcons的时候,顺便将FIFO也准备好。这样一来,任务A就能知 道task_cons所使用的FIFO的地址, 我们的问题便迎刃而解了。 等等,这样还是太麻烦了,我们还是把struct FIFO放到struct TASK里面去吧。基本上没有什么任务是完全用不到FIFO的, 因此我们把它们绑定起来。

struct TASK {int sel, flags; int level, priority;struct FIFO32 fifo;/* 这里 */struct TSS32 tss;
};

接下来我们来修改 一 HariMain, 使其判断key_to的值并向task_cons的FIFO发送数据。

void HariMain(void)
{...for (;;) {io_cli();if (fifo32_status(&fifo) == 0) {task_sleep(task_a);io_sti();} else {i = fifo32_get(&fifo);io_sti();if (256 <= i && i <= 511) { /* 键盘数据 */sprintf(s, "%02X", i - 256);putfonts8_asc_sht(sht_back, 0, 16, COL8_FFFFFF, COL8_008484, s, 2);if (i < 0x54 + 256 && keytable[i - 256] != 0) { /* 一般字符 */if (key_to == 0) {	/* 发送给任务A */if (cursor_x < 128) {/* 显示一个字符之后将光标后移一位 */s[0] = keytable[i - 256];s[1] = 0;putfonts8_asc_sht(sht_win, cursor_x, 28, COL8_000000, COL8_FFFFFF, s, 1);cursor_x += 8;}} else {	/* 发送给命令行窗口 */fifo32_put(&task_cons->fifo, keytable[i - 256] + 256);}}if (i == 256 + 0x0e) {	/* 退格键 */if (key_to == 0) {	/* 发送给任务A */if (cursor_x > 8) {/* 用空白擦除光标后将光标前移一位 */putfonts8_asc_sht(sht_win, cursor_x, 28, COL8_000000, COL8_FFFFFF, " ", 1);cursor_x -= 8;}} else {	/* 发送给命令行窗口 */fifo32_put(&task_cons->fifo, 8 + 256);}}if (i == 256 + 0x0f) { /* Tab */if (key_to == 0) {key_to = 1;make_wtitle8(buf_win,  sht_win->bxsize,  "task_a",  0);make_wtitle8(buf_cons, sht_cons->bxsize, "console", 1);} else {key_to = 0;make_wtitle8(buf_win,  sht_win->bxsize,  "task_a",  1);make_wtitle8(buf_cons, sht_cons->bxsize, "console", 0);}sheet_refresh(sht_win,  0, 0, sht_win->bxsize,  21);sheet_refresh(sht_cons, 0, 0, sht_cons->bxsize, 21);}/* 重新显示光标 */boxfill8(sht_win->buf, sht_win->bxsize, cursor_c, cursor_x, 28, cursor_x + 7, 43);sheet_refresh(sht_win, cursor_x, 28, cursor_x + 8, 44);} else if (512 <= i && i <= 767) { ...} else if (i <= 1) { /*  */...}}}
}

系统会向命令行窗口任务发送键盘数据, 支持一般的字符输入和退格键。当key_to不为0时, 由于在命令行窗口中也使用了定时器等,为了不与键盘数据冲突,我们在写入FIFO的时候将键盘数据的值加上256。

在向命令行窗口发送键盘数据的时候,并不是直接发送从键盘接收到的原始数据,而是发送 经过keytable[]转换后的值。究其原因,是由于这 样做可以省去在命令行窗口任务中将按键编码转 换成字符编码的步骤。 对于退格键,我们将它的字符编码定义为8,因为在ASCII码中, 编码8就对应着退格键,我们只是和它接轨而已。当然,如果你不想用8也完全没有问题。

console_task也需要改写一下,因为我们必须让它能够接收并处理键盘数据。此外, 我们还得把&fifo改写成&task-fifo:

void console_task(struct SHEET *sheet)
{struct TIMER *timer;struct TASK *task = task_now();int i, fifobuf[128], cursor_x = 16, cursor_c = COL8_000000;char s[2];fifo32_init(&task->fifo, 128, fifobuf, task);timer = timer_alloc();timer_init(timer, &task->fifo, 1);timer_settime(timer, 50);/* 显示提示符 */putfonts8_asc_sht(sheet, 8, 28, COL8_FFFFFF, COL8_000000, ">", 1);for (;;) {io_cli();if (fifo32_status(&task->fifo) == 0) {task_sleep(task);io_sti();} else {i = fifo32_get(&task->fifo);io_sti();if (i <= 1) { /* 光标用定时器 */if (i != 0) {timer_init(timer, &task->fifo, 0); /* 下次 0 */cursor_c = COL8_FFFFFF;} else {timer_init(timer, &task->fifo, 1); /* 下次 1 */cursor_c = COL8_000000;}timer_settime(timer, 50);}if (256 <= i && i <= 511) { /* 键盘数据 */if (i == 8 + 256) {/* 退格键 */if (cursor_x > 16) {/* 用空白擦除光标后将光标前移一位 */putfonts8_asc_sht(sheet, cursor_x, 28, COL8_FFFFFF, COL8_000000, " ", 1);cursor_x -= 8;}} else {/* 一般字符 */if (cursor_x < 240) {/* 显示一个字符之后将光标后移一位 */s[0] = i - 256;s[1] = 0;putfonts8_asc_sht(sheet, cursor_x, 28, COL8_FFFFFF, COL8_000000, s, 1);cursor_x += 8;}}}/* 重新显示光标 */boxfill8(sheet->buf, sheet->bxsize, cursor_c, cursor_x, 28, cursor_x + 7, 43);sheet_refresh(sheet, cursor_x, 28, cursor_x + 8, 44);}}
}

上面的程序基本是照猫画虎而来, 唯一的一点区别就是开头显示提示符“>”的 地方了。退格键的处理上也对可以删除的界限作了调整,以避免退格时擦掉提示符。
我们来运行一下看看, make run
在这里插入图片描述

成功了,真是个伟大的胜利。现在我们可以输入英文、数字和符号了,但还无法输入“!” 和“%”。

4.符号的输入

我们这就要实现“!”和“%” 的输人。为了能够输人“!”和“%”,我们必须要处理Shif键。根据按键编码表,Shif键的按键编码如下(觉得看表格看麻烦的话,可以自己 “make run" 一下, 按键的时候屏幕上会显示出按键编码哦)。
在这里插入图片描述

因此, 我们准备一个key_shift变量, 当左Shif按下时置为1, 右Shif按下时置为2,两个都不按时置为0,两个都按下(有人会这么干吗? )的时候就置为3。当key_shift为0时,我们用keytable0[ ]将按键编码转换为字符编码,而当key_shift不为0时,则 使用keytable1[ ]进行转换。(大家还有什么好的想法吗?)

就是下面这样将上面的思路用程序写到HariMain中:

void HariMain(void)
{...static char keytable0[0x80] = {0,   0,   '1', '2', '3', '4', '5', '6', '7', '8', '9', '0', '-', '^', 0,   0,'Q', 'W', 'E', 'R', 'T', 'Y', 'U', 'I', 'O', 'P', '@', '[', 0,   0,   'A', 'S','D', 'F', 'G', 'H', 'J', 'K', 'L', ';', ':', 0,   0,   ']', 'Z', 'X', 'C', 'V','B', 'N', 'M', ',', '.', '/', 0,   '*', 0,   ' ', 0,   0,   0,   0,   0,   0,0,   0,   0,   0,   0,   0,   0,   '7', '8', '9', '-', '4', '5', '6', '+', '1','2', '3', '0', '.', 0,   0,   0,   0,   0,   0,   0,   0,   0,   0,   0,   0,0,   0,   0,   0,   0,   0,   0,   0,   0,   0,   0,   0,   0,   0,   0,   0,0,   0,   0,   0x5c, 0,  0,   0,   0,   0,   0,   0,   0,   0,   0x5c, 0,  0};static char keytable1[0x80] = {0,   0,   '!', 0x22, '#', '$', '%', '&', 0x27, '(', ')', '~', '=', '~', 0,   0,'Q', 'W', 'E', 'R', 'T', 'Y', 'U', 'I', 'O', 'P', '`', '{', 0,   0,   'A', 'S','D', 'F', 'G', 'H', 'J', 'K', 'L', '+', '*', 0,   0,   '}', 'Z', 'X', 'C', 'V','B', 'N', 'M', '<', '>', '?', 0,   '*', 0,   ' ', 0,   0,   0,   0,   0,   0,0,   0,   0,   0,   0,   0,   0,   '7', '8', '9', '-', '4', '5', '6', '+', '1','2', '3', '0', '.', 0,   0,   0,   0,   0,   0,   0,   0,   0,   0,   0,   0,0,   0,   0,   0,   0,   0,   0,   0,   0,   0,   0,   0,   0,   0,   0,   0,0,   0,   0,   '_', 0,   0,   0,   0,   0,   0,   0,   0,   0,   '|', 0,   0};int key_to = 0, key_shift = 0;...for (;;) {io_cli();if (fifo32_status(&fifo) == 0) {task_sleep(task_a);io_sti();} else {i = fifo32_get(&fifo);io_sti();if (256 <= i && i <= 511) { /* 键盘数据 */sprintf(s, "%02X", i - 256);putfonts8_asc_sht(sht_back, 0, 16, COL8_FFFFFF, COL8_008484, s, 2);if (i < 0x80 + 256) { /* 将按键编码转换为字符编码 */if (key_shift == 0) {s[0] = keytable0[i - 256];} else {s[0] = keytable1[i - 256];}} else {s[0] = 0;}if (s[0] != 0) { /* 一般字符 */if (key_to == 0) {	/* 发送给任务A */if (cursor_x < 128) {/* 显示一个字符之后将光标后移一位 */s[1] = 0;putfonts8_asc_sht(sht_win, cursor_x, 28, COL8_000000, COL8_FFFFFF, s, 1);cursor_x += 8;}} else {	/* 发送给命令行窗口 */fifo32_put(&task_cons->fifo, s[0] + 256);}}if (i == 256 + 0x0e) {	/* 退格键 */...}if (i == 256 + 0x0f) {	/* Tab */...}if (i == 256 + 0x2a) {	/* 左 Shift ON */key_shift |= 1;}if (i == 256 + 0x36) {	/* 右 Shift ON */key_shift |= 2;}if (i == 256 + 0xaa) {	/* 左 Shift OFF */key_shift &= ~1;}if (i == 256 + 0xb6) {	/* 右 Shift OFF */key_shift &= ~2;}...} else if (512 <= i && i <= 767) {...} else if (i <= 1) { ...}
}

关于keytable0[ ]和keytable1[ ],考虑到顺便支持“\”和“_”的输入也不错,就让它一直支持到0x80的转换吧。在keytable1[ ]中,对于英文字母和小键盘的部分没有进行改动。
程序的原理是,先将按键编码转换成字符编码, 则向s[0]存入0。如果遇到无法转换的按键编码,则向s[0]存入0。剩下的部分没有什么难度,只要仔细读读程序应该就可以理解了吧。
在这里插入图片描述

虽然我们只修改了任务A,但在命令行窗口中也可以输入符号了,真不错。不过现在我们输入的英文字母都显示成大写,这个看着实在不舒服,因此我们需要实现小写字母的输入,一起看下一小节吧。

5.大小写字母

要实现区分大写、小写字母的输入,我们必须要 #同时判断Shif键的状态以及CapsLock的状态。
CapsLock 为 OFF & Shift健为0FF → 小写英文字母
CapsLock为OFF & Shift健为ON → 大写英文字
CapsLock 为ON & Shift健为OFF → 大写英文字
CapsLock为ON & Shift健为ON → 小写英文字母

我们可以将需要转换为小写字母的条件总结如下:
输人的字符为英文字母
“CapsLock为OFF&Shif键为OFF” 或者 “CapsLock为ON& Shif键为ON”

我们已经知道如何获取Shif键的状态, 但是CapsLock的状态要如何获取呢?BIOS知道 CapsLock 的状态, 可现在我们处于32位模式, 没办法向BIOS查询。不过别担心, asmhead.nas 中我们已经从BIOS获取到了键盘状态, 就保存在binfo→>leds中。
binfo->leds的第4位→ScrollLock状态
binfo->leds的第5位→ NumLock状态
binfo->leds的第6位→CapsLock状态
我们就可以处理大小写字母的输入了。 只要使用上述数据,

在i语句中,除了 && 运算符,还有一个 || 运算符,这个运算符代表“只要其中任意一个条件成立即可”的意思,我们可以用它来改写HariMain, 使其能够实现小写字母的输入。

void HariMain(void)
{...int key_to = 0, key_shift = 0, key_leds = (binfo->leds >> 4) & 7;...for (;;) {io_cli();if (fifo32_status(&fifo) == 0) {task_sleep(task_a);io_sti();} else {i = fifo32_get(&fifo);io_sti();if (256 <= i && i <= 511) { sprintf(s, "%02X", i - 256);putfonts8_asc_sht(sht_back, 0, 16, COL8_FFFFFF, COL8_008484, s, 2);if (i < 0x80 + 256) { /* 将按键编码转换为字符编码 */					...}else{s[0] = 0;}/* 这里开始 */if ('A' <= s[0] && s[0] <= 'Z') {	/* 当输入字符为英文字母时 */if (((key_leds & 4) == 0 && key_shift == 0) ||((key_leds & 4) != 0 && key_shift != 0)) {s[0] += 0x20;	/* 将大写字母转换为小写字母 */}}/* 这里结束 */if (s[0] != 0) {...} else if (512 <= i && i <= 767) {...} else if (i <= 1) {...}...}
}

在这里插入图片描述

呀,这样就变得酷多了吧。啥?不能满足于这点成绩?其实这样挺好的不是吗,一步一个脚印,享受每一次进步,这样才有成就感呀。我们已经实现了根据CapsLock的状态来切换大小写字母的输入, 那大家想不想实现在按下CapsLock键的时候切换CapsLock的状态呢?一定很想吧?

6.对各种锁定键的支持

好,让我们开始吧。回头再看一遍第14天的编码表(不想看表格的同学还是可以自己按键盘看编码哦),我们可以得到:
0x3a: CapsLock 0x45: NumLock 0x46: Scrolllock
只要将binfo->leds中对应的位置改写就可以了。因此当我们接收到上述按键编码时, 这和key_shift基本上是一样的,很容易实现。 到这里,我们已经实现了锁定键模式的切换,不过现在还是有一个问题,模式是可以切换了, 但是键盘上面的指示灯却不会发生变化。这样就可能会发生下述情况:明明CapsLock灯没亮,但在系统中却是处于CapsLock模式。这个问题我们最好想办法解决它。

关于LED的控制可采用下面的方法向键盘发送指令和数据。
■对于NumLock和CapsLock等LED的控制, 读取状态寄存器, 等待bit1的值变为0。
■向数据输出(0060)写入要发送的1个字节数据。
■等待键盘返回1个字节的信息,这和等待键盘输入所采用的方法相同(用IRQ等待或者用轮询状态寄存器bit1的值直到其变为0都可以)。
■返回的信息如果为0xfa,表明1个字节的数据已成功发送给键盘。如为0xfe则表明 发送失败,需要返回第1步重新发送。
■要控制LED的状态,需要按上述方法执行两次,向键盘发送EDxx数据。其中,xx的bit 0代表ScrollLock, bit 1代表NumLock,bit 2代表CapsLock (0表示熄灭,1表示点亮)。 bit 3~7为保留位, 置0即可。

有了这些信息,我们总算看到希望了, 于是我们写了以下程序。

#define KEYCMD_LED		0xedvoid HariMain(void)
{...struct FIFO32 fifo, keycmd;int fifobuf[128], keycmd_buf[32];...int key_to = 0, key_shift = 0, key_leds = (binfo->leds >> 4) & 7, keycmd_wait = -1;.../* 为了避免和键盘当前状态冲突,在一开始先进行设置 */fifo32_put(&keycmd, KEYCMD_LED);fifo32_put(&keycmd, key_leds);fifo32_init(&keycmd, 32, keycmd_buf, 0);for (;;) {/*从此开始*/if (fifo32_status(&keycmd) > 0 && keycmd_wait < 0) {/* 如果存在向健盘控制器发送的数据,则发送它 */keycmd_wait = fifo32_get(&keycmd);wait_KBC_sendready();io_out8(PORT_KEYDAT, keycmd_wait);}/*到此结束*/io_cli();if (fifo32_status(&fifo) == 0) {task_sleep(task_a);io_sti();} else {i = fifo32_get(&fifo);io_sti();if (256 <= i && i <= 511) {.../*从此开始*/if (i == 256 + 0x3a) {	/* CapsLock */key_leds ^= 4;fifo32_put(&keycmd, KEYCMD_LED);fifo32_put(&keycmd, key_leds);}if (i == 256 + 0x45) {	/* NumLock */key_leds ^= 2;fifo32_put(&keycmd, KEYCMD_LED);fifo32_put(&keycmd, key_leds);}if (i == 256 + 0x46) {	/* ScrollLock */key_leds ^= 1;fifo32_put(&keycmd, KEYCMD_LED);fifo32_put(&keycmd, key_leds);}if (i == 256 + 0xfa) {	/* 键盘成功接收到数据 */keycmd_wait = -1;}if (i == 256 + 0xfe) {	/* 键盘没有成功接收到数据 */wait_KBC_sendready();io_out8(PORT_KEYDAT, keycmd_wait);}...} else if (512 <= i && i <= 767) {...} else if (i <= 1) { ...}}
}		

程序的工作原理是这样的。首先,我们创建了一个叫keycmd的FIFO缓冲区,它不是用来接收中断请求的,而是用来管理由任务A向键盘控制器发送数据的顺序的。如果有数据要发送到键盘控制器,首先会在这个keycmd中累积起来。 keycnd_wait变量,用来表示向键盘控制器发送数据的状态。当keycmd_wait的值为-1时,表示键盘控制器处于通常状态,可以发送指令;当值不为-1时,表示键盘控制器正在等待发送的数据,这时要发送的数据被保存在keycmd_wait变量中。
在for循环的开头, 当keycmd中有数据, 且keycmd_wait为-1时,向键盘发送1个字节的数据, 在开始发送数据的同时,keycmd_wait变为非-1的值。随后, 当从键盘接收到0xfa返回信息时, 继续发送下一个数据。 当从键盘接收到的返回信息为0xfe时,则重新发 keycmd_wait恢复为-1, 送刚才的数据。 在for循环前面,我们向键盘控制器设置了指示灯的状态,也许这一段是可有可无的,不过这样可以保证key_leds的值和实际的键盘指示灯状态绝对不会发生冲突的情况, 因此保险起见还是设置了。

好了,我们来“make run”。本来想贴一张运行时的截图,不过这里发生变化的不是屏幕画 面,而是键盘的指示灯,所以很遗憾, 没有办法给大家展示这个令人感动的场面了。 嗯?不管怎么按CapsLock键, 都无法点亮指示灯,不过NumLock和 按Shift +CapsLock指示灯就亮了,好奇怪啊,我们明明不是这样设计的。
由于实在无法理解这一现象, 又重新 make run了harib14f,按下NumLock进行实验, harib14f中也可以点亮NumLock指示灯”。 看起来在这个QEMU模拟器中,键盘的指示灯貌似并不是由模拟器管理, 而是由Windows管理的。

总结

好,今天就到这里吧, 明天我们继续来做命令行窗口哦!

今天是小年,预热春节,祝大家小年安康!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/diannao/67839.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

R语言学习笔记之开发环境配置

一、概要 整个安装过程及遇到的问题记录 操作步骤备注&#xff08;包含遇到的问题&#xff09;1下载安装R语言2下载安装RStudio3离线安装pacman提示需要安装Rtools4安装Rtoolspacman、tidyfst均离线安装完成5加载tidyfst报错 提示需要安装依赖&#xff0c;试错逐步下载并安装…

数据结构 链表2

目录 前言&#xff1a; 一&#xff0c;反转一个链表(迭代) 二&#xff0c;打印一个链表&#xff08;递归&#xff09; 三&#xff0c;反转一个链表(递归) 四&#xff0c;双向链表 总结 前言&#xff1a; 我们根据 [文章 链表1] 可以知道链表相比较于数组的优缺点和计算机…

考研408笔记之数据结构(五)——图

数据结构&#xff08;五&#xff09;——图 1. 图的基本概念 1.1 图的定义 1.2 有向图和无向图 在有向图中&#xff0c;使用圆括号表示一条边&#xff0c;圆括号里元素位置互换没有影响。 在无向图中&#xff0c;使用尖括号表示一条边&#xff0c;尖括号里元素位置互换则表示…

游戏设备升级怎么选?RTX4070独显,ToDesk云电脑更具性价比

过新年、添喜气&#xff01;正逢节期来临不知道各位是否都跟小编一样在考虑购置生活中的各样所需呐&#xff1f; 25年可谓是3A游戏大作之年&#xff0c;例如《GTA6》《文明7》《死亡搁浅2》《刺客信条&#xff1a;影》下半年落地的《塞尔达传说&#xff1a;新篇章》《生化危机9…

C语言初阶牛客网刷题——HJ73 计算日期到天数转换【难度:简单】

1. 题目描述——HJ73 计算日期到天数转换 牛客网OJ题链接 描述 每一年中都有 12 个月份。其中&#xff0c;1,3,5,7,8,10,12 月每个月有 31 天&#xff1b; 4,6,9,11 月每个月有 30 天&#xff1b;而对于 2 月&#xff0c;闰年时有29 天&#xff0c;平年时有 28 天。 现在&am…

【深度学习基础】多层感知机 | 权重衰减

【作者主页】Francek Chen 【专栏介绍】 ⌈ ⌈ ⌈PyTorch深度学习 ⌋ ⌋ ⌋ 深度学习 (DL, Deep Learning) 特指基于深层神经网络模型和方法的机器学习。它是在统计机器学习、人工神经网络等算法模型基础上&#xff0c;结合当代大数据和大算力的发展而发展出来的。深度学习最重…

实现酷炫粒子背景效果

使用 particles.vue3 实现酷炫粒子背景效果 在这篇博客中&#xff0c;我们将介绍如何使用 particles.vue3 实现动态粒子背景&#xff0c;并详细讲解其配置参数和常见问题的解决方法。通过本文&#xff0c;你可以轻松在项目中应用并自定义粒子效果。 什么是 particles.vue3&am…

ubuntu16.04 VSCode下cmake+clang+lldb调试c++

VSCode下cmakeclanglldb调试c Ubuntu16.04 安装OpenCV4.5.4 文章目录 VSCode下cmakeclanglldb调试c1.安装clangclangdcmake2、打开VSCode&#xff0c;安装扩展插件3、编译4、Debug4.1 创建launch.json。4.2 配置setting.json 5. vscode安装配置clang-format插件5.1 Linux系统安…

在vue3中使用datav完整引入时卡在加载页面的解决方法

文件修改 文件&#xff1a;node_modules/dataview\datav-vue3/package.json // "module": "./es/index.js","module": "./es/index.mjs", // 修改后使用完整引入&#xff0c;需要为datav配置文件添加相应方法 文件&#xff1a;node…

AI agent 在 6G 网络应用,无人机群控场景

AI agent 在 6G 网络应用,无人机群控场景 随着 6G 时代的临近,融合人工智能成为关键趋势。借鉴 IT 行业 AI Agent 应用范式,提出 6G AI Agent 技术框架,包含多模型融合、定制化 Agent 和插件式环境交互理念,构建了涵盖四层结构的框架。通过各层协同实现自主环境感知等能力…

跨境电商SEO起步:关键词研究方法

SEO的重要性和必要性不言而喻&#xff0c;而在SEO的各大流程中&#xff0c;关键词研究同样重要&#xff0c;因为它在网站内容优化、产品标题和描述优化等方面都发挥重要作用。 一、从消费者视角出发 SEO是为了增加让消费者看到自己产品的可能性&#xff0c;因此要从消费者搜索…

开发环境搭建-1:配置 WSL (类 centos 的 oracle linux 官方镜像)

一些 Linux 基本概念 个人理解&#xff0c;并且为了便于理解&#xff0c;可能会存在一些问题&#xff0c;如果有根本上的错误希望大家及时指出 发行版 WSL 的系统是基于特定发行版的特定版本的 Linux 发行版 有固定组织维护的、开箱就能用的 Linux 发行版由固定的团队、社区…

【三维分割】Gaga:通过3D感知的 Memory Bank 分组任意高斯

文章目录 摘要一、引言二、主要方法2.1 3D-aware Memory Bank2.2 三维分割的渲染与下游应用 三、实验消融实验应用: Scene Manipulation 地址&#xff1a;https://www.gaga.gallery 标题&#xff1a;Gaga: Group Any Gaussians via 3D-aware Memory Bank 来源&#xff1a;加利福…

UE5 开启“Python Remote Execution“

demo 代码 remote_execution.py 远程调用UE5 python代码-CSDN博客 在启用 Unreal Engine 5&#xff08;UE5&#xff09;的“Python 远程执行”功能后&#xff0c;UE5 会启动一个 UDP 组播套接字服务&#xff0c;以监听来自外部应用程序的 Python 命令。 具体行为如下&#xf…

TangoFlux 本地部署实用教程:开启无限音频创意脑洞

一、介绍 TangoFlux是通过流匹配和 Clap-Ranked 首选项优化&#xff0c;实现超快速、忠实的文本到音频生成的模型。 本模型由 Stability AI 提供支持&#x1f680; TangoFlux 可以在单个 A40 GPU 上在 ~3 秒内生成长达 34.1kHz 的立体声音频。 二、部署 安装方式非常简单 1…

Python数据类型间的转换及eval函数

1.数据类型间的转换 x 10 y 3 z x / y # 除法运算&#xff0c;将运算的结果赋值给z print(z,type(z)) # 隐式转换&#xff0c;通过运算隐式地传了结果的类型# float类型转换为int类型&#xff0c;只保留整数部分&#xff0c;不会进行四舍五入 print(int(3.1542)) print(i…

influxdb+grafana+jmeter

influxdb influxd先启动 启动完成后执行 influxdb的端口号 grafana的启动 通过grafana-server.exe启动grafana 启动后打开 http://localhost:8087/

GeekHour

Linux Linux的是类Unix系统&#xff0c;作者是Linus&#xff0c;也是git的作者。符合GPL&#xff08;General Public License&#xff09;就可以Linux的使用、修改、再发布。 Linux四部分&#xff1a; 内核&#xff1a;驱动、内存管理、进程管理、文件系统、网络协议栈…。作…

【SpringCloud】黑马微服务学习笔记

目录 1. 关于微服务 ?1.1 微服务与单体架构的区别 ?1.2 SpringCloud 技术 2. 学习前准备 ?2.1 环境搭建 ?2.2 熟悉项目 3. 正式拆分 ?3.1 拆分商品功能模块 ?3.2 拆分购物车功能模块 4. 服务调用 ?4.1 介绍 ?4.2 RustTemplate?的使用 4.3 服务治理-注册中…

安装matlab2024a错误license checkout failed Error-8

问题&#xff1a; 忘记截图了&#xff0c;借用博主的图片。 记得安装过程中&#xff0c;目标网址才是你的安装地址&#xff0c;而不是前面的安装包地址。 解决方法&#xff1a; 1.将破解文件中"Crack\R2020a\bin\win64\matlab_startup_plugins\lmgrimpl"目录下的l…