linux下异步通知fasync的理解

本文深入探讨了异步通知机制的原理和实现过程,包括内核如何管理设备文件和进程之间的通信,以及应用程序如何配置接收SIGIO信号并处理异步事件。详细解释了从驱动到应用层的异步通信流程,以及如何使用fcntl和signal函数进行配置。

异步通知,的确是一种很好处理机制,熟练掌握这种手段,对我们写高质量的应用程序很有帮助。下面说说对它的理解:


异步通知,就是让驱动去告诉应用,底层硬件发生了什么事,而不是应用主动地去查询驱动,这对系统的性能有一个很大的提升。

首先,在驱动中

内核首先定义一个结构体struct fasync_struct,这个结构体用来存放对应设备文件的信息(fd, filp)并交给内核来管理。一但收到信号,内核就会在这个所谓的异步队列头找到相应的文件(fd),并在filp->owner中找到对应的进程PID,这样就确定了向谁发。不过,此时fd, filp都还不确定,这就需要借助于fasync_helper(fd, filp, mode, &dev->async_queue),fd,filp和定义的结构体传给内核,这样就完成了fd、filp、结构体三者的衔接。

而fasync_helper则需要通过ops结构体的fasync成员调用 int test_fasync (int fd, struct file *filp, int mode) ,在后者中完成对fasync_helper的调用,从而完成对异步队列的填充。

从这个过程中,也可以看到,哪个进程完成对ops成员fasync成员的调用,就会把fd、filp传给异步队列,kill_fasync发出信号就会到这个进程中,这就是驱动层次的理解。

其次,就是在应用层次

首先,要用fcntl对此文件进行设置,

fcntl(fd, F_SETOWN, getpid());设置接收SIGIO信号的进程组

Oflags = fcntl(fd, F_GETFL);  得到现在文件的标志位

fcntl(fd, F_SETFL, Oflags | FASYNC);对当前文件的标志位加上一个FASYNC属性,每当FASYNC标志改变时,驱动程序中的fasync()函数将得以执行。

然后就是应用程序得到SIGIO信号后,要绑定新的操作函数

signal(SIGIO, my_signal_fun);

这样就完成了异步通信的过程,当驱动发送kill_fasync时就会在应用程序中调用my_signal_fun函数去处理。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值