Linux设备驱动中的异步通知与异步I/O
异步通知概念: 异步通知的意识是,一旦设备就绪,则主动通知应用程序,这样应用程序根本就不需要查询设备状态,这一点非常类似于硬件上的“中断”概念,比较准确的称谓是“信号驱动的异步IO”,信号是在软件层次上对中断机制的一种模拟,在原理上,一个进程收到一个信号与处理器收到一个中断请求是一样的。信号是异步的,一个进进程不必通过任何操作来等待信号的到达,事实上,进程也不知道信号到底什么时候到达。
异步通知与设备访问: 阻塞IO意味着一直等待设备可访问后访问;非阻塞IO使用poll意味着查询设备是否可以访问;而异步通知则意味着设备通知自身可访问,实现了异步I/O。
用kill -l 命令可以查看Linux可用的信号
信号的接收
signal()函数来设置对应信号的处理函数。
void (*signal(int signum,void(*handler))(int)))(int);
//第一个参数指定信号的值;第二个参数指定针对前面信号值的处理函数,若位SIN_IGN,表示忽略该信号;若位SIG_DFL,表示采用系统默认方式处理信号;若位用户自定义
函数,则信号被捕获到后,该函数将被执行
//该函数原型比较难理解,可以分解为:typedef void(*sighandler_t)(int); sighandler_t signal(int signum,sighandler_t handler));
如果signal调用成功,它返回最后一次信号signum绑定的处理函数handler值,失败则返回SIG_ERR.
sigaction()函数可用于改变进程接收到特定信号后的行为。
int sigaction(int signum,const struct sigaction *act,struct sigaction *oldact));
//该函数的第一个参数是信号的值,可以为除SIGKILL及SIGSTOP外的任何一个特定有效的信号;第二个参数是指向结构体sigaction的一个实例的指针,在sigaction的实例中
指定了对特定信号的处理函数,若为空,则进程会以缺省方式对信号处理;第三个参数oldact指向的对象用来保存原来对相应信号的处理函数,可指定oldact为NULL.
如果把第二和第三个参数都设为NULL,那么该函数用来检测信号的有效性。
使用信号实现异步通知的应用程序实例:
#include <sys/types.h>
#include <sys/stat.h>
#include <stdio.h>
#include <fcntl.h>
#include <signal.h>
#include <unistd.h>
#define MAX_LEN 100
void input handler(int num)
{
char data[MAX_LEN];
int len;
len=read(STDIN_FILENO,&data,MAX_LEN); //读取并输出STDIN_FILENO上的输入。
data[len]=0;
printf("input available:%s\n",data);
}
main()
{
int oflags;
//启动信号驱动机制
signal(SIGIO,input_handler); //为SIGIO信号安装input_handler()作为处理函数
fcntl(STDIN_FILENO,F_SETOWN,getpid());//这里的getpid()是获得当前进程,注意,没有这一步,内核不知道应该将信号发给哪个进程。
fcntl是一个非常通用的函数,它还可以改变文件进程各方面的属性,在本节中,主要介绍它建立记录锁的方法,关于它其他用户感兴趣的读者可以参看fcntl手册。
oflags=fcntl(STDIN_FILENO,F_GETFL); //为了启动一步通知机制,还需对设备设置FASYNC标志
fcntl(STDIN_FILENO,F_SETFL,oflags|FASYNC);
//最后进入一个死循环,仅为保持进程不终止,如果程序中没有死循环会立即执行完毕
while(1);
}
信号释放:
在设备驱动程序与应用程序的异步通知交互中,应用程序捕获信号,驱动程序释放信号。
为了使设备支持异步通知机制,驱动程序中涉及3项工作。
(1)支持F_SETOWN命令,能在这个控制命令处理中设置filp->f_owner为对应进程ID。不过此项工作已由内核完成,设备驱动无需处理。
(2)支持F_SETFL命令的处理,每当FASYNC标志改变时,驱动程序中的fasync()函数将得以执行。因此,驱动中应该实现fasync()函数。
(3)在设备资源可获得时,调用kill_fasync()函数激发相应的信号。
异步通知处理过程中用户空间和设备驱动的交互。

