Linux设备驱动--异步通知

Linux的异步通知是使用信号实现的。

信号:是在软甲层上对中断机制的一种模拟。设备就绪,驱动会主动通知应用程序

Linux中的信号:

请添加图片描述
—— 《linux设备驱动开发详解》

异步通知相关数据结构
struct fasync_struct { 
    spinlock_t          fa_lock; 
    int                magic; 
    int                fa_fd; 
    struct fasync_struct *fa_next;  
    struct file         *fa_file; 
    struct rcu_head       fa_rcu; 
};

file_operations 结构体中与异步通知相关的函数指针:

int (*fasync) (int fd, struct file *filp, int on);

一般在fasync函数中使用以下函数初始化异步通知struct fasync_struct结构体

int fasync_helper(int fd, struct file * filp, int on, struct fasync_struct **fapp) 

应用层上实现异步通知的例子

#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;

  /*  读取并输出 STDIN_FILENO 上的输入  */
  len = read(STDIN_FILENO, &data, MAX_LEN);
  data[len] = 0;
  printf("input available:%s\n", data);
}

int main(int argc, char **argv)
{
  `int oflags;

  /*  启动信号驱动机制  */
    signal(SIGIO, input_handler);
    fcntl(STDIN_FILENO, F_SETOWN, getpid());
    oflags = fcntl(STDIN_FILENO, F_GETFL);
    fcntl(STDIN_FILENO, F_SETFL, oflags | FASYNC); /*  最后进入一个死循环,仅为保持进程不终止,如果程序中
  没有这个死循会立即执行完毕  */
    while (1);
}

在用户处理设备释放的信号的必须关键步骤:

1signal(SIGIO, input_handler);   //安装信号处理函数
    
2fcntl(STDIN_FILENO, F_SETOWN, getpid());  //设置本进程为STDIN_FILENO文件的阿拥有者,没有这一步,内核不知道将信号发给哪个进程。

3、oflags = fcntl(STDIN_FILENO, F_GETFL);
  fcntl(STDIN_FILENO, F_SETFL, oflags | FASYNC);   //启用异步通知机制,必须对设备设置FASYNC标志。

驱动支持异步通知必须完成的工作:

1、支持支持F_SETOWN命令,在这个命令处理中设置filp->f_owner为对应进程ID,此项已经由内核完成。
2、支持F_SETFL命令的处理, 当FASYNC标志改变, 驱动中国的fasybc()函数能够执行。
3、在设备资源可获得时,调用kill_fasync()函数激发相应的信号

请添加图片描述
—— 《linux设备驱动开发详解》

设备驱动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);
}

设备资源可获取时,使用kil_fasync()函数发送SIGIO信号,可读kil_fasync()函数第三个参数设置为POLL_IN,可写kil_fasync()函数第三个参数设置为POLL_OUT

设备关闭realse函数调用设备驱动的fasync()函数将文件从异步通知列表中删除。

static int xxx_release(struct inode *inode, struct file *filp)
{
   /*  将文件从异步通知列表中删除  */
   xxx_fasync(-1, filp, 0);
   ...
   return 0;
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

欲盖弥彰1314

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值