【genius_platform软件平台开发】第七十讲:linux系统驱动开发之-中断号、中断向量和中断处理函数

本文主要是介绍【genius_platform软件平台开发】第七十讲:linux系统驱动开发之-中断号、中断向量和中断处理函数,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

在这里插入图片描述

1. 概述

  • 假如存在这样一个需求:应用程序需要监控某个硬件GPIO口的电平状态,当发生变化时,应用程序就做出相应的动作。利用之前已经介绍的知识,是可以完成这个需求的。比如:在驱动程序中不停的读取GPIO口的状态,一旦发生变化,就把新的电平状态通过信号发送到应用层。这样的方式称作:轮询
  • 轮询方式的缺点显而易见:轮询的时间间隔应该是多少毫秒(or 微秒),才比较合适呢?
    轮询太慢:可能会丢失信号;轮询太快:消耗 CPU 资源!
    因此,在实际的产品中,用中断触发的方式才是更切合实际的选择!
    本文所有的描述和测试,都是在 x86 平台上完成的;

2. 中断分类

  • Linux 的版本在持续更新,对中断的处理方式也在不停的发生变化。下面几张图,是以前在学习时画的思维导图。这几张图比较清晰地描述了在Linux操作系统中,关于中断的一些基本概念。
    在这里插入图片描述

  • 这张图的结构还是比较清晰的,基本上概括了Linux系统中的中断分类。
    另外,在很多关于中断的书籍中,大部分都是从基础的 PIC(可编程中断控制器)开始讲解的。

3. 中断号和中断向量

  • 中断号与中断控制器(PIC/APIC)相关
  • 中断向量与CPU相关
    在这里插入图片描述

4. 中断处理

  • 中断服务程序,就是针对每一个中断如何进行处理。如果您了解Linux中断的相关内容,一定会看到这样的描述:中断处理分为上半部分和下半部分。
    在这里插入图片描述

