12 中断

news2025/1/24 17:40:59

12 中断

  • 1、内核中断编程
  • 2、顶半部和底半部机制
    • 2.1 任务的相关概念
      • 2.1.1 分类
      • 2.1.2 优先级
      • 2.1.3 进程调度
      • 2.1.4 休眠sleep
    • 2.2 顶半部和底半部实现机制
      • 2.2.1 顶半部特点
      • 2.2.2 底半部特点
      • 2.2.3 底半部实现方法之:tasklet
      • 2.2.4 底半部实现机制之工作队列
      • 2.2.5 底半部实现机制之软中断
      • 2.2.6 总结

1、内核中断编程

函数原型:

int request_irq(unsigned int irq, irq_handler_t handler, unsigned long flags, const char *name, void *dev)
- 功能:在linux内核中,处理器的任何一个硬件中断资源对于linux内核来说都是一种宝贵的资源,如果驱动想要访问某个硬件中断资源,必须先向内核申请这个硬件中断资源,一旦申请成功,然后向内核注册这个硬件中断对应的中断处理函数,一旦注册成功,静静等待着硬件中断触发,一旦触发将来内核自动调用注册的中断处理函数
所以此函数完成两个工作:
	1.申请硬件中断资源
	2.注册中断处理函数
- 参数:
	- irq:在linux内核中,linux内核给每个硬件中断都分配一个软件编号
	- handler:传递要注册的中断处理函数,其实就是传递要注册的中断处理函数名
	- flags:中断触发的类型
	- name:指定中断的名称

void free_irq(int irq, void *dev)
功能:如果驱动不再使用某个硬件中断资源,必须要释放这个硬件中断资源并且删除之前注册的中断处理函数
参数:
irq:传递要释放的硬件中断的中断号
dev:传递给中断处理函数的参数,此参数务必要和request_irq的最后一个参数保持一致,否则系统崩溃
  • 函数说明
中断号:中断号由GPIO编号经过gpio_to_irq函数进行换算而得:
	硬件GPIO		GPIO编号		     中断号
	GPIOA28		PAD_GPIO_A+28  	gpio_to_irq(PAD_GPIO_A + 28);
	GPIOB9		PAD_GPIO_B+9	gpio_to_irq(PAD_GPIO_B + 9);
	...	        	...	        	...
中断处理函数:一旦注册完毕,静静等待硬件中断触发,一旦触发,内核自动调用此函数
	编写一个中断处理函数例子:
	irqreturn_t 中断处理函数名(int  irq, void *dev) {
		//根据用户需求编写中断处理函数
		....
		return IRQ_NONE; //中断处理函数执行失败
		或者
		return IRQ_HANDLED; //中断处理函数执行成功	
	}
	注意中断处理函数形参问题:
	irq:保存当前触发中断的中断号,例如:GPIOA28产生的中断,irq=gpio_to_irq(PAD_GPIO_A+28);
	dev:保存给中断处理函数传递的参数
		问:谁来传递参数呢?
		答:request_irq函数的最后一个形参(void *dev)用来给中断处理函数传递参数
		而中断处理函数的第二个形参dev来保存传递的参数!
中断触发的类型:
	IRQF_TRIGGER_HIGH:指定为高电平触发
	IRQF_TRIGGER_LOW:指定为低电平触发
	IRQF_TRIGGER_FALLING:指定为下降沿触发
	IRQF_TRIGGER_RISING:指定为上升沿触发
	IRQF_TRIGGER_FALLING|IRQF_TRIGGER_RISING:指定为双边沿触发
	对于内部中断(就是各种控制器给中断控制器发送的中断,此中断线不可见),一律给0
  • 案例
#include <linux/init.h>
#include <linux/module.h>
#include <linux/irq.h>
#include <linux/interrupt.h>
#include <linux/gpio.h>
#include <mach/platform.h>

