草庐IT

myos3 大学生利用C++构建一个完整的操作系统之代码重构并实现键盘打字和鼠标移动

落子无悔! 2023-04-11 原文

myos1 大学生利用C++构建一个完整的操作系统打印helloworld
myos2 大学生利用C++构建一个完整的操作系统之响应键盘中断
myos3 大学生利用C++构建一个完整的操作系统之代码重构并实现键盘打字和鼠标移动

1. 按键中断

计算机自制操作系统(十六):中断—键盘驱动 - 知乎 (zhihu.com)

在中断IDT256个中断源产生的中断在IDT表中注册时, 全部指向了同一个中断服务程序 class InterruptHandler, 其实是不准确的, 因为并没有建立各自中断号和中断服务程序之间的一一对应关系, 所以接下来利用按键中断进行演示, 展示通过按键类继承中断服务器程序来实现按键的实际中断操作;

// keyboard.h
class KeyBoardDriver: public InterruptHandler { // 这里面继承中断服务程序, 在键盘初始化时
public:
    KeyBoardDriver(InterruptManager *manager);
    ~KeyBoardDriver();
    virtual uint32_t HandleInterrupt(uint32_t esp); // 继承与中断管理器中的中断处理方法
private:
    Port8Bit dataport; // 数据端口
    Port8Bit commandport;  // 命令端口
};

1.1 键盘中断的初始化

中断控制器8259
CPU
键盘按键(8048芯片)
8042芯片

键盘中断的具体流程就是当我们点击键盘时, 键盘上面的芯片8048会把键盘扫描码发送给主板上的8042, 而8042是按照字节码处理数据, 并存储到 输出缓冲区 , 然后8042会给中断代理 8059A发送中断信号, 这样中断控制器通过读取8042的输出缓冲区寄存器, 获得键盘的扫描码; 总的来讲, 操作系统其实是在对8042这个芯片进行数据的操作(读写)来识别中断的

动手写操作系统9----键盘&鼠标中断实现_大魔王-CSDN博客

键盘中断通过主8259A的IRQ1触发, 而CPU通过中断向量号来寻址待执行的中断代码

中 断 向 量 号 = 起 始 向 量 号 + 中 断 请 求 号 中断向量号 = 起始向量号 + 中断请求号 =+ H a r d w a r e I n t e r r u p t O f f s e t + 0 x 01 HardwareInterruptOffset+0x01 HardwareInterruptOffset+0x01

如此相当于在中断描述符表中注册了键盘中断对应的向量, 然后在重写中断处理函数HandleInterrupt就能实现按键中断

剩下的端口定义以及对端口的读写, 见i8042 键盘控制器-------详细介绍 - LinKArftc - 博客园 (cnblogs.com)

同时还需要有一个知识点 电脑键盘的通码与断码 :

类型:通码(make code)和断码(break code)。

  1. 当一个键被按下或持续按住时,键盘会将该键的通码发送给主机;
  2. 而当一个键被释放时,键盘会将该键的断码发送给主机。

根据键盘按键扫描码的不同,在此可将按键分为如下几类:

  1. 第一类按键,通码为1字节, 断码为 0 x F 0 + 通 码 形 式 0xF0+通码形式 0xF0+。如A键,其通码为 0 x 1 C 0x1C 0x1C,断码为 0 x F 0 , 0 x 1 C 0xF0, 0x1C 0xF0,0x1C
  2. 第二类按键,通码为2字节,$ 0xE0 + 0xNN$形式,断码为 0 x E 0 + 0 x F 0 + 0 x N N 0xE0+0xF0+0xNN 0xE0+0xF0+0xNN形式。如 right ctrl键,其通码为 0 x E 00 x 14 0xE0 0x14 0xE00x14,断码为 0 x E 00 x F 00 x 14 0xE0 0xF0 0x14 0xE00xF00x14,
  3. 第三类特殊按键有两个,print screen键通码为 0 x E 00 x 120 x E 00 x 7 C 0xE0 0x12 0xE0 0x7C 0xE00x120xE00x7C,断码为 0 x E 00 x F 00 x 7 C 0 x E 00 x F 00 x 12 0xE0 0xF0 0x7C 0xE0 0xF0 0x12 0xE00xF00x7C0xE00xF00x12; pause键通码为 0 x E 10 x 140 x 770 x E 10 x F 00 x 140 x F 00 x 77 0xE1 0x14 0x77 0xE1 0xF0 0x14 0xF0 0x77 0xE10x140x770xE10xF00x140xF00x77; 断码为空。

