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

linux irq

中断上下部

软中断、tasklet、工作对列

软中断优点:运行在软中断上下文,优先级比普通进程高,调度速度快。

缺点:由于处于中断上下文,所以不能睡眠。

相对于软中断/tasklet,工作对列运行在进程上下文

https://blog.csdn.net/jsn_ze/article/details/50740002

https://blog.csdn.net/jsn_ze/article/details/50740058

tasklet

https://blog.csdn.net/lovemengx/article/details/125947279

// kernel\linux-5.10\include\interrupt.h
static inline void tasklet_schedule(struct tasklet_struct *t)
{if (!test_and_set_bit(TASKLET_STATE_SCHED, &t->state))__tasklet_schedule(t);
}// kernel\linux-5.10\kernel\softirq.c
void __tasklet_schedule(struct tasklet_struct *t)
{__tasklet_schedule_common(t, &tasklet_vec,TASKLET_SOFTIRQ);
}static void __tasklet_schedule_common(struct tasklet_struct *t,struct tasklet_head __percpu *headp,unsigned int softirq_nr)
{struct tasklet_head *head;unsigned long flags;local_irq_save(flags);head = this_cpu_ptr(headp);	// 将指定的 tasklet 加入到链表内并设置软中断t->next = NULL;*head->tail = t;head->tail = &(t->next);raise_softirq_irqoff(softirq_nr);local_irq_restore(flags);
}

irq_work

中断上下文执行回调函数的机制,使能该功能需要开启CONFIG_IRQ_WORK。主要逻辑是先通过enqueue work(NMI save的),然后触发一个IPI中断,然后在IPI中断中执行enqueue的work func。其它路径下也有调用回调函数,比如offline cpu、进入idle等。

[Linux内核机制—irq_work](https://www.cnblogs.com/hellokitty2/p/16414217.html#top)

开关中断

1. 关中断

可以通过下面两个函数中的其中任何一个 关闭当前处理器上的所有中断处理, 这两个函数定义在

void local_irq_save(unsigned long flags);
void local_irq_disable(void);

local_irq_save 的调用把当前的中断状态(开或关)保存到flags中,然后禁用当前处理器上的中断。注意, flags 被直接传递, 而不是通过指针来传递,这是由于 local_irq_save被实现为宏 。

local_irq_disable 不保存状态而关闭本地处理器上的中断发送; 只有我们知道中断并未在其他地方被禁用的情况下,才能使用这个版本。

2. 开中断

可通过如下函数打开中断:

void local_irq_restore(unsigned long flags);
void local_irq_enable(void);

local_irq_restore将 保存的flags状态值恢复(即 local_irq_save的入参flag ), 恢复之前的状态(开或关)。

local_irq_enable则无条件打开中断。

在一个关闭中断的环境中调用 local_irq_disable和 local_irq_enable后会破坏之前的中断响应状态。尽管调用 local_irq_disable前是关中断的环境,但是在调用 local_irq_enable后却变成开中断,这显然不是我们希望的 。

调用 local_irq_restore后不一定会开启中断,只会恢复调用 local_irq_save之前的中断状态,如果调用 local_irq_save之前是开中断,那么就打开中断; 如果调用 local_irq_save之前是关中断,那么就关闭中断。

所以 local_irq_save local_irq_restore会更安全。

没有方法全局禁用整个系统的所有中断。 内核开发者认为关闭所有中断的代价太高,因此没有必要提供这个能力。

3. 屏蔽/使能单个中断号

屏蔽中断号

 void disable_irq(int irq);void disable_irq_nosync(int irq);

在全局范围内屏蔽某一个中断号(irq num)。该irq num对应的irq handler不会在任何一个CPU上执行。这个操作是通过设置中断控制器中的寄存器来对指定中断进行屏蔽,而其他未屏蔽的中断依然可以正常送往CPU。

disable_irq 关闭中断并等待中断处理完后返回,会有发生阻塞的可能,因此在中断上半部不能使用disable_irq; 而disable_irq_nosync关闭中断后立即返回,不会发生阻塞。

使能中断号

 void enable_irq(int irq);
static DECLARE_DELAYED_WORK(work, do_softint);
static void do_softint(struct work_struct *work)
{......if (touched) {input_report_key(hp680_ts_dev, BTN_TOUCH, 1);input_report_abs(hp680_ts_dev, ABS_X, absx);input_report_abs(hp680_ts_dev, ABS_Y, absy);} else {input_report_key(hp680_ts_dev, BTN_TOUCH, 0);}input_sync(hp680_ts_dev);enable_irq(HP680_TS_IRQ);
}static irqreturn_t hp680_ts_interrupt(int irq, void *dev)
{disable_irq_nosync(irq);schedule_delayed_work(&work, HZ / 20);return IRQ_HANDLED;
}
http://www.lryc.cn/news/91346.html

相关文章:

  • 串口流控(CTS/RTS)使用详解
  • kube-proxy模式详解
  • 汽车EDI:如何与Stellantis建立EDI连接?
  • 【SCI征稿】1区计算机科学类SCI, 自引率低,对国人友好~
  • Vue.js优化策略与性能调优指南
  • HEVC环路后处理核心介绍
  • 从组件化角度聊聊设计工程化
  • apache的配置和应用
  • Buf 教程 - 使用 Protobuf 生成 Golang 代码和 Typescript 类型定义
  • Java 锁 面试题(ReentrantLock、synchronized)
  • Python中的缩进是什么意思?
  • 2023年9月数学建模:最小二乘优化、曲线拟合与函数逼近
  • java8内部调用无法引用值的问题
  • 《嵌入式系统》知识总结10:使用位带操作操纵GPIO
  • leetcode 2.两数相加(链表操作)
  • Jenkins是什么?以及Jenkins有哪些具体的应用呢?
  • 2023年数学建模:参数估计与假设检验:自助法(Bootstrap)详解
  • 华为OD机试真题 Java 实现【字符串通配符】【2022Q4 200分】
  • Android 11.0 user模式下解除系统进入recovery功能的限制
  • TDEngine3.0 环境安装、配置及使用经验总结
  • Redis7实战加面试题-高阶篇(Redlock算法和底层源码分析)
  • 保持Git历史提交整洁,解决冲突
  • CompletableFuture使用详解,多线程相关
  • (3)NUC980 kenerl编译
  • 华为OD机试真题 Java 实现【分奖金】【2022Q4 100分】
  • 迅为国产化RK3588开发板在安防前后端应用解决方案
  • Windows 安装 GCC
  • 下载安装LabVIEW
  • 从C语言到C++_14(vector的常用函数+相关选择题和OJ题)
  • Java NIO-非阻塞I/O(二)