// 声明gpio
struct key_gpio{
	int gpio;
	char* name;
};
// 定义GPIO
static struct key_gpio key_info[]={
	{
		.name="key_1",
		.gpio=PAD_GPIO_A+28
	},
	{
		.name="key_2",
		.gpio=PAD_GPIO_B+9
	}
};
// 中断处理函数
static irqreturn_t key_interupt (int irq,void *dev){
	struct key_gpio * key = (struct key_gpio *)dev;
	//获取按键的状态
    	int kstate;
    	kstate = gpio_get_value(key->gpio);

    	//打印按键的状态
    	printk("%s: 按键[%s]的状态是[%s]\n", 
                        __func__, key->name, 
                        kstate?"松开":"按下");
    	return IRQ_HANDLED; //成功,失败返回IRQ_NONE
}
static int key_init(void){
	// 申请GPIO资源
	int i=0;
	for(i=0;i<ARRAY_SIZE(key_info);i++){
		gpio_request(key_info[i].gpio,key_info[i].name);
		gpio_direction_input(key_info[i].gpio);
		// 配置中断
		int irq = gpio_to_irq(key_info[i].gpio); // 生成中断号
	    request_irq(irq,key_interupt,IRQF_TRIGGER_FALLING|IRQF_TRIGGER_RISING,"key_interupt",&key_info[i]);// 申请硬件中断资源
		printk("%s init success\n",key_info[i].name);
	}
	return 0;
};

static void key_exit(void){	
	int i=0;
	for(i=0;i<ARRAY_SIZE(key_info);i++){
		gpio_free(key_info[i].gpio);
		int irq = gpio_to_irq(key_info[i].gpio); // 生成中断号
		free_irq(irq,&key_info[i]);// 释放硬件中断资源
		printk("%s free success\n",key_info[i].name);
	}
};
module_init(key_init);
module_exit(key_exit);
MODULE_LICENSE("GPL");
  • 前置要求
    配置linux内核去除官方的按键驱动:
make menuconfig
	Device Drivers->
		Input device support->
			<*>Keyboard->
				<*>..... //最后一个选项,这个选项对应的按键驱动就是官方的按键驱动
					按N键去除
保存退出

2、顶半部和底半部机制

2.1 任务的相关概念

2.1.1 分类

计算机中的任务细分三类:进程,硬件中断和软中断

  • 进程:进程一开始在用户空间运行,一旦调用系统调用立刻陷入内核空间继续运行,运行完毕又回到用户空间继续运行
  • 硬件中断:外设给CPU发送的中断信号,内核执行其硬件中断处理函数,此函数永远运行在内核空间
  • 软中断:软件单独调用swi或svc指令立刻触发软中断异常,立刻执行其软中断处理函数,此函数同样位于内核空间
    注意:任何一种任务要想运行必须先获取到CPU资源!

2.1.2 优先级

优先级越高,获取CPU资源的能力越强,越能够及早运行,在linux内核中,三类任务的优先级划分:

  • 硬件中断高于软中断,软中断高于进程
  • 进程之间存在优先级之分(nice命令可以设置优先级)
  • 软中断之间存在优先级
  • 硬件中断无优先级之分(哪怕中断控制器支持优先级,内核不认)

2.1.3 进程调度

inux内核会给每个进程分配时间片,进程一旦轮到其运行,在此时间片之内可以一直运行,直到时间片到期,进程调度器会将当前进程的CPU资源撤下给其他进程使用
切记:中断不参与进程调度,中断来了,直接会抢夺CPU资源,因为优先级高

2.1.4 休眠sleep

休眠只能用于进程,不可用于中断,当进程进入休眠时,进程会立马释放占用的CPU资源给其他进程

结论:由以上概念得到:在linux内核中,内核希望中断处理函数的执行速度要越快越好,如果中断处理函数长时间占用CPU资源,势必影响系统的并发能力(其他任务无法获取CPU资源投入运行)和响应能力(其他的硬件中断再也无法得到响应,各种卡)。
问:不可能所有的中断处理函数都能够满足内核的希望
例如:网卡利用中断获取网络数据包,其中断处理函数本身执行的速度就很慢,如果长时间占用CPU资源,就会造成丢包现象!
答:对于此种情况,必须要采用内核提供的顶半部和底半部机制来优化