通码断码表如下:

根据上述知识点, 可以获得下面的对应端口读写来初始化键盘 :

// keyboard.cpp
#include "drivers/keyboard.h"
KeyBoardDriver::KeyBoardDriver(InterruptManager* manager) 
    : InterruptHandler(0x01 + manager->HardwareInterruptOffset(), manager),
    dataport(0x60),  // 这里设定键盘控制器的数据端口
    commandport(0x64){ // 这里设定键盘控制器的命令端口
    while(commandport.Read() & 0x01){ // 清空键盘的input_buffer
        dataport.Read();
    }
    commandport.Write(0xae); // 激活键盘
    commandport.Write(0x20); // 8042芯片, 要读取一个字节
    uint8_t status = (dataport.Read() | 1) & ~0x10; // 开启键盘中断
    commandport.Write(0x60); // 告诉控制器, 我要开始写入了
    dataport.Write(status);
    dataport.Write(0xf4);
    printf("finish keyboard init ! \n");
}

1.2 键盘中断的处理函数

这是一个简单的键盘中断的处理函数, 打印不同按键被按下和弹起后, 8042芯片所读到的字节码printf到屏幕上

// 这里会产生两次按键中断, 是因为啥呢, 是因为啥呢, 是因为按下和弹起
uint32_t KeyBoardDriver::HandleInterrupt(uint32_t esp) {
    printf("have keyboard Interrupt !!! \n");
    uint8_t key = dataport.Read();
    char* foo = (char*)"KEYBOARD 0X00  ";
    const char* hex = "0123456789ABCDEF";
    foo[11] = hex[(key >> 4) & 0x0f];
    foo[12] = hex[key & 0x0f];
    printf((const char*)foo);
    printf("\n");
    return esp;
}

1.3 结果

2. 鼠标中断

源码见我的gitee

2.1 鼠标中断初始化

其实鼠标中断和键盘中断是一样的, 都是通过对寄存器进行读写, 只不过鼠标中断的控制器在每次读写时, 还有三个字节的数据, 鼠标每一次动作都是3个字节数据,为什么是3个。想想也知道:两个坐标,一个状态;

#ifndef __MOUSE_H__
#define __MOUSE_H__

#include "common/types.h"
#include "hardware/interrupts.h"
#include "hardware/port.h"

using namespace myos;
using namespace myos::common;
using namespace myos::hardware;

// 鼠标中断是IQ2
class MouseDriver: public InterruptHandler {
public:
    MouseDriver(InterruptManager *manager);
    ~MouseDriver();
    virtual uint32_t HandleInterrupt(uint32_t esp); // 继承与中断管理器中的中断处理方法
private:
    Port8Bit dataport; // 数据端口
    Port8Bit commandport;  // 命令端口
    uint8_t buffer[3]; 
    // 每次mouse读取数据是数据流, 
    // 两种模式: 三个字节和四个字节的包, 这里是使用的是三个字节的包
    // xy的信息和左右中拿个按下的信息, x移动的距离, y移动的距离
    // 触发一次中断, 读取一个字节
    uint8_t offset; // 读取的哪个字节
    uint8_t buttons; // 哪个按键被按下

    int8_t x, y; // 鼠标位置初始化
};

这里面的初始化设置中间点显示白色


MouseDriver::MouseDriver(InterruptManager* manager) 
    : InterruptHandler(0x0C + manager->HardwareInterruptOffset(), manager),
    dataport(0x60),  // 这里设定键盘鼠标控制器的数据端口
    commandport(0x64),  // 这里设定键盘鼠标控制器的命令端口
    offset(0), 
    buttons(0), 
    x(40), y(12){  // 默认鼠标在中间

    uint16_t* VideoMemory = (uint16_t*)0xb8000;
    VideoMemory[y * 80 + x] = ((VideoMemory[y * 80 + x] & 0xf000) >> 4) |
                            ((VideoMemory[y * 80 + x] & 0x0f00) << 4) |
                            (VideoMemory[y * 80 + x] & 0x00ff); // 让中间的点变白

    commandport.Write(0xa8);
    commandport.Write(0x20);
    uint8_t status = (dataport.Read() | 2) & ~0x20; // 按位设置开启鼠标
    commandport.Write(0x60); // 告诉我们要些数据了
    dataport.Write(status); 
    commandport.Write(0xd4); // 写入鼠标寄存器写数据了
    dataport.Write(0xf4); // 告诉键盘或者数据开启鼠标
    dataport.Read(); // 读取数据流
    printf("finish mouse init ! \n");
}

