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

Linux中断操作

一、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)

四、

五、

六、

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

相关文章:

  • 看看CabloyJS是如何异步加载并执行go wasm模块的
  • 嵌入式C语言九大数据结构操作方式详解
  • 【C++学习】栈 | 队列 | 优先级队列 | 反向迭代器
  • Python—看我分析下已经退市的 可转债 都有什么特点
  • 【第八课】空间数据基础与处理——数据结构转化
  • MATLAB绘制三Y轴坐标图:补充坐标轴及字体设置
  • springboot项目中Quartz
  • Presto本地开发,plugin的设置
  • 2023年3月西安/杭州/深圳/东莞NPDP产品经理认证考试报名
  • Vue3笔记01 创建项目,Composition API,新组件,其他
  • pandas数据分析(二)
  • Spring实现[拦截器+统一异常处理+统一数据返回]
  • MySQL——插入加锁/唯一索引插入死锁/批量插入效率
  • 【专项训练】数组、链表
  • 基于Jeecgboot前后端分离的ERP系统开发代码生成(六)
  • 什么?同步代码块失效了?-- 自定义类加载器引起的问题
  • CHAPTER 4 文件共享 - Samba
  • 深入分析@Configuration源码
  • Unity 代码优化 内存管理优化
  • 设计模式~门面(外观)模式(Facade)-08
  • C++面向对象编程之一:封装
  • IDEA插件系列(3):Maven Helper插件
  • SAP 更改物料基本计量单位
  • 蓝桥web基础知识学习
  • Python+ChatGPT制作一个AI实用百宝箱
  • Python中格式化字符串输出的4种方式
  • C#基础教程15 枚举与类
  • 三步 让你的 vscode 自动编译ts文件
  • STM32程序下载和启动方式
  • 基础01-ajax fetch axios 的区别