2.2 顶半部和底半部实现机制

在这里插入图片描述

2.2.1 顶半部特点

  • 硬件中断触发,立刻执行
  • 顶半部本质就是中断处理函数,只是现在里面做紧急,耗时较短的内容
  • CPU在执行顶半部中断处理函数期间不允许发生CPU资源的切换
  • 顶半部同样不能休眠

2.2.2 底半部特点

  • 底半部对应的处理函数做不紧急,并且耗时较长的内容
  • CPU在"适当"的时候执行底半部的处理函数,在执行期间允许高优先级的任务来抢夺CPU资源,等处理完毕再回到底半部继续运行,所以底半部的处理函数实现可以基于软中断或者进程实现
    底半部实现方法三种:tasklet,工作队列,软中断
  • 底半部的本质就是延后执行的一种手段,并不是非要和顶半部配合使用,也就是可以单独用底半部来将你想要延后执行的事情进行延后,可以将宝贵的CPU资源给其他高优先级的任务使用

2.2.3 底半部实现方法之:tasklet

  • 特点:
    基于软中断实现,它对应的处理函数的优先级高于进程而低于硬件中断
    tasklet对应的处理函数中不能进行休眠操作
    tasklet对应的处理函数又称延后处理函数,里面做不紧急,耗时较长的内容
  • 描述tasklet属性的结构体
struct tasklet_struct {
	void (*func)(unsigned long data);
	unsigned long data;
	...
};	
- 功能:描述tasklet的属性
- func:指定tasklet的延后处理函数,将来内核会在适当时候调用其延后处理函数,不能进行休眠操作,形参data保存给这个延后处理函数传递的参数		
- data:给延后处理函数传递的参数
  • 相关函数
void tasklet_init(struct tasklet_struct *tasklet,void (*func)(unsigned long),unsigned long data)	
- 功能:初始化tasklet对象,给tasklet对象指定一个延后处理函数并且给延后处理函数传递参数
- 参数:
	- tasklet:tasklet对象的地址
	- func:给tasklet指定一个延后处理函数
	- data:给延后处理函数传递的参数
void tasklet_schedule(struct tasklet_struct *tasklet)
- 功能:向内核注册tasklet对象和其延后处理函数,一旦注册成功,内核会在适当的时候执行其延后处理函数
  • 案例:优化上面的按键中断,使里面的printk使用tasklet实现
// 底半部
static struct tasklet_struct key_tasklet;
struct key_gpio *key; // 指向按键按下的GPIO
// 定义tasklet处理函数
static void key_tasklet_function(unsigned long data){
	//获取按键的状态
	int kstate;
	kstate = gpio_get_value(key->gpio);
	//打印按键的状态
	printk("%s: 按键[%s]的状态是[%s]\n",__func__, key->name,kstate?"松开":"按下");
}
// 顶半部 中断处理函数
static irqreturn_t key_interupt (int irq,void *dev){
	// 保存按下的按键信息
	key = (struct key_gpio*) dev;
    	// 向内核注册tasklet延后处理函数
	tasklet_schedule(&key_tasklet);
	return IRQ_HANDLED; //成功,失败返回IRQ_NONE
}
static int key_init(void){
	// 申请GPIO资源
	int i=0;
	for(i=0;i<ARRAY_SIZE(key_info);i++){
		gpio_request(key_info[i].gpio,key_info[i].name);
		gpio_direction_input(key_info[i].gpio);
		// 配置中断
		int irq = gpio_to_irq(key_info[i].gpio); // 生成中断号
		request_irq(irq,key_interupt,IRQF_TRIGGER_FALLING|IRQF_TRIGGER_RISING,"key_interupt",&key_info[i]);// 申请硬件中断资源
		printk("%s init success\n",key_info[i].name);
	}
	// 初始化tasklet对象,指定延后处理函数
	tasklet_init(&key_tasklet,key_tasklet_function,0);
	return 0;
};