2.2 中断处理函数

鼠标的中断处理函数
根据每次read到的三个字节的数据, 进行xy位置以及不同状态的显示

uint32_t MouseDriver::HandleInterrupt(uint32_t esp) {
    uint8_t status = commandport.Read();
    if (!(status & 0x20)) return esp; // 不是鼠标就直接返回esp

    buffer[offset] = dataport.Read(); // 先读一个数据
    offset = (offset + 1) % 3; // offset偏移一位

    // offset为0, 则表示x, y操作过了, 只需要控制一下x,y即可
    if (offset == 0) {
        uint16_t* VideoMemory = (uint16_t*)0xb8000;
        VideoMemory[y * 80 + x] = ((VideoMemory[y * 80 + x] & 0xf000) >> 4) |
                                ((VideoMemory[y * 80 + x] & 0x0f00) << 4) |
                                (VideoMemory[y * 80 + x] & 0x00ff);

        x += buffer[1];
        if (x < 0) x = 0;
        else if (x >= 80) x = 79;

        y -= buffer[2];
        if (y < 0) y = 0;
        else if (y >= 25) y = 24;

        VideoMemory[y * 80 + x] = ((VideoMemory[y * 80 + x] & 0xf000) >> 4) |
                                ((VideoMemory[y * 80 + x] & 0x0f00) << 4) |
                                (VideoMemory[y * 80 + x] & 0x00ff);

        for (uint8_t i = 0; i < 3; i++) {
            // 判断buf是否是按键按下, 如果按下就进行翻转一下, 这里只需要判断buffer[0]的下三位表示左中右, 
            if ((buffer[0] & (1 << i)) != (buttons & (1 << i))) {
                VideoMemory[y * 80 + x] = ((VideoMemory[y * 80 + x] & 0xf000) >> 4) |
                            ((VideoMemory[y * 80 + x] & 0x0f00) << 4) |
                            (VideoMemory[y * 80 + x] & 0x00ff);
            }
        }
        buttons = buffer[0];
    }
    return esp;
}

