《Linux内核设计的艺术:图解Linux操作系统架构设计与实现原理》——2.7 与建立人机交互界面相关的外设的中断服务程序挂接

简介: 本节书摘来自华章计算机《Linux内核设计的艺术:图解Linux操作系统架构设计与实现原理》一书中的第2章,第2.7节,作者:新设计团队著, 更多章节内容可以访问云栖社区“华章计算机”公众号查看。

2.7 与建立人机交互界面相关的外设的中断服务程序挂接

Linus在操作系统源代码中本来设计了chr_dev_init()函数,明显是要用这个函数初始化字符设备,但我们可以看到这是一个空函数。Linus又设计了tty_init()函数,内容就是初始化字符设备。有人解释tty是teletype。
字符设备的初始化为进程与串行口(可以通信、连接鼠标……)、显示器以及键盘进行I/O通信准备工作环境,主要是对串行口、显示器、键盘进行初始化设置,以及与此相关的中断服务程序与IDT挂接。在tty_init()函数中,先调用rs_init()函数来设置串行口,再调用con_init()函数来设置显示器,具体执行代码如下:

//代码路径:init/main.c:
void main(void)
{
    …
    tty_init();
    …
}
//代码路径:kernel/chr_dev/tty_io.c:
void tty_init(void)
{
    rs_init();
    con_init()
}

2.7.1 对串行口进行设置
把两个串行口中断服务程序与IDT相挂接,然后根据tty_table数据结构中的内容对这两个串行口进行初始化设置,包括设置线路控制寄存器的DLAB 位、设置发送的波特率因子、设置DTR和RTS……最后,允许主8259A 芯片的IRQ3和IRQ4 发送中断请求。
挂接的具体过程及挂接后的效果如图2-13所示。
执行代码如下:

//代码路径:kernel/chr_dev/serial.c:
void rs_init(void)
{
    set_intr_gate(0x24,rs1_interrupt);    //设置串行口1中断,参看2.5
    set_intr_gate(0x23,rs2_interrupt);    //设置串行口2中断
    init(tty_table[1].read_q.data);    //初始化串行口1
    init(tty_table[2].read_q.data);     //初始化串行口2
    outb(inb_p(0x21)&0xE7,0x21);        //允许IRQ3,IRQ4
}

两个串行口中断处理程序与IDT的挂接函数set_intr_gate()与2.5中介绍过的set_trap_gate()函数类似,可参看前面对set_trap_gate()函数的讲解。它们的差别是set_trap_gate()函数的type是15(二进制的1111),而set_intr_gate()的type是14(二进制的1110)。

image

2.7.2 对显示器进行设置
根据机器系统数据提供的显卡是“单色”还是“彩色”来设置配套信息。由于在Linux 0.11那个时代,大部分显卡器是单色的,所以我们假设显卡的属性是单色EGA。那么显存的位置就要被设置为0xb0000~0xb8000,索引寄存器端口被设置为0x3b4,数据寄存器端口被设置为0x3b5,再将显卡的属性——EGA这三个字符,显示在屏幕上。另外,再初始化一些用于滚屏的变量,其中包括滚屏的起始显存地址、滚屏结束显存地址、最顶端行号以及最低端行号。效果如图2-14所示。

image

2.7.3 对键盘进行设置
对键盘进行设置是先将键盘中断服务程序与IDT相挂接,然后取消8259A中对键盘中断的屏蔽,允许IRQ1发送中断信号,通过先禁止键盘工作、再允许键盘工作,键盘便能够使用了。键盘中断处理程序与IDT的挂接函数set_intr_gate()与前面讲解过的set_trap_gate()函数类似,参看对set_trap_gate()函数的讲解。
效果如图2-15所示。

image

执行代码如下:

//代码路径:kernel/chr_dev/console.c:
    …
#define ORIG_X            (*(unsigned char *)0x90000)
#define ORIG_Y            (*(unsigned char *)0x90001)
#define ORIG_VIDEO_PAGE        (*(unsigned short *)0x90004)
#define ORIG_VIDEO_MODE        ((*(unsigned short *)0x90006) & 0xff)
#define ORIG_VIDEO_COLS         (((*(unsigned short *)0x90006) & 0xff00) >> 8)
#define ORIG_VIDEO_LINES        (25)
#define ORIG_VIDEO_EGA_AX        (*(unsigned short *)0x90008)
#define ORIG_VIDEO_EGA_BX        (*(unsigned short *)0x9000a)
#define ORIG_VIDEO_EGA_CX        (*(unsigned short *)0x9000c)

#define VIDEO_TYPE_MDA        0x10 /* Monochrome Text Display    */
#define VIDEO_TYPE_CGA        0x11 /* CGA Display             */
#define VIDEO_TYPE_EGAM        0x20 /* EGA/VGA in Monochrome Mode    */
#define VIDEO_TYPE_EGAC        0x21 /* EGA/VGA in Color Mode        */

#define NPAR 16
    …