2.2.4 底半部实现机制之工作队列

  • 工作队列特点:
    工作队列基于进程实现,所以其延后处理函数可以进行休眠操作
    工作队列诞生的本质就是解决tasklet的延后处理函数不能休眠的问题,所以如果在延后执行的内容中有休眠操作,必须采用工作队列
    工作队列的延后处理函数的优先级也是最低的
  • 工作队列属性的数据结构:
struct work_struct {
	void (*func)(struct work_struct *work);
};
- func:工作队列的延后处理函数,基于进程实现,可以进行休眠操作
		形参work:指向驱动定义初始化的work_struct对象,work指向自己用于给延后处理函数传递参数
问:如何利用自己的地址给延后处理函数传递参数呢?
答:必须配合container_of宏来实现传递参数!
  • 相关函数
INIT_WORK(struct work_struct *work, void (*func)(struct work_struct *work));
- 功能:给work对象指定一个延后处理函数
- 参数:
	- work:work对象的地址
	- func:指定延后处理函数
schedule_work(struct work_struct *work);
- 功能:向内核登记注册一个延后处理函数,一旦注册成功,内核在适当的时候调用其延后处理函数
  • 案例:优化上面的按键中断,使里面的printk使用工作队列实现
// 底半部
static struct work_struct key_work;
struct key_gpio *key; // 指向按键按下的GPIO
// 定义work处理函数
static void key_work_function(struct work_struct *work){
	//获取按键的状态
	int kstate;
	kstate = gpio_get_value(key->gpio);
	//打印按键的状态
	printk("%s: 按键[%s]的状态是[%s]\n",__func__, key->name,kstate?"松开":"按下");
}
// 顶半部 中断处理函数
static irqreturn_t key_interupt (int irq,void *dev){
	// 保存按下的按键信息
	key = (struct key_gpio*) dev;
    // 向内核注册工作队列延后处理函数
	schedule_work(&key_work);
	return IRQ_HANDLED; //成功,失败返回IRQ_NONE
}
static int key_init(void){
	// 申请GPIO资源
	int i=0;
	for(i=0;i<ARRAY_SIZE(key_info);i++){
		gpio_request(key_info[i].gpio,key_info[i].name);
		gpio_direction_input(key_info[i].gpio);
		// 配置中断
		int irq = gpio_to_irq(key_info[i].gpio); // 生成中断号
		request_irq(irq,key_interupt,IRQF_TRIGGER_FALLING|IRQF_TRIGGER_RISING,"key_interupt",&key_info[i]);// 申请硬件中断资源
		printk("%s init success\n",key_info[i].name);
	}
	// 初始化work对象,指定延后处理函数
	INIT_WORK(&key_work,key_work_function);
	return 0;
};

2.2.5 底半部实现机制之软中断

  • 软中断特点:
    软中断同样有对应的延后处理函数,此函数可以同时运行在多个CPU核上,效率极其高,这就要求其延后处理函数如果进行全局变量的访问,记得要做好互斥保护,当一个CPU核在访问全局变量是,禁止其他CPU核访问,但是这种互斥保护势必降低代码的效率
    tasklet基于软中断实现,但是它的延后处理函数同一时刻使能运行在一个CPU核上也不用关心互斥访问的问题!
    软中断的延后处理函数不能insmod和rmmod动态的安装和卸载,必须和uImage编写在一起编译在一起,无形加大了开发的难度和加大了代码的维护难度,但是tasklet就可以动态的insmod和rmmod

2.2.6 总结

  • tasklet基于软中断实现,工作队列基于进程实现
  • 如果要进行延后执行,并且延后执行的内容中有休眠操作,只能用工作队列
  • 如果要进行延后执行,延后执行的内容中无休眠操作,但又要考虑效率问题,建议使用tasklet
  • 如果要进行延后执行,延后执行的内容中无休眠操作,但又不考虑效率问题,建议使用工作队列
  • 如果要进行延后执行,延后执行的内容中无休眠操作,并且对效率要求极高,并且想让多核同时运行处理,建议采用软中断,如果有对全局变量共享资源的同时访问,建议还是tasklet吧!

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

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

相关文章

微信会员卡怎么做_便捷生活新风尚

