Linux 软中断
本文转载自: http://blog.chinaunix.net/uid-9620812-id-3833377.html,如有需要,请移步访问。
---------------------------------------我是分割线----------------------------------------
一、软中断注册
和硬中断类似,软中断也有类似的中断向量表,只不过是用“软件”实现的。
struct softirq_action softirq_vec[32]是软中断向量表
(文件linux_2_6_24/kernel/softirq.c)
- struct softirq_action
- {
- void (*action)(struct softirq_action *); //钩子函数
- void *data; //钩子函数的形参
- };
内核用到的中断向量(其实就是数组下标)如下所示
(文件linux_2_6_24/include/linux/interrupt.h)
- enum{
- HI_SOFTIRQ=0, //高优先级的tasklet
- TIMER_SOFTIRQ, //内核定时器
- NET_TX_SOFTIRQ, //网络发送
- NET_RX_SOFTIRQ, //网络接收
- BLOCK_SOFTIRQ, //???
- TASKLET_SOFTIRQ, //普通的tasklet
- SCHED_SOFTIRQ
- }
内核注册一个软中断用函数,本质上就是就是初始化数组某一元素
- void open_softirq(int nr, void (*action)(struct softirq_action*), void *data)
- { //nr 即软中断向量编号
- softirq_vec[nr].data = data;
- softirq_vec[nr].action = action;
- }
内核注册软中断的地方有:
- start_kernel()
- -->init_timers()
- -->open_softirq(TIMER_SOFTIRQ,run_timer_softirq,NULL)
- -->softirq_init()
- -->open_softirq(TASKLET_SOFTIRQ, tasklet_action,NULL)
- -->open_softirq(HI_SOFTIRQ,tasklet_hi_action,NULL)
- -->do_initcall()
- -->net_dev_init()
- -->open_softirq(NET_TX_SOFTIRQ, net_tx_action, NULL);
- -->open_softirq(NET_RX_SOFTIRQ, net_rx_action, NULL);
- -->blk_dev_init()
- -->open_softirq(BLOCK_SOFTIRQ, blk_done_softirq,NULL)
二、软中断触发
前面注册完了,现在开始触发。
内核用一个数据结构来标记曾经有“软中断”发生过(或者说成软中断被触发过)
__softirq_pending 共32bit,即每个bit对应软中断的一个向量,实际使用了6个bit
第n个bit置1,即softirq_vec[n]有软中断发生。
- typedef struct {
- unsigned int __softirq_pending; /* set_bit is used on this */
- unsigned int __last_jiffy_stamp;
- } ____cacheline_aligned irq_cpustat_t;
- extern irq_cpustat_t irq_stat[]; /* defined in asm/hardirq.h */
- #define __IRQ_STAT(cpu, member) (irq_stat[cpu].member)
- #define local_softirq_pending() \
- __IRQ_STAT(smp_processor_id(), __softirq_pending)
- #define set_softirq_pending(x) (local_softirq_pending() = (x))
- #define or_softirq_pending(x) (local_softirq_pending() |= (x))
- #define __raise_softirq_irqoff(nr) do { or_softirq_pending(1UL << (nr)); } while (0)
常用的软中断触发函数
- void raise_softirq_irqoff(int nr){ //nr 即软中断向量编号
- __raise_softirq_irqoff(nr);
- }
但这只是“触发”软中断,软中断并不会立即被处理
三、软中断处理
函数_ _do_softirq是一次性按照向量表从高到低循环处理所有软中断(潜台词,软中断不可嵌套)
_ _do_softirq()的调用时机:
1. irq_exit() 硬件中断处理完,返回时调用
- do_IRQ() -->irq_exit()
- -->local_softirq_pending()
- -->_ _do_softirq()
2. ksoftirqd() 用于辅助处理软中断的内核线程,每一个CPU上都运行着一个ksoftirqd。
- start_kernel() --> kernel_init() -->do_pre_smp_initcalls()
- -->spawn_ksoftirqd() -->cpu_callback()
- -->kthread_create(ksoftirqd, hcpu, "ksoftirqd/%d", hotcpu)
- -->ksoftirqd()
- -->local_softirq_pending()
- -->_ _do_softirq()
3. local_bh_enable()时,发现有待处理的软中断且当时没处在软硬中断上下文中
- local_bh_enable()
- -->local_softirq_pending()
- -->_ _do_softirq()
处理过程代码详解
- asmlinkage void __do_softirq(void)
- {
- struct softirq_action *h;
- __u32 pending;
- int max_restart = MAX_SOFTIRQ_RESTART;
- int cpu;
- pending = local_softirq_pending();
- account_system_vtime(current);
- /*软中断处理中,禁止软中断再次进入,软中断处理是不可重入的*/
- __local_bh_disable((unsigned long)__builtin_return_address(0));
- trace_softirq_enter();
- cpu = smp_processor_id();
- restart:
- /* Reset the pending bitmask before enabling irqs
- 下面首先清除pending,以便系统可以激活其它软件中断,
- 然后使能外部中断
- 系统在下面的处理中,将使能外部中断以提高系统的响应,
- 注意,必须先清除pending,再使能外部中断,否则死锁*/
- set_softirq_pending(0);
- local_irq_enable();
- h = softirq_vec;
- /*下面按照从高到低调用open_softirq注册的句柄*/
- do {
- if (pending & 1) {
- h->action(h); //关键的一句,tasklet、内核timer、网络中断都是在这里搞的
- rcu_bh_qsctr_inc(cpu);
- }
- h++;
- pending >>= 1;
- } while (pending);
- local_irq_disable();
- pending = local_softirq_pending();
- if (pending && --max_restart)
- goto restart;
- if (pending)
- wakeup_softirqd();
- trace_softirq_exit();
- account_system_vtime(current);
- _local_bh_enable();
- }
Linux 软中断的更多相关文章
- linux软中断与硬中断实现原理概述
linux软中断与硬中断实现原理概述. 1.软中断通过open_softirq注册一个软中断处理函数,即在软中断向量表softirq_vec数组中添加新的软中断处理action函数. 2.调用rais ...
- 怎么理解Linux软中断?
1.什么是中断 中断是系统用来响应硬件设备请求的一种机制,它会打断进程的正常调度和执行,然后调用内核中的中断处理程序来响应设备的请求. 2.为什么要有中断呢? "举个生活中的例子" ...
- Linux软中断、tasklet和工作队列
Linux内核中的软中断.tasklet和工作队列详解 引言 软中断.tasklet和工作队列并不是Linux内核中一直存在的机制,而是由更早版本的内核中的“下半部”(bottom half)演变而来 ...
- linux 软中断过高性能优化案例
案例如下: 发现cpu0上的软中断高达50%
- Linux碎碎念
在学习Linux过程中,有许多有用的小技巧.如果放在纸质的笔记本上,平时查阅会相当不方便.现在以一种“碎碎念”的方式,汇集整理在此,目前还不是很多,但随着学习.工作的深入,后续会陆陆续续添加更多的小技 ...
- Intel 80x86 Linux Kernel Interrupt(中断)、Interrupt Priority、Interrupt nesting、Prohibit Things Whthin CPU In The Interrupt Off State
目录 . 引言 . Linux 中断的概念 . 中断处理流程 . Linux 中断相关的源代码分析 . Linux 硬件中断 . Linux 软中断 . 中断优先级 . CPU在关中断状态下编程要注意 ...
- 嵌入式系统Linux内核开发工程师必须掌握的三十道题(转)
嵌入式系统Linux内核开发工程师必须掌握的三十道题 如果你能正确回答以下问题并理解相关知识点原理,那么你就可以算得上是基本合格的Linux内核开发工程师,试试看! 1) Linux中主要有哪几种内核 ...
- Linux中断管理
CPU和外设之间的交互,或CPU通过轮询机制查询,或外设通过中断机制主动上报. 对大部分外设中断比轮询效率高,但比如网卡驱动采取轮询比中断效率高. 这里重点关注ARM+Linux组合下中断管理,从底层 ...
- Linux 驱动开发
linux驱动开发总结(一) 基础性总结 1, linux驱动一般分为3大类: * 字符设备 * 块设备 * 网络设备 2, 开发环境构建: * 交叉工具链构建 * NFS和tftp服务器安装 3, ...
随机推荐
- 【译文】漫谈ASP.NET中的Session
最近这两天被一个Web Farm环境下的Session处理问题虐得很痛苦,网上到处找解决方案,在无意中翻看到这篇文章,感觉很不错,顺手查了一下,貌似没有现成的译文,于是一咬牙一跺脚把这篇文章翻译出来了 ...
- oc-26-动态类型检测
/** 1).判断对象是不是指定类的对象或者指定类的子类对象. - (BOOL)isKindOfClass:(Class)aClass; 2).判断对象是不是1个特定类型的对象,不包括子类. - (B ...
- 对jquery的 attr()和prop()理解
jquery1.6版本后引入了prop()方法,很多时候总是混淆attr()与prop()这俩,下面分析一下这俩的区别 在此之前,先来了解一下html 的attribute和property,因为jq ...
- BM25相关度打分公式
BM25算法是一种常见用来做相关度打分的公式,思路比较简单,主要就是计算一个query里面所有词和文档的相关度,然后在把分数做累加操作,而每个词的相关度分数主要还是受到tf/idf的影响.公式如下: ...
- 关于删除7z历史解压路径
目前7zip尚无此功能,只能从注册表入手了,7zip历史记录保存在注册表 HKEY_CURRENT_USER\Software\7-Zip\FM 下的CopyHistory和FolderHistory ...
- html元素英文含义
常用html标签的英语全称及简单功能描述 <a>:anchor 定义锚 <abbr>:abbreviation 定义缩写 <acronym>: 定义只取消首字母的缩 ...
- n-1位数
描述 已知w是一个大于10但不大于1000000的无符号整数,若w是n(n≥2)位的整数,则求出w的后n-1位的数. 输入 第一行为M,表示测试数据组数.接下来M行,每行包含一个测试数据. 输出 ...
- solr中通过SFTP访问文件建立索引
需求: 从oracle数据库中根据记录的文件名filename_html(多个文件以逗号隔开),文件路径path,备用文件名bakpath中获取 主机172.21.0.31上对应的html文件内容,并 ...
- ios-uitableviewcell展开
#import <UIKit/UIKit.h> @interface ZSDHelpCell : UITableViewCell @property (weak, nonatomic) I ...
- WebApi 部署后一直返回404的解决办法
Fixing ASP.NET MVC 4 Web API 404 Posted on November 5, 2012 For a Web Service providing some REST-st ...