大家好,我是王鸽,这篇文章主要是讲Linux中断管理之中断屏蔽,屏蔽中断有哪些方法,屏蔽CPU还是屏蔽中断号区别等内容。什么是中断屏蔽
中断屏蔽的本质是:通过修改 CPU 寄存器(如 ARM 的 CPSR/I 位、x86 的 EFLAGS/IF 位)或中断控制器配置,让 CPU 暂时不响应指定的中断。全局屏蔽
屏蔽当前 CPU 的所有可屏蔽中断(硬中断);
局部屏蔽
只屏蔽指定 IRQ 号的中断,或屏蔽软中断 /tasklet。
Linux 内核通过「本地中断屏蔽」和「中断优先级掩码」来控制嵌套,但是不可屏蔽中断(Non Maskable Interrupt,NMI)是个例外。
1.全局硬中断屏蔽
禁止CPU中断接口如下
(1)local_irq_disable()。
(2)local_irq_save(flags):首先把中断状态保存在参数flags中,
然后禁止中断。
这两个接口只能禁止本处理器的中断,不能禁止其他处理器的中断。
禁止中断以后,处理器不会响应中断请求,全局硬中断屏蔽(最常用)。
开启中断的接口如下
(1)local_irq_enable()。
(2)local_irq_restore(flags):恢复本处理器的中断状态。
local_irq_disable()和local_irq_enable()不能嵌套使用,
local_irq_save(flags)和local_irq_restore(flags)可以嵌套使用。
以local_开头的方法的作用范围是本CPU内。
| | |
|---|
local_irq_disable() | | |
local_irq_enable() | | 配合 local_irq_disable() 使用 |
local_irq_save(flags) | | |
local_irq_restore(flags) | | 配合 local_irq_save(),避免误开启 |
关键说明
flags是 unsigned long 类型,仅在当前 CPU 有效;这类函数只影响硬中断,不影响软中断 /tasklet;
不能在中断上下文之外的 “睡眠场景” 使用(如持有 mutex 时)。
伪代码
// 场景:保护一个全局变量的读写(避免中断打断) unsigned long flags; int global_counter = 0; // 方式1:简单屏蔽/开启(确定当前中断是开启的) local_irq_disable(); global_counter++; // 临界区(必须极短) local_irq_enable(); // 方式2:安全屏蔽/恢复(通用推荐) local_irq_save(flags); // 保存状态 + 屏蔽中断 global_counter++; local_irq_restore(flags); // 恢复到原来的状态(无论之前是否屏蔽)
禁止中断的函数 local_irq_disable()内核这块是怎么实现可以追一下代码
kernel/include/linux/irqflags.h#define local_irq_disable() do { raw_local_irq_disable(); } while (0)#define raw_local_irq_disable() arch_local_irq_disable()
arch_local_irq_disable() 在kernel/arch/arm64/include/asm/irqflags.hstaticinlinevoidarch_local_irq_disable(void){asmvolatile("msr daifset, #2 // arch_local_irq_disable" : : : "memory");}
把处理器状态的中断掩码位设置成 1,从此以后处理器不会响应中断请求。同样的开启中断的函数 local_irq_enable()local_irq_enable() -> raw_local_irq_enable() -> arch_local_irq_enable()arch/arm64/include/asm/irqflags.hstaticinlinevoidarch_local_irq_enable(void){asmvolatile("msr daifclr, #2 // arch_local_irq_enable" : : : "memory");}
2.仅屏蔽软中断(不影响硬中断)
| | |
|---|
local_bh_disable() | 屏蔽当前 CPU 的软中断(bottom half)和 tasklet | |
local_bh_enable() | | |
关键说明
“bh” 是 bottom half 的缩写,即软中断的旧称;
这类函数只屏蔽软中断 /tasklet,硬中断仍能正常响应;
常用于保护被软中断访问的临界区(如网络协议栈、块设备驱动)。
// 场景:修改一个被 NET_RX_SOFTIRQ(网络软中断)访问的全局变量local_bh_disable();// 临界区:修改网络相关全局变量(软中断不会打断)net_stat.count++;local_bh_enable();
3. 屏蔽指定 IRQ 号的中断(精准屏蔽)
软件可以禁止某个外围设备的中断,中断控制器不会把该设备发送的中断转发给处理器。 | | |
|---|
disable_irq(unsigned int irq) | | |
disable_irq_nosync(unsigned int irq) | | |
enable_irq(unsigned int irq) | | |
irq_set_affinity_hint() | | |
关键说明
irq
参数是要屏蔽的中断号(如2、30);
disable_irq
会等待当前中断处理完成,不能在该中断的处理函数内调用(会死锁),因此如果在n号中断的顶半部调用disable_irq(n), 会引起系统的死锁, 这种情况下, 只能调用disable_irq_nosync(n) 。
disable_irq_nosync
无等待,可在中断处理函数内使用(但需注意竞态)。
伪代码
// 场景1:卸载驱动时,永久屏蔽并释放 IRQ30(网卡中断)disable_irq(30); // 等待当前中断处理完成free_irq(30, dev); // 释放 IRQ 资源// 场景2:中断处理函数内,临时屏蔽当前 IRQ(避免重入)irqreturn_t irq_handler(int irq, void *dev_id) { disable_irq_nosync(irq); // 不等待,直接屏蔽 // 核心处理逻辑(避免该中断再次触发) handle_hardware_event(); enable_irq(irq); // 恢复中断 return IRQ_HANDLED;}
对于 ARM64 架构的 GIC 控制器,如果需要开启硬件中断 n,那么设置分发器的寄存器 GICD_ISENABLERn( Interrupt Set-Enable Register);如果需要禁止硬件中断 n,那么设置分发器的寄存器 GICD_ICENABLERn( Interrupt Clear-Enable Register)。4.临时屏蔽所有中断(睡眠安全版)
| | |
|---|
spin_lock_irqsave(&lock, flags) | | |
spin_unlock_irqrestore(&lock, flags) | | |
mutex_lock() | | |
关键说明
自旋锁 + 中断屏蔽是 “原子操作” 的黄金组合,避免中断嵌套导致的死锁;
互斥锁不屏蔽中断,但会触发进程调度(只能在进程上下文使用)。
伪代码
// 场景:临界区需要加锁,且防止中断打断导致锁竞争DEFINE_SPINLOCK(my_lock);unsigned long flags;spin_lock_irqsave(&my_lock, flags); // 加锁 + 屏蔽中断// 临界区:操作共享资源(如硬件寄存器)hw_reg_write(REG_DATA, 0x1234);spin_unlock_irqrestore(&my_lock, flags); // 解锁 + 恢复中断
四、使用注意事项(避坑)
屏蔽时间越短越好
全局中断屏蔽会导致所有硬中断延迟,长时间屏蔽会引发系统卡死(如网卡丢包、串口无响应);
避免嵌套屏蔽
local_irq_disable()调用多次后,需对应次数的local_irq_enable()才能恢复;
中断上下文禁止睡眠
屏蔽中断后(如local_irq_disable()),不能调用msleep()、copy_to_user()等可能睡眠的函数;
优先精准屏蔽
能屏蔽指定 IRQ 就不全局屏蔽,能屏蔽软中断就不屏蔽硬中断(减少对系统的影响)。
总结
全局硬中断屏蔽
用local_irq_save/restore(通用)、local_irq_disable/enable(简单场景);
指定 IRQ 屏蔽
用disable_irq_nosync/enable_irq(中断上下文)、disable_irq(进程上下文);
软中断屏蔽
用local_bh_disable/enable;
临界区保护
优先用spin_lock_irqsave/restore(原子操作),需要睡眠时用互斥锁(不屏蔽中断)。
谢谢阅读点赞关注收藏!