在这个快节奏的时代&#xff0c;每一次消费不仅是简单的商品交换&#xff0c;更是一场关于个性、便捷与尊享的深刻体验。随着数字化浪潮的汹涌澎湃&#xff0c;微信会员卡应运而生&#xff0c;它以独特的魅力&#xff0c;重新定义了商家与顾客之间的关系&#xff0c;开启了一场…

网鼎杯-2018-Web-Unfinish

先尝试万能注入&#xff1a; 如果万能注入缺少符号&#xff0c;如果加符又进不去&#xff0c;那我们尝试扫描文件,然后发现有一个register.php的文件&#xff0c;应该是注册页面&#xff0c;我们去打开 知道存储的文件&#xff0c;并利用状态码进行过滤 我们注册的用户名就是aa…

鼠害监测系统:科技守护农业安全

在农业生产中&#xff0c;鼠害一直是威胁作物安全、影响产量的重要因素。然而&#xff0c;随着科技的飞速发展&#xff0c;鼠害监测系统正逐步成为现代农业防治鼠害的重要利器。 鼠害监测系统巧妙融合了现代光电、数控及物联网技术&#xff0c;实现了诱鼠、投喂鼠药、鼠情监测及…

RoBERTa-www-ext 解读及使用方法

背景&#xff1a;模型及名字 hfl/chinese-roberta-wwm-ext 是 Hugging Face 提供的一个中文预训练模型&#xff0c;它是 RoBERTa 的一个变种&#xff0c;特别针对中文进行了优化。这个模型的名称中的“wwm”代表“word-level whole word masking”&#xff0c;意味着在预训练过…

爬虫案例3——爬取彩票双色球数据

简介&#xff1a;个人学习分享&#xff0c;如有错误&#xff0c;欢迎批评指正 任务&#xff1a;从500彩票网中爬取双色球数据 目标网页地址&#xff1a;https://datachart.500.com/ssq/ 一、思路和过程 目标网页具体内容如下&#xff1a; ​​​​​ 我们的任务是将上图中…

数学建模——启发式算法(蚁群算法)

算法原理 蚁群算法来自于蚂蚁寻找食物过程中发现路径的行为。蚂蚁并没有视觉却可以寻找到食物&#xff0c;这得益于蚂蚁分泌的信息素&#xff0c;蚂蚁之间相互独立&#xff0c;彼此之间通过信息素进行交流&#xff0c; 从而实现群体行为。 蚁群算法的基本原理就是蚂蚁觅食的过程…

一套完整的NVR方案与部分NVR录像机GUI源码剖析

一、部分功能展示 1.1 通道管理部分 在NVR系统中&#xff0c;通道管理是核心功能之一。通过通道管理&#xff0c;用户可以对连接的摄像头进行配置和监控。 通道连接使能&#xff1a;用户可以选择开启或关闭特定通道的连接功能&#xff0c;以实现灵活的设备管理。 时间同步&…

Aigtek高压功率放大器指标参数是什么

高压功率放大器是一种用于将电信号放大到较高电压水平的设备。其指标参数对于选择、设计和应用都至关重要。以下是一些常见的高压功率放大器指标参数&#xff0c;详细了解这些参数可以帮助工程师更好地了解设备的性能和适用范围。 电压增益&#xff1a; 电压增益是功率放大器输…

每日一题,力扣leetcode Hot100之189.轮转数组

解题思路&#xff0c;把数组轮换想成栈的出栈后又入栈即可&#xff0c;当然要判断好K的量&#xff0c;因为K有可能会超过数组长度 class Solution:def rotate(self, nums: List[int], k: int) -> None:"""Do not return anything, modify nums in-place ins…

25款极氪007上市,小米SU7就不该买?

文 | AUTO芯球 作者 | 谦行 我是刚刚才知道 买小米SU7的原来是盯着他这两个功能 可爱的小女孩喊小爱同学帮她停个车 妈妈给她说SU7自己能停好&#xff0c;她还叮嘱一句“小爱同学你给我好好停” SU7滴溜溜的就停在车位上&#xff0c;全程不到一分钟 视频属实温馨&#x…

