转自:https://www.cnblogs.com/xiaojiang1025/p/6376561.html

异步通知的全称是"信号驱动的异步IO",通过"信号"的方式,放期望获取的资源可用时,驱动会主动通知指定的应用程序,和应用层的"信号"相对应,这里使用的是信号"SIGIO"。操作步骤是

  1. 应用层程序将自己注册为接收来自设备文件的SIGIO信号的进程
  2. 驱动实现相应的接口,以期具有向所有注册接收这个设备驱动SIGIO信号的应用程序发SIGIO信号的能力。
  3. 驱动在适当的位置调用发送函数,应用程序即可接收到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项工作。

  1. 支持F_SETOWN命令,能在这个命令中下设置filp->f_owner为对应进程的ID,这部分内核已经做了
  2. 支持F_SETFL,每当FASYNC标志改变时,驱动程序中的fasync()将得以执行,so,驱动中要实现fasync()
  3. 当设备资源可用时,通过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);
...
}

最新文章

  1. LinkedHashMap源码阅读笔记(基于jdk1.8)
  2. 服务器Ubuntu16.04下连接锐捷
  3. SVN的简单使用
  4. Visual Studio 当前上下文中不存在名称“ConfigurationManager”
  5. RootKit学习之 IDT Hook
  6. 如何在 Swift 语言下使用 iOS Charts API 制作漂亮图表?
  7. 过滤文本文档中的数据并插入Cassandra数据库
  8. SQLSERVER 启用跨库查询脚本
  9. [OJ] Wildcard Matching (Hard)
  10. USB C和USB 3.1傻傻分不清?这篇文章可以帮你
  11. 使用ARM和VMSS创建自动扩展的web集群
  12. 201521123087 《Java程序设计》第5周学习总结
  13. WebGL之通过外部传入a_PontSize值改变点着色器vshader内置变量gl_PointSize的值
  14. django模板引擎自定义变量
  15. PLSQL僵死
  16. k8s简单的来部署一下tomcat,并测试自愈功能
  17. 基于MAVEN使用IDEA创建dubbo入门项目图文教程
  18. 【blog】推荐一个博客系统后台管理模板 - pinghsu
  19. MySQL字符串列与整数比较
  20. HDU1559

热门文章

  1. [2017BUAA软工]结对项目-数独程序扩展
  2. c++ std::function
  3. BZOJ2744 HEOI2012朋友圈(二分图匹配)
  4. Django_博客_XSS 攻击防范
  5. Lua 调试库
  6. R vs Python,数据分析中谁与争锋?
  7. 安利一个很火的 Github 滤镜项目
  8. jquery生成二维码并实现图片下载
  9. 关于xmlhttp会使用ie的缓存的问题及解决
  10. spring@Transactional的一点理解