4.1 上半部

  • 上半部分不能消耗太多的时间,主要处理与硬件相关的重要工作`;其他不重要的工作,都放在下半部分去做。

4.2 下半部

  • 下列图示是针对每一种“下半部分”处理机制的一些特点,注意:有些机制在新版本中已经废弃不用了,了解即可

4.2.1 软中断(soft_irq)

在这里插入图片描述

4.2.2 小任务(tasklet)

在这里插入图片描述

4.2.3 工作队列(work_queue)

在这里插入图片描述

  • 用来完成下半部分工作有好几种机制可以选择,每一种方式都是针对不同的需求场景。在每一种下半部分机制中,Linux都设计了非常方便的接口函数。作为开发者的我们来说,使用这些下半部分的机制很简单,只需要几个函数调用即可。例如:如果使用工作队列来实现下半部分的工作,只需要2步动作

4.3 注册中断函数

  • 中断注册,就是告诉操作系统:我对哪个中断感兴趣。当这些中断发生的时候,请通知我。通知的方式就是:调用一个预先注册好的回调函数。驱动程序可以通过函数 request_irq(),向操作系统注册,并且激活指定的中断线
int request_irq(unsigned int irq, irq_handler_t handler, unsigned long flags, const char *devname, void *dev_id);
参数说明:
irq: 申请的硬件中断号;
handler: 中断处理函数。一旦中断发生,这个函数就被调用;
flags: 中断的属性,例如:IRQF_DISABLED,IRQF_TIMER,IRQF_SHARED;
devname: 中断驱动程序的名称,在 /proc/interrupts 文件中看到对应的内容;
dev_id: 中断程序的唯一标识,比如:在共享中断中,可以用来区分不同的中断处理程序;

驱动程序通过函数 free_irq(),向操作系统注销一个中断处理函数:

void free_irq(unsigned int irq, void *dev_id);
参数说明:
irq: 硬件中断号;
dev_id: 中断程序的唯一标识;
  • 在中断处理函数中,注册函数
INIT_WORK(&mywork, mywork_handler);                                                         
schedule_work(&mywork);

4.4 定义处理函数

static struct work_struct mywork;static void mywork_handler(struct work_struct *work)
{printk("This is myword_handler...\n");
}

4.5 捕获键盘中断

示例代码,捕获键盘的中断,在中断处理函数中,打印出按键的扫描码,如果是 ESC 键被按下,就打印出指定的信息。操作的目录位于:/linux-4.15/drivers 目录下。

$ mkdir my_driver_interrupt
$ touch driver_interrupt.c

4.5.1 driver_interrupt.c

#include <linux/kernel.h>
#include <linux/module.h>
#include <linux/interrupt.h>// 中断号
static int irq; // 驱动程序名称 
static char * devname;          // 用来接收加载驱动模块时传入的参数
module_param(irq, int, 0644);
module_param(devname, charp, 0644);// 定义驱动程序的 ID,在中断处理函数中用来判断是否需要处理            
#define MY_DEV_ID           1211// 驱动程序数据结构
struct myirq
{int devid;
};// 保存驱动程序的所有信息
struct myirq mydev  ={ MY_DEV_ID };// 键盘相关的 IO 端口
#define KBD_DATA_REG        0x60  
#define KBD_STATUS_REG      0x64
#define KBD_SCANCODE_MASK   0x7f
#define KBD_STATUS_MASK     0x80// 中断处理函数
static irqreturn_t myirq_handler(int irq, void * dev)
{struct myirq mydev;unsigned char key_code;mydev = *(struct myirq*)dev;    // 检查设备 id,只有当相等的时候才需要处理if (MY_DEV_ID == mydev.devid){// 读取键盘扫描码key_code = inb(KBD_DATA_REG);/* 这里如果放开,每次按键都会打印出很多信息printk("key_code: %x %s\n",key_code & KBD_SCANCODE_MASK,key_code & KBD_STATUS_MASK ? "released" : "pressed");*/// 判断:是否为 ESC 键if (key_code == 0x01){printk("EXC key is pressed! \n");}}   return IRQ_HANDLED;
}// 驱动模块初始化函数
static int __init myirq_init(void)
{printk("myirq_init is called. \n");// 注册中断处理函数if(request_irq(irq, myirq_handler, IRQF_SHARED, devname, &mydev)!=0){printk("register irq[%d] handler failed. \n", irq);return -1;}printk("register irq[%d] handler success. \n", irq);return 0;
}// 驱动模块退出函数
static void __exit myirq_exit(void)
{printk("myirq_exit is called. \n");// 注销中断处理函数free_irq(irq, &mydev);
}MODULE_LICENSE("GPL");
module_init(myirq_init);
module_exit(myirq_exit);
  • 上面的代码,有两个小的知识点。

4.6 加载驱动时传参

示例代码中,在调用 request_irq 时,需要指定中断号和驱动程序的名称。这两个参数是在加载驱动模块的时候,从命令行传入的。在驱动程序中,通过下面两行代码即可实现参数的接收:

module_param(irq, int, 0644);
module_param(devname, charp, 0644);

module_param 是一个宏定义,定义在 include/linux/moduleparam.h 文件中,具体定义如下:

#define module_param(name, type, perm)                module_param_named(name, name, type, perm);

name: 存储参数的变量名;
type: 变量的类型;
perm: 访问参数的权限,表示此参数在sysfs文件系统中所对应的文件节点的属性;

4.8 IO地址

读取 IO 外设的两种不同方式:IO内存和IO端口

4.8.1 IO内存

4.8.1 IO端口

IO 端口有两种编址方式:统一编址和独立编址。

4.8.1.1 统一编址

  • 把主存单元所在的地址空间,划出一部分出来,专门用来把IO外设寄存器的地址映射到这部分划出来的内存地址空间中。统一编址的好处是:读取IO外设的时候,就好像读取普通的内存地址空间中的数据一样。

4.8.1.2 独立编址

  • IO 外设的地址空间,与主存单元的地址空间是两个独立的地址空间,此时,IO地址一般称作: IO端口。我们在读写IO外设的时候,从这些 “IO端口” 中读写就可以了。不同的外设,被分配了不同的 IO 端口号。CPU 提供了一些列函数来读写 IO 端口,例如:
// 读写一个字节
unsigned inb(unsigned port);
void outb(unsigned char byte, unsigned port);// 读写一个字
unsigned inw(unsigned port);
void outw(unsigned short word, unsigned port);

4.7 编译、验证

编译驱动模块:

$ make
输出文件:driver_interrupt.ko

因为我们捕获的是键盘中断(中断号:1),先看一下在加载驱动模块之前的中断驱动程序 head /proc/interrupts
在这里插入图片描述
可以把 demsg 的输出也清理一下:dmesg -c
执行下面指令来加载驱动模块(传递2个参数)insmod driver_interrupt.ko irq=1 devname=myirq再次执行一下指令 head /proc/interrupts 查看驱动程序:
在这里插入图片描述
中断号 1 的右侧,是不是看到了我们的驱动程序:my_irq再来看一下 dmesg 的输出信息:
在这里插入图片描述
成功注册了中断号1的处理函数!
此时,按几次键盘左上角的 ESC 键,然后再查看 dmesg 的输出信息:

这篇关于【genius_platform软件平台开发】第七十讲:linux系统驱动开发之-中断号、中断向量和中断处理函数的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/525040

相关文章

VSCode开发中有哪些好用的插件和快捷键

《VSCode开发中有哪些好用的插件和快捷键》作为全球最受欢迎的编程工具,VSCode的快捷键体系是提升开发效率的核心密码,:本文主要介绍VSCode开发中有哪些好用的插件和快捷键的相关资料,文中... 目录前言1、vscode插件1.1 Live-server1.2 Auto Rename Tag1.3

Agent开发核心技术解析以及现代Agent架构设计

《Agent开发核心技术解析以及现代Agent架构设计》在人工智能领域,Agent并非一个全新的概念,但在大模型时代,它被赋予了全新的生命力,简单来说,Agent是一个能够自主感知环境、理解任务、制定... 目录一、回归本源:到底什么是Agent?二、核心链路拆解:Agent的"大脑"与"四肢"1. 规划模

Linux内核定时器使用及说明

《Linux内核定时器使用及说明》文章详细介绍了Linux内核定时器的特性、核心数据结构、时间相关转换函数以及操作API,通过示例展示了如何编写和使用定时器,包括按键消抖的应用... 目录1.linux内核定时器特征2.Linux内核定时器核心数据结构3.Linux内核时间相关转换函数4.Linux内核定时

Linux镜像文件制作方式

《Linux镜像文件制作方式》本文介绍了Linux镜像文件制作的过程,包括确定磁盘空间布局、制作空白镜像文件、分区与格式化、复制引导分区和其他分区... 目录1.确定磁盘空间布局2.制作空白镜像文件3.分区与格式化1) 分区2) 格式化4.复制引导分区5.复制其它分区1) 挂载2) 复制bootfs分区3)

Python+wxPython开发一个文件属性比对工具

《Python+wxPython开发一个文件属性比对工具》在日常的文件管理工作中,我们经常会遇到同一个文件存在多个版本,或者需要验证备份文件与源文件是否一致,下面我们就来看看如何使用wxPython模... 目录引言项目背景与需求应用场景核心需求运行结果技术选型程序设计界面布局核心功能模块关键代码解析文件大

C++多线程开发环境配置方法

《C++多线程开发环境配置方法》文章详细介绍了如何在Windows上安装MinGW-w64和VSCode,并配置环境变量和编译任务,使用VSCode创建一个C++多线程测试项目,并通过配置tasks.... 目录下载安装 MinGW-w64下载安装VS code创建测试项目配置编译任务创建 tasks.js

Linux服务器数据盘移除并重新挂载的全过程

《Linux服务器数据盘移除并重新挂载的全过程》:本文主要介绍在Linux服务器上移除并重新挂载数据盘的整个过程,分为三大步:卸载文件系统、分离磁盘和重新挂载,每一步都有详细的步骤和注意事项,确保... 目录引言第一步:卸载文件系统第二步:分离磁盘第三步:重新挂载引言在 linux 服务器上移除并重新挂p

python3中正则表达式处理函数用法总结

《python3中正则表达式处理函数用法总结》Python中的正则表达式是一个强大的文本处理工具,用于匹配、查找、替换等操作,在Python中正则表达式的操作主要通过内置的re模块来实现,这篇文章主要... 目录前言re.match函数re.search方法re.match 与 re.search的区别检索

Linux下屏幕亮度的调节方式

《Linux下屏幕亮度的调节方式》文章介绍了Linux下屏幕亮度调节的几种方法,包括图形界面、手动调节(使用ACPI内核模块)和外接显示屏调节,以及自动调节软件(CaliseRedshift和Reds... 目录1 概述2 手动调节http://www.chinasem.cn2.1 手动屏幕调节2.2 外接显

Linux(centos7)虚拟机没有IP问题及解决方案

《Linux(centos7)虚拟机没有IP问题及解决方案》文章介绍了在CentOS7中配置虚拟机网络并使用Xshell连接虚拟机的步骤,首先,检查并配置网卡ens33的ONBOOT属性为yes,然后... 目录输入查看ZFhrxIP命令:ip addr查看,没有虚拟机IP修改ens33配置文件重启网络Xh