异步通知变成主要用到一项数据结构和两个函数。数据结构是fasync_struct结构体,两个函数是fasync_helper,kill_fasync。
int fasync_helper(int fd,struct file *filp,int mode,struct fasync_struct **fa); //处理FASYNC标志变更
void kill_fasync(struct fasync_struct **fa,int sig,int band); //释放信号
支持异步通知的设备结构体模板
struct xxx_dev{
struct cdev cdev; //cdev结构体
...
struct fasync_struct *async_queue; //异步结构体指针
};
支持异步通知的设备驱动fasync()函数模板
static int xxx_fasync(int fd,struct file *filp,int mode)
{
struct xxx_dev *dev=filp->private_data;
return fasync_helper(fd,filp,mode,&dev->async_queue);
}
在设备资源可以获得时,应该调用kill_fasync()释放SIGIO信号,可读时第三个参数设置为POLL_IN,可写时第三个参数设置为POLL_OUT.
支持异步通知的设备驱动信号释放范例:
static ssize_t xxx_write(struct file *filp,const char_user *buf,size_t count,loff_t *f_pos)
{
struct xx_dev *dev=filp->private_data;
...
//产生异步读信号
if(dev->async_queue)
kill_fasync(&dev->async_queue,SIGIO,POLL_IN);
...
}
最后在关闭文件时候,即在release函数中,应该将文件从异步通知的列表中删除。
static int xxx_release(struct inode *inode,struct file *filp)
{
//将文件从异步通知列表中删除
xxx_fasync(-1,filp,0);
...
return 0;
}
Linux设备驱动中的异步通知与异步I/O的更多相关文章
- Linux设备驱动中的阻塞和非阻塞I/O
[基本概念] 1.阻塞 阻塞操作是指在执行设备操作时,托不能获得资源,则挂起进程直到满足操作所需的条件后再进行操作.被挂起的进程进入休眠状态(不占用cpu资源),从调度器的运行队列转移到等待队列,直到 ...
- Linux设备驱动中的阻塞和非阻塞I/O <转载>
Green 博客园 首页 新随笔 联系 订阅 管理 Linux设备驱动中的阻塞和非阻塞I/O [基本概念] 1.阻塞 阻塞操作是指在执行设备操作时,托不能获得资源,则挂起进程直到满足操作所需的条件 ...
- 蜕变成蝶~Linux设备驱动中的阻塞和非阻塞I/O
今天意外收到一个消息,真是惊呆我了,博客轩给我发了信息,说是俺的博客文章有特色可以出本书,,这简直让我受宠若惊,俺只是个大三的技术宅,写的博客也是自己所学的一些见解和在网上看到我一些博文以及帖子里综合 ...
- Linux设备驱动中的IO模型---阻塞和非阻塞IO【转】
在前面学习网络编程时,曾经学过I/O模型 Linux 系统应用编程——网络编程(I/O模型),下面学习一下I/O模型在设备驱动中的应用. 回顾一下在Unix/Linux下共有五种I/O模型,分别是: ...
- Linux设备驱动中的软件架构思想
目录 更新记录 一.Linux驱动的软件架构 1.1 出发点 1.2 分离思想 1.3 分层思想 二.platform设备驱动 2.1 platform设备 2.2 platform驱动 2.3 pl ...
- Linux设备驱动中的ioctl
memdev.h #ifndef _MEMDEV_H #define _MEMDEV_H #define MEM_MAGIC 'm' #define MEM_RESTART _IO(MEM_MAGIC ...
- Linux设备驱动中的并发控制
1.并发是指多个执行单元同时.并行的执行.并发的执行单元对共享资源的访问很容易导致竞态. 在 Linux 内核中,主要的竞态发生于如下几种情况: ①对称多处理器(SMP)的多个 CPU ②单CPU内进 ...
- 蜕变成蝶~Linux设备驱动中的并发控制
并发和竞争发生在两类体系中: 对称多处理器(SMP)的多个CPU 内核可抢占的单CPU系统 访问共享资源的代码区域称为临界区(critical sections),临界区需要以某种互斥机制加以保护.在 ...
- linux设备驱动中的并发控制【转】
转自:http://www.cnblogs.com/plinx/archive/2013/01/28/2873121.html 在linux内核中,主要的静态发生于以下几种情况: 1.对称多处理器(S ...
随机推荐
- HBase的安装与使用
1.安装 由于还是学习阶段,所以没有在生产环境练习,就在本地建了个虚拟机进行HBase的安装. 下载地址http://www.apache.org/dyn/closer.cgi/hbase/,选择一个 ...
- sum_series() 求一列数的指定个数的数和(5个数字的和)
#include <stdio.h> #include <stdarg.h> /*用sum_series() 求一列数的指定个数的数和(5个数字的和)*/ double sum ...
- 深入浅出JMS(二)——JMS的组成
JMS Provider 实现了JMS规范的消息系统,该系统还提供必须的用于管理和控制全方位的功能,如这里的ActiveMQ. Administered Objects Administered Ob ...
- DB天气app冲刺第七天
今天估计得分应该是最差的了. 今天因为完全没准备所以在界面UI上面根本就是相当于没弄.结果今天的各组报告里自己做得很不好.不过好在只是项目的中间期,不碍大事. 今天也完成了任务.可以说超额了也.因为吃 ...
- OpenCV学习-b
OpenCV是开源计算机视觉和机器学习库.包含成千上万优化过的算法.项目地址:http://opencv.org/about.html.官方文档:http://docs.opencv.org/modu ...
- HDU 5008 Boring String Problem
题意:给定一个串长度<=1e5,将其所有的不同的字串按照字典序排序,然后q个询问,每次询问字典序第k小的的起始坐标,并且起始坐标尽量小. 分析: 一开始看错题意,没有意识到是求不同的字串中第k小 ...
- SQLite Version3.3.6源代码文件结构
Version 3.3.6源代码文件结构 ——整理:hustcat 2009-2-19 文件名称 大小by ...
- Android 使控件各占屏幕的一半
在xml中将两个要占屏幕一半的控件都加上android:layout_weight="1": 注意:weight只能用在LinearLayout布局中. 在LinearLayout ...
- 【Quick 3.3】资源脚本加密及热更新(三)热更新模块
[Quick 3.3]资源脚本加密及热更新(三)热更新模块 注:本文基于Quick-cocos2dx-3.3版本编写 一.介绍 lua相对于c++开发的优点之一是代码可以在运行的时候才加载,基于此我们 ...
- 在Sublime Text 3中配置编译和运行Java程序
设置Java的PATH环境变量 在Java的jdk的bin目录下建立新文件runJava.bat 文件内容如下: @echo off cd %~dp1 echo Compiling %~nx1.... ...