linux信号驱动IO(高级字符设备四)

一、linux信号驱动IO介绍

  信号驱动 IO 不需要应用程序查询设备的状态,一旦设备准备就绪,会触发SIGIO信号,进而调用注册的处理函数。

1.1、linux信号驱动IO应用层

  如果要实现信号驱动 IO,需要应用程序和驱动程序配合,应用程序使用信号驱动IO的步骤有三步:
  步骤 1 :注册信号处理函数 应用程序使用 signal 函数来注册SIGIO 信号的信号处理函数。
  步骤 2: 设置能够接收这个信号的进程
  步骤 3: 开启信号驱动 IO 通常使用 fcntl 函数的 F_SETFL 命令打开FASYNC 标志。
  fcntl 函数如下所示:
  函数原型:
    int fcntl(int fd,int cmd, …)
  函数功能:
    fcntl 函数可以用来操作文件描述符
  函数参数:
    fd: 被操作的文件描述符
    cmd: 操作文件描述符的命令,cmd 参数决定了要如何操作文件描述符fd
    …: 根据 cmd 的参数来决定是不是需要使用第三个参数
  操作文件描述符的命令如下表

命令名描述
F_DUPFD复制文件描述符
F_GETFD获取文件描述符标志
F_SETFD设置文件描述符标志
F_GETFL获取文件状态标志
F_SETFL设置文件状态标志
F_GETLK获取文件锁
F_SETLK设置文件锁
F_SETLKW类似 F_SETLK,但等待返回
F_GETOWN获取当前接收 SIGIO 和SIGURG 信号的进程ID和进程组 ID
F_SETOWN设置当前接收 SIGIO 和SIGURG 信号的进程ID和进程组 ID

1.2、linux信号驱动IO驱动层

  当应用程序开启信号驱动 IO 时,会触发驱动中的 fasync 函数。所以首先在file_operations结构体中实现 fasync 函数,函数原型如下:

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

  在驱动中的 fasync 函数调用 fasync_helper 函数来操作 fasync_struct 结构体,fasync_helper函数原型如下:

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

  当设备准备好的时候,驱动程序需要调用 kill_fasync 函数通知应用程序,此时应用程序的SIGIO 信号处理函数就会被执行。kill_fasync 负责发送指定的信号,函数原型如下:

void kill_fasync(struct fasync_struct **fp,int sig,int band)

  函数参数:
    fp: 要操作的 fasync_struct
    sig: 发送的信号
    band: 可读的时候设置成 POLLIN ,可写的时候设置成POLLOUT

二、代码示例

2.1、应用层程序

#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>
#include <stdlib.h>
#include <poll.h>
#include <fcntl.h>
#include <signal.h>int fd;
char buf1[32] = {0};   //SIGIO信号的信号处理函数
static void func(int signum)
{read(fd,buf1,32);printf ("buf is %s\n",buf1);
}
int main(int argc, char *argv[])  
{int ret;int flags;fd = open("/dev/test", O_RDWR);  //打开led驱动if (fd < 0){perror("open error \n");return fd;}signal(SIGIO,func);  //步骤一:使用signal函数注册SIGIO信号的信号处理函数//步骤二:设置能接收这个信号的进程//fcntl函数用来操作文件描述符,//F_SETOWN 设置当前接收的SIGIO的进程IDfcntl(fd,F_SETOWN,getpid()); flags = fcntl(fd,F_GETFD); //获取文件描述符标志//步骤三  开启信号驱动IO 使用fcntl函数的F_SETFL命令打开FASYNC标志fcntl(fd,F_SETFL,flags| FASYNC);while(1);close(fd);     //关闭文件return 0;
}

2.2、驱动层程序

