当前位置: 首页 > news >正文

Linux中断下半部分:软中断,tasklet和工作队列

为什么要有下半部分

  1. 中断会打断其他程序,为了打断其他程序时间短,就需要中断处理程序快。
  2. 执行中断处理程序后,相同中断不会触发,甚至所有中断都不能触发(设置IRQF_DISABLED,其他硬件与操作系统无法通信)
  3. 中断上下文下不能阻塞
    所以将中断分为上下部分,上部分处理反应很快的部分,下半部分处理对时间要求宽松的事件。
    上半部分需要处理硬件,比如将网卡接收的数据包复制到操作系统的缓存区。
    上半部分保证不被中断。
    其他所有都放在下部分。

下半部分的运行时机

通常下半部分在中断处理程序一返回就会马上运行,下半部分执行的关键是,它们运行时允许中断。
如果我们需要在一段时间后允许,我们可以使用内核定时器实现下半部分的任务。

内核提供的下半部分的实现方式

现在能用的有软中断,tasklet,工作队列。

软中断

软中断使用较少(相对于tasklet),但是tasklet是基于软中断实现的。
软中断是在编译期间静态分配的,不像tasklet那样能动态地注册或者注销。
内核可以注册32个软中断,而当前内核版本只注册了9个。

软中断结构体:
struct softirq_action{void (*action)(struct softirq_action*);
};
内核中的全局变量:
static struct softirq_action softirq_vec[NR_SOFTIRQS];//32个软中断
软中断处理函数action函数原型:
void softirq_handler(struct softirq_action *);

当只有一个处理器上,一个软中断不会抢占另一个软中断,唯一可以抢占软中断的是中断处理程序。不过,软中断可以在不同处理器上执行。

触发软中断

一个注册的软中断只有在标记后才会执行,中断处理程序会在返回前标记它的软中断,使其在稍后执行。
在下列地方,会遍历寻找待处理的软中断并且执行:

  • 从中断处理程序返回后
  • 在ksoftirqd内核线程中
  • 在那些主动执行检查软中断的代码中

具体代码

u32 pending;
pending=local_softirq_pending();
if(pending){
struct softirq_action *h;
set_softirq_pending(0);
h=softirq_vec;do{if(pending & 1)h->action(h);h++;pending>>=1;}while(pending);
}
  1. 是执行do_softirq(),通过将宏local_softirq_pending()的返回值保存到局部变量pending,它是一个32位的位图(如果第n位设置为1,则执行n位的软中断处理程序)。
  2. 重置宏为全0
  3. 依次遍历位图,遇到1则执行相应的软中断处理程序。

使用软中断

软中断保留给系统中对时间最敏感最严格的下半部分使用。目前,只有两个子系统(网络和SCSI)直接使用软中断。此外,内核定时器和tasklet都是建立在软中断上的。相比之下,tasklet可以动态生成,使用更方便。

分配索引

软中断有32位,因为do_softirq()是从下到大的遍历位图,所以小的索引会在大的索引之前执行,索引具有优先级。

tasklet优先级软中断描述
HI_SOFTIRQ0优先级高的tasklet
TIMER_SOFTIRQ1定时器的下半部
NET_TX_SOFTIRQ2发送网络数据包
NET_RX_SOFTIRQ3接收网络数据包
BLOCK_SOFTIRQ4BLOCK装置
TASKLET_SOFTIRQ5正常优先级的tasklet
SCHED_SOFTIRQ6调度程序
HRTIMER_SOFTIRQ7高分辨率定时器
RCU_SOFTIRQ8RCU锁定
注册软中断处理程序

open_softirq(NET_TX_SOFTIRQ,net_t_action);//网络子系统注册自己的软中断处理函数。
有两个参数,第一个是索引号,第二个是处理函数。

触发软中断

raise_softirq(NET_TX_SOFTIRQ);//将中断号为2的网络发送标志设置为1
在do_softirq()中会触发其函数net_t_action();

tasklet

只在高频率使用软中断,大多数情况使用tasklet。
tasklet本质上也是软中断,只不过同一个处理程序的多个实例不能在多处理器上同时运行。所以软中断需要考虑多个处理器同时运行软中断执行函数。而tasklet不需要考虑并发问题。

