Linux中断操作
创始人
2024-05-30 14:15:59
0

一、thread_irq

在内核中, 除了可以通过request_irq() 、 devm_request_irq()申请中断以外, 还可以通过以下二个函数申请( 它们比request_irq和devm_request_irq多了一个参数thread_fn)。

用这两个API申请中断的时候, 内核会为相应的中断号分配一个对应的内核线程。 注意这个线程只针对这个中断号, 如果其他中断也通过request_threaded_irq() 申请, 自然会得到新的内核线程。

int request_threaded_irq(unsigned int irq, irq_handler_t handler,irq_handler_t thread_fn, unsigned long irqflags,const char *devname, void *dev_id)int devm_request_threaded_irq(struct device *dev, unsigned int irq,irq_handler_t handler, irq_handler_t thread_fn,unsigned long irqflags, const char *devname,void *dev_id)

参数handler对应的函数执行于中断上下文, thread_fn参数对应的函数则执行于内核线程。 如果handler结束的时候, 返回值是IRQ_WAKE_THREAD, 内核会调度对应线程执行thread_fn对应的函数。

另外这二个函数支持在irqflags中设置IRQF_ONESHOT标记,这样内核会自动帮助我们在中断上下文中屏蔽对应的中断号, 而在内核调度thread_fn执行后, 重新使能该中断号。 对于我们无法在上半部清除中断的情况, IRQF_ONESHOT特别有用, 避免了中断服务程序一退出, 中断就洪泛的情况。

handler参数可以设置为NULL, 这种情况下, 内核会用默认的irq_default_primary_handler()代替handler, 并会使用IRQF_ONESHOT标记。

/** Default primary interrupt handler for threaded interrupts. Is* assigned as primary handler when request_threaded_irq is called* with handler == NULL. Useful for oneshot interrupts.*/
static irqreturn_t irq_default_primary_handler(int irq, void *dev_id)
{return IRQ_WAKE_THREAD;
}

二、中断共享(IRQF_SHARED)

多个设备共享一根硬件中断线的情况在实际的硬件系统中广泛存在, Linux支持这种中断共享。 下面是中断共享的使用方法。

使用共享中断的设备驱动程序的模板(仅包含与共享中断机制相关的部分):

/* 在中断到来时, 会遍历执行共享此中断的所有中断处理程序, 直到某一个函数返回
IRQ_HANDLED。 在中断处理程序顶半部中, 应根据硬件寄存器中的信息比照传入的dev参数
迅速地判断是否为本设备的中断, 若不是, 应迅速返回IRQ_NONE*/
irqreturn_t xxx_interrupt(int irq, void *dev)
{....../*获知中断源*/int status = read_int_status();/*判断是否为本设备中断,若不是立即返回*/if(!is_myint(dev, status))return IRQ_NONE;/*若是本设备中断,进行处理*/....../*表明中断已被处理*/return IRQ_HANDLED;
}/*设备驱动模块加载函数*/
int xxx_init(void)
{....../*共享中断的多个设备在申请中断时, 都应该使用IRQF_SHARED标志, 而且一个设备以IRQF_SHARED申请某中断成功的前提是该中断未被申请,或该中断虽然被申请了,但是之前申请该中断的所有设备也都以IRQF_SHARED标志申请该中断。内核为每个中断维护一个中断共享处理例程列表,dev就是区别不同处理例程的签名;因此最后一个参数dev必须唯一,任何指向模块地址空间的指针都行,但 dev绝不能设置为 NULL。一般将设备结构体指针作为参数。*/result = request_irq(sh_irq, xxx_interrupt, IRQF_SHARED, "xxx", xxx_dev);......
}/*设备驱动模块卸载*/
void xxx_exit(void)
{......free_irq(xxx_irq, xxx_interrupt);......  
}

一个使用共享处理例程的驱动需要小心:不能使用 enable_irq 或 disable_irq,否则,对其他共享这条线的设备就无法正常工作了。即便短时间禁止中断,另一设备也可能产生延时而为设备和其用户带来问题。

三、使能禁止中断API

1)使能或者禁止某一个中断API:

void enable_irq(unsigned int irq)
void disable_irq(unsigned int irq)

用于使能和禁止指定的中断, irq 就是要禁止的中断号。disable_irq 函数要等到当前正在执行的中断处理函数执行完才返回,因此使用者需要保证不会产生新的中断,并且确保所有已经开始执行的中断处理程序已经全部退出。在这种情况下,可以使用另外一个中断禁止函数:

void disable_irq_nosync(unsigned int irq)

函数调用以后立即返回,不会等待当前中断处理程序执行完毕。

2)使能或者禁止当前处理器的整个中断系统API:

local_irq_enable()
local_irq_disable()

以下这两个函数是一对, local_irq_save 函数用于禁止中断,并且将中断状态保存在 flags中。 local_irq_restore 用于恢复中断,将中断到 flags 状态。

local_irq_save(flags)
local_irq_restore(flags)

四、

五、

六、

相关内容

热门资讯

监控摄像头接入GB28181平... 流程简介将监控摄像头的视频在网站和APP中直播,要解决的几个问题是:1&...
Windows10添加群晖磁盘... 在使用群晖NAS时,我们需要通过本地映射的方式把NAS映射成本地的一块磁盘使用。 通过...
protocol buffer... 目录 目录 什么是protocol buffer 1.protobuf 1.1安装  1.2使用...
在Word、WPS中插入AxM... 引言 我最近需要写一些文章,在排版时发现AxMath插入的公式竟然会导致行间距异常&#...
【PdgCntEditor】解... 一、问题背景 大部分的图书对应的PDF,目录中的页码并非PDF中直接索引的页码...
修复 爱普生 EPSON L4... L4151 L4153 L4156 L4158 L4163 L4165 L4166 L4168 L4...
Fluent中创建监测点 1 概述某些仿真问题,需要创建监测点,用于获取空间定点的数据࿰...
educoder数据结构与算法...                                                   ...
MySQL下载和安装(Wind... 前言:刚换了一台电脑,里面所有东西都需要重新配置,习惯了所...
MFC文件操作  MFC提供了一个文件操作的基类CFile,这个类提供了一个没有缓存的二进制格式的磁盘...