#include <linux/init.h>
#include <linux/module.h>
#include <linux/kdev_t.h>
#include <linux/fs.h>
#include <linux/cdev.h>
#include <linux/uaccess.h>
#include <linux/io.h>
#include  <linux/wait.h>
#include <linux/poll.h>
#include <linux/fcntl.h>
#include <linux/signal.h>struct device_test{dev_t dev_num;  //设备号int major ;  //主设备号int minor ;  //次设备号struct cdev cdev_test; // cdevstruct class *class;   //类struct device *device; //设备char kbuf[32];int  flag;  //标志位struct fasync_struct *fasync;
};struct  device_test dev1;  DECLARE_WAIT_QUEUE_HEAD(read_wq); //定义并初始化等待队列头/*打开设备函数*/
static int cdev_test_open(struct inode *inode, struct file *file)
{file->private_data=&dev1;//设置私有数据return 0;
}/*向设备写入数据函数*/
static ssize_t cdev_test_write(struct file *file, const char __user *buf, size_t size, loff_t *off)
{struct device_test *test_dev=(struct device_test *)file->private_data;if (copy_from_user(test_dev->kbuf, buf, size) != 0) // copy_from_user:用户空间向内核空间传数据{printk("copy_from_user error\r\n");return -1;}test_dev->flag=1;wake_up_interruptible(&read_wq);kill_fasync(&test_dev->fasync,SIGIO,POLLIN);return 0;
}/**从设备读取数据*/
static ssize_t cdev_test_read(struct file *file, char __user *buf, size_t size, loff_t *off)
{struct device_test *test_dev=(struct device_test *)file->private_data;if(file->f_flags & O_NONBLOCK ){if (test_dev->flag !=1)return -EAGAIN;}wait_event_interruptible(read_wq,test_dev->flag);if (copy_to_user(buf, test_dev->kbuf, strlen( test_dev->kbuf)) != 0) // copy_to_user:内核空间向用户空间传数据{printk("copy_to_user error\r\n");return -1;}return 0;
}static int cdev_test_release(struct inode *inode, struct file *file)
{return 0;
}static  __poll_t  cdev_test_poll(struct file *file, struct poll_table_struct *p){struct device_test *test_dev=(struct device_test *)file->private_data;  //设置私有数据__poll_t mask=0;    poll_wait(file,&read_wq,p);     //应用阻塞if (test_dev->flag == 1)    {mask |= POLLIN;}return mask;}static int cdev_test_fasync (int fd, struct file *file, int on)
{struct device_test *test_dev=(struct device_test *)file->private_data;  //设置私有数据return  fasync_helper(fd,file,on,&test_dev->fasync);
}
/*设备操作函数*/
struct file_operations cdev_test_fops = {.owner = THIS_MODULE, //将owner字段指向本模块,可以避免在模块的操作正在被使用时卸载该模块.open = cdev_test_open, //将open字段指向chrdev_open(...)函数.read = cdev_test_read, //将open字段指向chrdev_read(...)函数.write = cdev_test_write, //将open字段指向chrdev_write(...)函数.release = cdev_test_release, //将open字段指向chrdev_release(...)函数.poll = cdev_test_poll,  //将poll字段指向chrdev_poll(...)函数.fasync = cdev_test_fasync,   //将fasync字段指向cdev_test_fasync(...)函数
};static int __init chr_fops_init(void) //驱动入口函数
{/*注册字符设备驱动*/int ret;/*1 创建设备号*/ret = alloc_chrdev_region(&dev1.dev_num, 0, 1, "alloc_name"); //动态分配设备号if (ret < 0){goto err_chrdev;}printk("alloc_chrdev_region is ok\n");dev1.major = MAJOR(dev1.dev_num); //获取主设备号dev1.minor = MINOR(dev1.dev_num); //获取次设备号printk("major is %d \r\n", dev1.major); //打印主设备号printk("minor is %d \r\n", dev1.minor); //打印次设备号/*2 初始化cdev*/dev1.cdev_test.owner = THIS_MODULE;cdev_init(&dev1.cdev_test, &cdev_test_fops);/*3 添加一个cdev,完成字符设备注册到内核*/ret =  cdev_add(&dev1.cdev_test, dev1.dev_num, 1);if(ret<0){goto  err_chr_add;}/*4 创建类*/dev1. class = class_create(THIS_MODULE, "test");if(IS_ERR(dev1.class)){ret=PTR_ERR(dev1.class);goto err_class_create;}/*5  创建设备*/dev1.device = device_create(dev1.class, NULL, dev1.dev_num, NULL, "test");if(IS_ERR(dev1.device)){ret=PTR_ERR(dev1.device);goto err_device_create;}return 0;err_device_create:class_destroy(dev1.class);                 //删除类err_class_create:cdev_del(&dev1.cdev_test);                 //删除cdeverr_chr_add:unregister_chrdev_region(dev1.dev_num, 1); //注销设备号err_chrdev:return ret;
}static void __exit chr_fops_exit(void) //驱动出口函数
{/*注销字符设备*/unregister_chrdev_region(dev1.dev_num, 1); //注销设备号cdev_del(&dev1.cdev_test);                 //删除cdevdevice_destroy(dev1.class, dev1.dev_num);       //删除设备class_destroy(dev1.class);                 //删除类
}
module_init(chr_fops_init);
module_exit(chr_fops_exit);

2.3、linux信号驱动IO使用API要点

应用层

    signal(SIGIO,func);  //步骤一:使用signal函数注册SIGIO信号的信号处理函数//步骤二:设置能接收这个信号的进程//fcntl函数用来操作文件描述符,//F_SETOWN 设置当前接收的SIGIO的进程IDfcntl(fd,F_SETOWN,getpid()); flags = fcntl(fd,F_GETFD); //获取文件描述符标志//步骤三  开启信号驱动IO 使用fcntl函数的F_SETFL命令打开FASYNC标志fcntl(fd,F_SETFL,flags| FASYNC);

驱动层

kill_fasync(&test_dev->fasync,SIGIO,POLLIN);
static int cdev_test_fasync (int fd, struct file *file, int on)
{struct device_test *test_dev=(struct device_test *)file->private_data;  //设置私有数据return  fasync_helper(fd,file,on,&test_dev->fasync);
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/119678.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

SpringSecurity分布式安全框架

Spring Security是一个基于Spring框架的安全框架&#xff0c;它提供了全面的安全解决方案&#xff0c;包括用户认证和用户授权等Web应用安全性问题。Spring Security可以轻松扩展以满足自定义需求&#xff0c;它的真正强大之处在于它可以轻松扩展以满足自定义要求。 对于分布式…

JavaWeb——IDEA操作:Project最终新建module

在project中创建新的module&#xff1a; 创建一个新的module很容易&#xff0c;但是它可能连接不上Tomcat&#xff0c;因此需要修改一些配置&#xff1a; 将以下地址修改为新module的地址

PROFINET通信介绍

S7-1200和汇川变频器的PROFINET通信应用&#xff0c;请参考下面文章链接&#xff1a; PN通信组态(汇川变频器和S7-1200PN通信)-CSDN博客文章浏览阅读1.2k次。ABB变频器的PN通信相关设置&#xff0c;请参看下面的文章链接博途PLC和ABB变频器PN通讯详解_abb 变频器 pn通信_RXXW_…

【JavaEE初阶】 线程安全的集合类

文章目录 &#x1f340;前言&#x1f332;多线程环境使用 ArrayList&#x1f6a9;自己使用同步机制 (synchronized 或者 ReentrantLock)&#x1f6a9;Collections.synchronizedList(new ArrayList);&#x1f6a9;使用 CopyOnWriteArrayList &#x1f38d;多线程环境使用队列&am…

echarts-进度条

echarts-进度条 option {title: {text:"xxxx统计",left: 1%,top: 0%,textStyle: {color: "#2E3033",fontSize:18,},},tooltip: {axisPointer: {type: "shadow",},},grid: {top: 9%,left: "12%",right:"22%",bottom:"0…

react笔记基础部分(组件生命周期路由)

注意点&#xff1a; class是一个关键字&#xff0c; 类。 所以react 写class, 用classname &#xff0c;会自动编译替换class 点击方法&#xff1a; <button onClick {this.sendData}>给父元素传值</button>常用的插件&#xff1a; 需要引入才能使用的&#xf…

如何使用vim粘贴鼠标复制的内容

文章目录 一、使用步骤1.找到要编辑的配置文件2.找到目标文件3.再回到vim编辑器 一、使用步骤 1.找到要编辑的配置文件 用sudo vim /etc/apt/sources.list编辑软件源配置文件 sudo vim /etc/apt/sources.listvim 在默认的情况下当鼠标选中的时候进入的 Visual 模式&#xff…

2520. 统计能整除数字的位数 --力扣 --JAVA

题目 给你一个整数 num &#xff0c;返回 num 中能整除 num 的数位的数目。 如果满足 nums % val 0 &#xff0c;则认为整数 val 可以整除 nums 。 解题思路 将整数num转换成字符串并读取每位字符进行运算 代码展示 class Solution {public int countDigits(int num) {int …

GitLab升级16.5.0后访问提示502

系统是兼容CentOS8的TencentOS3.1 GitLab原来的版本是16.4.1 使用yum升级时发现GitLab有新版本,决定升级。 升级过程无异常,出现升级成功的提示。 可是意外的时,访问站点时提示502. GitLab比较吃资源,启动的服务较多。之前也有等会就正常的情况。 这次没那么幸运,一…

Go语言入门心法(十四): Go操作Redis实战

Go语言入门心法(一): 基础语法 Go语言入门心法(二): 结构体 Go语言入门心法(三): 接口 Go语言入门心法(四): 异常体系 Go语言入门心法(五): 函数 Go语言入门心法(六): HTTP面向客户端|服务端编程 Go语言入门心法(七): 并发与通道 Go语言入门心法(八): mysql驱动安装报错o…

SpringMVC 异常处理器

1、基于配置的异常处理 SpringMVC提供了一个处理控制器方法执行过程中所出现的异常的接口&#xff1a;HandlerExceptionResolver HandlerExceptionResolver接口的实现类有&#xff1a;DefaultHandlerExceptionResolver和SimpleMappingExceptionResolver SpringMVC提供了自定…

【ARM Cortex-M 系列 4 番外篇 -- 常用 benchmark 介绍】

文章目录 1.1 CPU 性能测试 MIPS 计算1.1.1 Cortex-M7 CPI 1.2 benchmark 小节1.3.1 Geekbenck 介绍 1.3 编译参数配置 1.1 CPU 性能测试 MIPS 计算 每秒百万指令数 (MIPS)&#xff1a;在数据压缩测试中&#xff0c;MIPS 每秒测量一次 CPU 执行的低级指令的数量。越高越好&…

D71X-16Q手柄蝶阀型号解析

D71X-16Q型号字母含义解析 D71X-16Q是德特森阀门常用的手柄蝶阀型号字母分别代表的意思是: D——代表阀门类型《蝶阀》 7——代表连接方式《对夹》 1——代表结构形式《中线》 X——代表阀座材质《橡胶》 -代表分隔键 16——代表公称压力《1.6MPA》 Q——代表阀体材料《…

美创科技列为IDC中国数据安全市场代表厂商

近日&#xff0c;国际权威IT咨询机构IDC发布《中国数据安全市场发展趋势&#xff0c;2023》报告&#xff0c;报告针对中国数据安全市场的发展现状进行调研&#xff0c;明确了最终用户数据安全建设的痛点、难点&#xff0c;阐述了市场中各技术服务提供商的服务方案和优势。 美创…

如何恢复u盘删除文件?2023最新分享四种方法恢复文件

U盘上删除的文件怎么恢复&#xff1f;使用U盘存储文件是非常方便的&#xff0c;例如&#xff1a;在办公的时候&#xff0c;会使用U盘来存储网络上查找到的资料、产品说明等。在学习的时候&#xff0c;会使用U盘来存储教育机构分享的教学视频、重点知识等。而随着U盘存储文件的概…

css步骤条

html 代码以及样式 <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><title>css步骤条</title><style>.steps {display: flex;justify-content: space-between;padding: 0;margin: 20px 10px;lis…

数据结构笔记——树和图(王道408)(持续更新)

文章目录 传送门前言树&#xff08;重点&#xff09;树的数据结构定义性质 二叉树的数据结构定义性质储存结构 二叉树算法先中后序遍历层次展开法递归模拟法 层次遍历遍历序列逆向构造二叉树 线索二叉树&#xff08;难点&#xff09;定义线索化的本质 二叉树线索化线索二叉树中…

Elasticsearch分词器-中文分词器ik

文章目录 使用standard analysis对英文进行分词使用standard analysis对中文进行分词安装插件对中文进行友好分词-ik中文分词器下载安装和配置IK分词器使用ik_smart分词器使用ik_max_word分词器 借助Nginx实现ik分词器自定义分词网络新词 ES官方文档Text Analysis 使用standard…

ES修改字段类型详解

一、需求概述 ES修改字段类型是指在已有的索引中&#xff0c;通过特定的操作方式将某个字段的类型修改为其它类型。当ES在建立索引的时候&#xff0c;已经确定好了每个字段的类型&#xff0c;而如果在建立后发现类型不符需求&#xff0c;就需要修改字段类型。 二、修改字段类…

react-typescript-demo

1.使用 Context 来存储数据