剪辑新手必备:2024年爱剪辑官网免费版下载指南

嘿&#xff0c;朋友们&#xff01;现在视频在我们生活中就像空气一样&#xff0c;无处不在。无论是记录日常生活的小片段&#xff0c;还是制作一些有趣的视频内容&#xff0c;一款好的剪辑软件都特别关键。今天咱们就来聊聊2024年那些特别受欢迎的剪辑软件&#xff0c;顺便推荐…

Linux---03---网络及防火墙

课程回顾 虚拟安装 文件命令 本章重点 网络相关概念 静态网络配置 一、网络 1.1 什么是网络&#xff1f; 计算机网络是继电信网络、有线电视网络之后出现的世界级大型网络。 计算机网络由若干个结点和连接这些结点的链路组成。 网络中的结点可以是计算机、交换机、路…

虚幻5|给攻击添加特效

一&#xff0c;打开武器蓝图 选择武器网格体&#xff0c;在细节处找到组件开始重叠&#xff0c;点击 写下以下蓝图&#xff0c;这是最终蓝图&#xff0c;后面会分讲要点 二&#xff0c;actor拥有标签&#xff0c;就是被击打的敌人&#xff0c;我们给actor添加标签 到主界面&am…

Linux之sed命令和正则表达式

什么是sed编辑器? sed是一个命令行文本编辑工具&#xff0c;用于对文本进行处理和转换。它可以读取文本文件&#xff0c;对文件的各个行进行修改、删除和替换操作&#xff0c;并将结果输出到标准输出或者文件中。 sed 被广泛用于Unix和类Unix系统中的脚本和命令行操作中&#…

二极管作用

防止电源反接 电路目的是为了&#xff0c;防止电源反接&#xff0c;对电路中的电子元器件造成破坏&#xff0c;造成财产损失或者人身安全。 原理 二极管单向导电性&#xff0c;二极管0.7V正向压降不会对电路造成影响 原理图 这里U7是一个二极管&#xff0c;如果电源反接&…

C++STL初阶(10):list的简易实现(下)

在上一文中我们完成了链表的多数基本接口&#xff0c;本文主要围绕构造函数进行补充 1. 链表的拷贝 在前文中我们没有手动实现拷贝构造&#xff0c;所以使用的就是编译器自动生成的浅拷贝 先使用一下编译器自动生成的浅拷贝&#xff1a; 我们在打印li2之前给li1加入一个数据&…

Python 如何使用 Contextlib 模块

Python 中的 contextlib 模块提供了一些实用工具&#xff0c;帮助我们管理上下文管理器和与上下文相关的操作。上下文管理器是一种对象&#xff0c;它定义了进入和退出代码块时要执行的操作&#xff0c;通常用于资源管理&#xff0c;如文件操作、网络连接等。上下文管理器通常与…

【SPIE出版】第四届计算机视觉、应用与算法国际学术会议(CVAA 2024,10月11-13)

计算机视觉、应用与算法的领域&#xff0c;一直在飞速发展&#xff0c;第四届计算机视觉、应用与算法国际学术会议&#xff08;CVAA 2024) 将汇聚世界各地的顶尖学者、研究人员和企业代表&#xff0c;共同分享和交流计算机视觉在各个领域的最新研究成果、技术突破和产业应用。 …

ElementPlus table上移下移操作、表格嵌套树选择器

步骤条圆圈中的数字根据所选样式展示&#xff1a; <el-stepsstyle"margin-top: 20px; max-width: 700px"align-center:active"formModel.testData.length 1"><el-steptitle"Step 1"v-for"(item, index) in formModel.testData&qu…

技术应用 | 外语专业如何借助大模型转型升级?

一、选哪一条路&#xff1a;评测大模型、应用大模型、研发大模型 如果把大语言模型当作是“一个人”&#xff0c;那么既可以把这个“人”当作研究对象&#xff0c;研究它几岁了、智商如何了、能做什么、不能做什么、危不危险&#xff0c;也就是“评测大模型”&#xff1b;也可…