在linux驱动程序编写过程中,异步通知技术是一种非常重要的技术。它可以实现高效的事件处理和数据传输,提高系统的性能和响应速度。在本文中,我们将深入探讨linux驱动技术(四) _异步通知技术的实现原理和相关技术。
异步通知的全称是”信号驱动的异步IO”,通过”信号”的方式,期望获取的资源可用时,驱动会主动通知指定的应用程序,和应用层的”信号”相对应,这里使用的是信号”SIGIO“。操作步骤是
- 应用层程序将自己注册为接收来自设备文件的SIGIO信号的进程
- 驱动实现相应的接口,以期具有向所有注册接收这个设备驱动SIGIO信号的应用程序发SIGIO信号的能力。
- 驱动在适当的位置调用发送函数,应用程序即可接收到SIGIO信号。
整个机制的框架:
“
”
应用层接收SIGIO
和其他信号一样,应用层需要注册一个信号处理函数,
注册的方式还是使用signal()或sigaction()
此外,应用层还需要把自己加入到驱动的通知链表中,加入的代码如下
fcntl(dev_fd,F_SETOWN,getpid()); int oflags = fcntl(dev_fd,F_GETFL); fcntl(dev_fd,F_SETFL,oflags|FASYNC); ... while(1);登录后复制
完成了上面的工作,应用层的程序就可以静待SIGIO的到来了。
驱动发送SIGIO
应用层注册好了,最终的发送还是看设备驱动的处理方式,为了使设备支持异步通知机制,参照应用层的接口,驱动程序中涉及3项工作。
- 支持F_SETOWN命令,能在这个命令中下设置filp->f_owner为对应进程的ID,这部分内核已经做了
- 支持F_SETFL,每当FASYNC标志改变时,驱动程序中的**fasync()将得以执行,so,驱动中要实现fasync()**。
- 当设备资源可用时,通过kill_fasync()发送SIGIO
为了在内核中实现上面这三个功能,驱动需要使用1个结构+2个API,结构是struct fasync_struct,函数是fasync_helper()和kill_fasync()
struct fasync_struct { spinlock_t fa_lock; int magic; int fa_fd; struct fasync_struct *fa_next; /* singly linked list */ struct file *fa_file; struct rcu_head fa_rcu; };登录后复制
fasync_helper()的作用是将一个fasync_struct的对象注册进内核,应用层执行**fcntl(dev_fd,F_SETFL,oflags|FASYNC)时会回调驱动的fops.fasync(),所以通常将fasync_helper()放到fasync()**的实现中。
/** *fasync_helper - 将一个fasync_struct对象注册进内核 *@fd:文件描述符,由fasync传入 *@filp:file指针,由fasync传入 *@sig:信号类型,通常使用的就是SIGIO *@dev_fasync:事前准备的fasync_struct对象指针的指针 */ int fasync_helper(int fd, struct file * filp, int sig, struct fasync_struct ** dev_fasync);登录后复制
下面这个API就是释放SIGIO,根据需求的不同放到不同的位置。
/** *kill_fasync - 释放一个信号 *@dev_fasync:事前使用fasync_helper注册进内核的fasync_struct对象指针的指针 *@filp:file指针,由fasync传入 *@sig:信号类型,通常使用的就是SIGIO *@flag:标志,通常,如果资源可读用POLLIN,如果资源可写用POLLOUT */ void kill_fasync(struct fasync_struct **dev_fasync, int sig, int flag);登录后复制
驱动模板
下面这个驱动模板针对在硬件中断到来(资源可用)的时候向应用层发信号,实际的操作中表明资源可用的情境还有很多
static struct fasync_struct *fasync = NULL; static irqreturn_t handler(int irq, void *dev) { kill_fasync(&fasync, SIGIO, POLLIN); return IRQ_HANDLED; } static int demo_fasync(int fd, struct file *filp, int mode) { return fasync_helper(fd, filp, mode, &fasync); } struct file_operations fops = { ... .fasync = demo_fasync, ... } static int __init demo_init(void) { ... request_irq(irq, handler, IRQF_TRIGGER_RISING, "demo", NULL); ... }登录后复制
总之,异步通知技术是Linux驱动程序编写过程中不可或缺的一部分。它可以实现高效的事件处理和数据传输,提高系统的性能和响应速度。希望本文能够帮助读者更好地理解Linux驱动技术(四) _异步通知技术的实现原理和相关技术。
以上就是深入探讨Linux驱动技术(四) _异步通知技术的实现原理和相关技术的详细内容,更多请关注慧达安全导航其它相关文章!
免责 声明
1、本网站名称:慧达安全导航
2、本站永久网址:https//www.huida178.com/
3、本站所有资源来源于网友投稿和高价购买,所有资源仅对编程人员及源代码爱好者开放下载做参考和研究及学习,本站不提供任何技术服务!
4、本站所有资源的属示图片和信息不代表本站的立场!本站只是储蓄平台及搬运
5、下载者禁止在服务器和虚拟机下进行搭建运营,本站所有资源不支持联网运行!只允许调试,参考和研究!!!!
6、未经原版权作者许可禁止用于任何商业环境,任何人不得擅作它用,下载者不得用于违反国家法律,否则发生的一切法律后果自行承担!
7、为尊重作者版权,请在下载24小时内删除!请购买原版授权作品,支持你喜欢的作者,谢谢!
8.若资源侵犯了您的合法权益,请持 您的版权证书和相关原作品信息来信通知我们!QQ:1247526623我们会及时删除,给您带来的不便,我们深表歉意!
9、如下载链接失效、广告或者压缩包问题请联系站长处理
10、如果你也有好源码或者教程,可以发布到网站,分享有金币奖励和额外收入!
11、本站资源售价只是赞助,收取费用仅维持本站的日常运营所需
12、因源码具有可复制性,一经赞助,不得以任何形式退款。
13、本文内容由网友自发贡献和站长收集,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系1247526623@qq.com
转载请注明出处: 慧达安全导航 » 深入探讨Linux驱动技术(四) _异步通知技术的实现原理和相关技术
发表评论 取消回复