struct tasklet_struct {struct tasklet_struct *next;		//链表下一个unsigned long state;		//状态atomic_t count;		//引用计数void (*func)(unsigned long);	//核心处理函数unsigned long data;			//给func的参数
}

state状态有:0,TASKLET_STATE_SCHED(已被调度),TASKLET_STATE_RUN(正在运行,只在多处理器上使用)
每个tasklet都存放在两个链表中的其一: tasklet_vec(普通tasklet),tasklet_hi_vec(高优先级tasklet).

创建tasklet

静态创建:
DECLARE_TASKLET(tasklet_name,tasklet_func,tasklet_data);
tasklet不能睡眠,所以不能使用阻塞,信号量等。

两个相同的tasklet绝不会在多个处理器上同时执行,这点和软中断不同

tasklet的调度:tasklet_schedule()或者tasklet_hi_schdule()(高优先级tasklet);

tasklet_schedule()执行步骤:

  1. 检查tasklet的状态,如果不是0,直接返回。
  2. 调用_tasklet_schedule(),将tasklet状态设置为scheduled。
  3. 保存中断状态,禁止本地中断。
  4. 把需要调度的tasklet加到每个处理器的tasklet_vec链表表头。
  5. 将软中断的TASKLET_SOFTIRQ(5)设置为1,这样下次do_softirq()会执行这个tasklet。
  6. 恢复中断(中断返回会触发do_softirq() )

tasklet的执行

do_softirq()检测到位图的TASKLET_SOFTIRQ为1,则执行tasklet_action().
tasklet_action():

  1. 此时处于禁止中断的状态(因为do_softirq()禁止中断),并且为当前处理器检查tasklet_vec链表,检索完后清楚tasklet_vec链表。
  2. 允许中断,执行下半部分,执行每一个待处理的tasklet。
  3. 如果是多处理器,每个处理器都会检查tasklet是否为TASKLET_STATE_RUN状态(其他处理器已经运行了),如果正在其他处理器上运行那么就不运行了。
  4. 将tasklet状态改为TASKLET_STATE_RUN.
  5. 执行func函数(tasklet数据结构中的)
  6. 清楚tasklet的状态。
  7. 运行下一个tasklet,直到结束。

工作队列

当需要睡眠时,那么就使用工作队列。
将工作推后,交由一个内核进程去执行(在进程上下文中执行)。

http://www.lryc.cn/news/293353.html

相关文章:

  • Flink CEP实现10秒内连续登录失败用户分析
  • QSqlRelationalTableModel 关系表格模型
  • JS和CSS实现的原生轮播图
  • 【微服务】skywalking自定义链路追踪与日志采集
  • MYSQL基础问题
  • SpringBoot使用Guava实现日志脱敏(含源码)
  • 数据结构—动态查找
  • Tarjan算法学习笔记
  • vue 项目涉及的焦点聚焦、格式化日期、判断是否为对象或数组、判断是否为空、深拷贝、节流、防抖
  • 软件工程知识梳理6-运行和维护
  • docker- php7.4
  • 开发一个Android App,在项目中完成添加联系人的功能,通过ContentResolver向系统中添加联系人信息。
  • Flume搭建
  • Web APIs 1 DOM操作
  • dvwa,xss反射型lowmedium
  • 从云计算到物联网:虚拟化技术的演变与嵌入式系统的融合
  • linux 文件查看 head 、 cat 、 less 、tail 、grep
  • 13.2 Web与Servlet进阶(❤❤)
  • 记录解决报错--vue前后端分离,接口401(Unauthorized)
  • 【笔记】Android 常用编译模块和输出产物路径
  • 部署私有知识库项目FastGPT
  • 【2024-02-02】华为秋招笔试三道编程题解
  • 银行数据仓库体系实践(8)--主数据模型设计
  • vue在main.js中引入三方插件不生效的原因
  • chatgpt搭建
  • vue基本理解
  • NLP入门系列—Attention 机制
  • 实习记录——第十天
  • Java实现学生信息管理系统:从Excel中提取数据的实用方法
  • 幻兽帕鲁服务器怎么一键删除游戏存档?(阿里云服务器)