有关myos3 大学生利用C++构建一个完整的操作系统之代码重构并实现键盘打字和鼠标移动的更多相关文章

  1. ruby - 多次弹出/移动 ruby​​ 数组 - 2

    我的代码目前看起来像这样numbers=[1,2,3,4,5]defpop_threepop=[]3.times{pop有没有办法在一行中完成pop_three方法中的内容?我基本上想做类似numbers.slice(0,3)的事情,但要删除切片中的数组项。嗯...嗯,我想我刚刚意识到我可以试试slice! 最佳答案 是numbers.pop(3)或者numbers.shift(3)如果你想要另一边。 关于ruby-多次弹出/移动ruby​​数组,我们在StackOverflow上找到一

  2. ruby - 使用 Vim Rails,您可以创建一个新的迁移文件并一次性打开它吗? - 2

    使用带有Rails插件的vim,您可以创建一个迁移文件,然后一次性打开该文件吗?textmate也可以这样吗? 最佳答案 你可以使用rails.vim然后做类似的事情::Rgeneratemigratonadd_foo_to_bar插件将打开迁移生成的文件,这正是您想要的。我不能代表textmate。 关于ruby-使用VimRails,您可以创建一个新的迁移文件并一次性打开它吗?,我们在StackOverflow上找到一个类似的问题: https://sta

  3. ruby-on-rails - Rails - 一个 View 中的多个模型 - 2

    我需要从一个View访问多个模型。以前,我的links_controller仅用于提供以不同方式排序的链接资源。现在我想包括一个部分(我假设)显示按分数排序的顶级用户(@users=User.all.sort_by(&:score))我知道我可以将此代码插入每个链接操作并从View访问它,但这似乎不是“ruby方式”,我将需要在不久的将来访问更多模型。这可能会变得很脏,是否有针对这种情况的任何技术?注意事项:我认为我的应用程序正朝着单一格式和动态页面内容的方向发展,本质上是一个典型的网络应用程序。我知道before_filter但考虑到我希望应用程序进入的方向,这似乎很麻烦。最终从任何

  4. ruby-on-rails - 渲染另一个 Controller 的 View - 2

    我想要做的是有2个不同的Controller,client和test_client。客户端Controller已经构建,我想创建一个test_clientController,我可以使用它来玩弄客户端的UI并根据需要进行调整。我主要是想绕过我在客户端中内置的验证及其对加载数据的管理Controller的依赖。所以我希望test_clientController加载示例数据集,然后呈现客户端Controller的索引View,以便我可以调整客户端UI。就是这样。我在test_clients索引方法中试过这个:classTestClientdefindexrender:template=>

  5. ruby-on-rails - 如果 Object::try 被发送到一个 nil 对象,为什么它会起作用? - 2

    如果您尝试在Ruby中的nil对象上调用方法,则会出现NoMethodError异常并显示消息:"undefinedmethod‘...’fornil:NilClass"然而,有一个tryRails中的方法,如果它被发送到一个nil对象,它只返回nil:require'rubygems'require'active_support/all'nil.try(:nonexisting_method)#noNoMethodErrorexceptionanymore那么try如何在内部工作以防止该异常? 最佳答案 像Ruby中的所有其他对象

  6. ruby - 为什么 SecureRandom.uuid 创建一个唯一的字符串? - 2

    关闭。这个问题需要detailsorclarity.它目前不接受答案。想改进这个问题吗?通过editingthispost添加细节并澄清问题.关闭8年前。Improvethisquestion为什么SecureRandom.uuid创建一个唯一的字符串?SecureRandom.uuid#=>"35cb4e30-54e1-49f9-b5ce-4134799eb2c0"SecureRandom.uuid方法创建的字符串从不重复?

  7. ruby - 在 Ruby 中用键盘诅咒数组浏览 - 2

    我正在尝试在Ruby中制作一个cli应用程序,它接受一个给定的数组,然后将其显示为一个列表,我可以使用箭头键浏览它。我觉得我已经在Ruby中看到一个库已经这样做了,但我记不起它的名字了。我正在尝试对soundcloud2000中的代码进行逆向工程做类似的事情,但他的代码与SoundcloudAPI的使用紧密耦合。我知道cursesgem,我正在考虑更抽象的东西。广告有没有人见过可以做到这一点的库或一些概念证明的Ruby代码可以做到这一点? 最佳答案 我不知道这是否是您正在寻找的,但也许您可以使用我的想法。由于我没有关于您要完成的工作

  8. ruby-on-rails - Rails - 从另一个模型中创建一个模型的实例 - 2

    我有一个正在构建的应用程序,我需要一个模型来创建另一个模型的实例。我希望每辆车都有4个轮胎。汽车模型classCar轮胎模型classTire但是,在make_tires内部有一个错误,如果我为Tire尝试它,则没有用于创建或新建的activerecord方法。当我检查轮胎时,它没有这些方法。我该如何补救?错误是这样的:未定义的方法'create'forActiveRecord::AttributeMethods::Serialization::Tire::Module我测试了两个环境:测试和开发,它们都因相同的错误而失败。 最佳答案

  9. ruby-on-rails - 如何重命名或移动 Rails 的 README_FOR_APP - 2

    当我在我的Rails应用程序根目录中运行rakedoc:app时,API文档是使用/doc/README_FOR_APP作为主页生成的。我想向该文件添加.rdoc扩展名,以便它在GitHub上正确呈现。更好的是,我想将它移动到应用程序根目录(/README.rdoc)。有没有办法通过修改包含的rake/rdoctask任务在我的Rakefile中执行此操作?是否有某个地方可以查找可以修改的主页文件的名称?还是我必须编写一个新的Rake任务?额外的问题:Rails应用程序的两个单独文件/README和/doc/README_FOR_APP背后的逻辑是什么?为什么不只有一个?

  10. ruby - 用 Ruby 编写一个简单的网络服务器 - 2

    我想在Ruby中创建一个用于开发目的的极其简单的Web服务器(不,不想使用现成的解决方案)。代码如下:#!/usr/bin/rubyrequire'socket'server=TCPServer.new('127.0.0.1',8080)whileconnection=server.acceptheaders=[]length=0whileline=connection.getsheaders想法是从命令行运行这个脚本,提供另一个脚本,它将在其标准输入上获取请求,并在其标准输出上返回完整的响应。到目前为止一切顺利,但事实证明这真的很脆弱,因为它在第二个请求上中断并出现错误:/usr/b

随机推荐