void con_init(void)
{
    register unsigned char a;
    char *display_desc= "????";
    char *display_ptr;

    video_num_columns= ORIG_VIDEO_COLS;//参看机器系统数据
    video_size_row= video_num_columns * 2;
    video_num_lines= ORIG_VIDEO_LINES;
    video_page= ORIG_VIDEO_PAGE; //参看机器系统数据
    video_erase_char= 0x0720;
    
    if (ORIG_VIDEO_MODE== 7)    /* Is this a monochrome display? */
    {
         video_mem_start= 0xb0000;
         video_port_reg= 0x3b4;
         video_port_val= 0x3b5;
         if ((ORIG_VIDEO_EGA_BX & 0xff) != 0x10) //参看机器系统数据
         {
               video_type= VIDEO_TYPE_EGAM;
               video_mem_end= 0xb8000;
               display_desc= "EGAm";
         }
         else
         {
               video_type= VIDEO_TYPE_MDA;
               video_mem_end    = 0xb2000;
               display_desc= "*MDA";
         }
    }
    else                        /* If not, it is color. */
    {
         video_mem_start= 0xb8000;
         video_port_reg    = 0x3d4;
         video_port_val    = 0x3d5;
         if ((ORIG_VIDEO_EGA_BX & 0xff) != 0x10) //参看机器系统数据
         {
               video_type= VIDEO_TYPE_EGAC;
               video_mem_end= 0xbc000;
               display_desc= "EGAc";
         }
         else
         {
               video_type= VIDEO_TYPE_CGA;
               video_mem_end= 0xba000;
               display_desc= "*CGA";
         }
    }

    /* Let the user known what kind of display driver we are using */
    
    display_ptr= ((char *)video_mem_start) + video_size_row - 8;
    while (*display_desc)
    {
         *display_ptr++= *display_desc++;
         display_ptr++;
    }
    
    /* Initialize the variables used for scrolling (mostly EGA/VGA)    */
    
    origin    = video_mem_start;
    scr_end    = video_mem_start + video_num_lines * video_size_row;
    top    = 0;
    bottom    = video_num_lines;

    gotoxy(ORIG_X,ORIG_Y);    //参看机器系统数据
    set_trap_gate(0x21,&keyboard_interrupt);//设置键盘中断,参看2.5节
    outb_p(inb_p(0x21)&0xfd,0x21);// 取消对键盘中断的屏蔽,允许IRQ1
    a=inb_p(0x61);
    outb_p(a|0x80,0x61);    // 禁止键盘工作
    outb(a,0x61);        //再允许键盘工作
}
相关文章
|
3天前
|
算法 Linux 开发者
深入探究Linux内核中的内存管理机制
本文旨在对Linux操作系统的内存管理机制进行深入分析,探讨其如何通过高效的内存分配和回收策略来优化系统性能。文章将详细介绍Linux内核中内存管理的关键技术点,包括物理内存与虚拟内存的映射、页面置换算法、以及内存碎片的处理方法等。通过对这些技术点的解析,本文旨在为读者提供一个清晰的Linux内存管理框架,帮助理解其在现代计算环境中的重要性和应用。
|
1天前
|
缓存 网络协议 Linux
Linux操作系统内核
Linux操作系统内核 1、进程管理: 进程调度 进程创建与销毁 进程间通信 2、内存管理: 内存分配与回收 虚拟内存管理 缓存管理 3、驱动管理: 设备驱动程序接口 硬件抽象层 中断处理 4、文件和网络管理: 文件系统管理 网络协议栈 网络安全及防火墙管理
18 4
|
3天前
|
人工智能 算法 大数据
Linux内核中的调度算法演变:从O(1)到CFS的优化之旅###
本文深入探讨了Linux操作系统内核中进程调度算法的发展历程,聚焦于O(1)调度器向完全公平调度器(CFS)的转变。不同于传统摘要对研究背景、方法、结果和结论的概述,本文创新性地采用“技术演进时间线”的形式,简明扼要地勾勒出这一转变背后的关键技术里程碑,旨在为读者提供一个清晰的历史脉络,引领其深入了解Linux调度机制的革新之路。 ###
|
5天前
|
算法 Linux 定位技术
Linux内核中的进程调度算法解析####
【10月更文挑战第29天】 本文深入剖析了Linux操作系统的心脏——内核中至关重要的组成部分之一,即进程调度机制。不同于传统的摘要概述,我们将通过一段引人入胜的故事线来揭开进程调度算法的神秘面纱,展现其背后的精妙设计与复杂逻辑,让读者仿佛跟随一位虚拟的“进程侦探”,一步步探索Linux如何高效、公平地管理众多进程,确保系统资源的最优分配与利用。 ####
28 4
|
6天前
|
缓存 负载均衡 算法
Linux内核中的进程调度算法解析####
本文深入探讨了Linux操作系统核心组件之一——进程调度器,着重分析了其采用的CFS(完全公平调度器)算法。不同于传统摘要对研究背景、方法、结果和结论的概述,本文摘要将直接揭示CFS算法的核心优势及其在现代多核处理器环境下如何实现高效、公平的资源分配,同时简要提及该算法如何优化系统响应时间和吞吐量,为读者快速构建对Linux进程调度机制的认知框架。 ####
|
17天前
|
运维 安全 Linux
Linux中传输文件文件夹的10个scp命令
【10月更文挑战第18天】本文详细介绍了10种利用scp命令在Linux系统中进行文件传输的方法,涵盖基础文件传输、使用密钥认证、复制整个目录、从远程主机复制文件、同时传输多个文件和目录、保持文件权限、跨多台远程主机传输、指定端口及显示传输进度等场景,旨在帮助用户在不同情况下高效安全地完成文件传输任务。
119 5
|
16天前
|
Linux
Linux系统之expr命令的基本使用
【10月更文挑战第18天】Linux系统之expr命令的基本使用
51 4
|
3天前
|
缓存 监控 Linux
|
7天前
|
Linux Shell 数据安全/隐私保护
|
8天前
|
域名解析 网络协议 安全