0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
会员中心
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

只需要3分钟就能让你快速了解信号驱动式IO 快来看看吧

GReq_mcu168 来源:嵌入式Hacker 作者:吴伟东Jack 2021-03-18 11:50 次阅读

一、Linux 的 5 种 IO 模型

阻塞式 I/O:

系统调用可能因为无法立即完成而被操作系统挂起,直到等待的事件发生为止。

非阻塞式 I/O (O_NONBLOCK):

系统调用则总是立即返回,而不管事件是否已经发生。

I/O 复用 (select、poll、epoll):

通过 I/O 复用函数向内核注册一组事件,内核通过 I/O 复用函数把其中就绪的事件通知给应用程序。

信号驱动式 I/O (SIGIO):

为一个目标文件描述符指定宿主进程,当文件描述符上有事件发生时,SIGIO 的信号处理函数将被触发,然后便可对目标文件描述符执行 I/O 操作。

异步 I/O (POSIX 的 aio_ 系列函数):

异步 I/O 的读写操作总是立即返回,而不论 I/O 是否是阻塞的,真正的读写操作由内核接管。

思考一下,什么时候应该选择何种 I/O 模型?为何要这么选择?

下面重点关注信号驱动式 I/O 这一模型,其他模型可查阅文末参考书籍。

二、如何使用信号驱动式 I/O?

一般通过如下 6 个步骤来使用信号驱动式 I/O 模型。

1> 为通知信号安装处理函数。

通过 sigaction() 来完成:

intsigaction(intsignum,conststructsigaction*act,structsigaction*oldact);

默认情况下,这个通知信号为 SIGIO。

2> 为文件描述符的设置属主。

通过 fcntl() 的 F_SETOWN 操作来完成:

fcntl(fd,F_SETOWN,pid)

属主是当文件描述符上可执行 I/O 时,会接收到通知信号的进程或进程组。

pid 为正整数时,代表了进程 ID 号。

pid 为负整数时,它的绝对值就代表了进程组 ID 号。

3> 使能非阻塞 I/O。

通过 fcntl() 的 F_SETFL 操作来完成:

flags=fcntl(fd,F_GETFL); fcntl(fd,F_SETFL,flags|O_NONBLOCK);

4> 使能信号驱动 I/O。

通过 fcntl() 的 F_SETFL 操作来完成:

flags=fcntl(fd,F_GETFL); fcntl(fd,F_SETFL,flags|O_ASYNC);

5> 进程等待 "IO 就绪" 信号的到来。

当 I/O 操作就绪时,内核会给进程发送一个信号,然后调用在第 1 步中安装好的信号处理函数。

6> 进程尽可能多地执行 I/O 操作。

循环执行 I/O 系统调用直到失败为止,此时错误码为 EAGAIN 或 EWOULDBLOCK。

原因:

信号驱动 I/O 提供的是边缘触发通知,即只有当 I/O 事件发生时我们才会收到通知,

且当文件描述符收到 I/O 事件通知时,并不知道要处理多少 I/O 数据。

三、内核何时会发送 "IO 就绪" 信号?

对于不同类型的文件描述符,情况不一样。

1> 终端

对于终端,当有新的输入时会会产生信号。

2> 管道和 FIFO

对于读端,下列情况会产生信号:

数据写入到管道中;

管道的写端关闭;

对于写端,下列情况会产生信号:

对管道的读操作增加了管道中的空余空间大小。

管道的读端关闭;

3> 套接字

对于 UDP 套接字,下列情况会产生信号:

数据报到达套接字;

套接字上发生异步错误;

对于 TCP 套接字,信号驱动式 I/O 近乎无用。

太多情况都会产生信号,而我们又无法得知事件类型,因此这里就不再列举其产生信号的情况。

四、最简单的示例

信号处理函数:

staticvolatilesig_atomic_tgotSigio=0; staticvoidhandler(intsig) { gotSigio=1; }

主程序:

intmain(intargc,char*argv[]) { intflags,j,cnt; structtermiosorigTermios; charch; structsigactionsa; intdone; /*Establishhandler*/ sigemptyset(&sa.sa_mask); sa.sa_flags=SA_RESTART; sa.sa_handler=handler; if(sigaction(SIGIO,&sa,NULL)==-1){ perror("sigaction() "); exit(1); } /*Setownerprocess*/ if(fcntl(STDIN_FILENO,F_SETOWN,getpid())==-1){ perror("fcntl()/F_SETOWN "); exit(1); } /*Enable"I/Opossible"signalingandmakeI/Ononblocking*/ flags=fcntl(STDIN_FILENO,F_GETFL); if(fcntl(STDIN_FILENO,F_SETFL,flags|O_ASYNC|O_NONBLOCK)==-1){ perror("fcntl()/F_SETFL "); exit(1); } for(done=0,cnt=0;!done;cnt++){ sleep(1); if(gotSigio){ gotSigio=0; /*Readallavailableinputuntilerror(probablyEAGAIN) orEOF*/ while(read(STDIN_FILENO,&ch,1)>0&&!done){ printf("cnt=%d;read%c ",cnt,ch); done=ch=='#'; } } } exit(0); }

运行效果:

./build/sigio a cnt=0;reada cnt=0;read abc cnt=4;reada cnt=4;readb cnt=4;readc cnt=4;read # cnt=7;read#

该程序会先使能信号驱动 IO,然后循环执行计数操作。

当有 IO 就绪信号到来时,会去终端读取数据并打印出来,然后继续执行计数操作。

五、扩展知识

I/O 多路复用 、信号驱动 I/O 以及 epoll 机制可用于监视多个文件描述符。

它们并不实际执行 I/O 操作,当某个文件描述符处于就绪态,仍需采用传统的 I/O 系统调用来完成 I/O 操作。

相比 I/O 多路复用,当监视大量的文件描述符时信号驱动 I/O 有着显著的性能优势,原因是内核能够帮进程记录了正在监视的文件描述符列表。

信号驱动 I/O 的缺点:

信号的处理流程较为复杂;

无法指定需要监控的事件类型。

Linux 特有的 epoll 是一个更好的选择。

六、相关参考

UNIX 网络编程卷1

6.2 I/O模型

25 信号驱动式I/O

Linux-UNIX 系统编程手册

63 其他备选的I/O模型

Linux 高性能服务器编程

8.3 I/O 模型

Linux 多线程服务端编程_使用muduo C++网络库

7.4.1 muduo的IO模型

编辑:jq

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • 驱动
    +关注

    关注

    12

    文章

    1844

    浏览量

    85359

原文标题:3 分钟快速了解信号驱动式 IO

文章出处:【微信号:mcu168,微信公众号:硬件攻城狮】欢迎添加关注!文章转载请注明出处。

收藏 人收藏

    评论

    相关推荐

    为什么嵌入驱动开发工程师可以拿高薪?

    技术领域具有不可替代的地位。 成为一名优秀的嵌入驱动工程师只需要深耕以下几点: 1)坚实的技术基础: 首先,需要有坚实的基础,包括计算机组成原理、微处理器与微控制器架构以及电路设计
    发表于 01-07 16:56

    MR30分布式IO模块助力PLC,打造高效智能仓储系统

    详细介绍如何通过明达技术MR30分布式IO模块配合西门子PLC,实现智能仓储系统中的各种信号采集与控制。
    的头像 发表于 01-03 10:16 97次阅读
    MR30<b class='flag-5'>分布式</b><b class='flag-5'>IO</b>模块助力PLC,打造高效智能仓储系统

    ADS1115接线是只需要接SCL与SDA吗?ADDR与ALERT/RDY需要接线吗?

    ADS1115接线是只需要接SCL与SDA吗?ADDR与ALERT/RDY需要接线吗? 差输入与单输入有什么区别,怎样配置两种模式?
    发表于 12-31 06:27

    AD模数转换器敷铜是AGND和DGND分开敷,还是只需要在顶层和底层敷AGND?

    AD模数转换器敷铜,双面布板,AGND和DGND已通过0欧姆电阻连接好,我想问,敷铜是AGND和DGND分开敷,还是只需要在顶层和底层敷AGND?
    发表于 12-27 08:16

    ADS1246校正是不是只需要发送校正命令就行了?

    第一次作信号测量工作,好多基础的知识不甚了解 ADS1246校正是不是只需要发送校正命令就行了,后面测量模拟信号转换出来的二进制数直接使用就行了吗?这些数据不
    发表于 12-27 06:19

    双通道AD只需要用一路,另一通道不需要时模拟输入引脚如何处理?

    双通道AD只需要用一路,另一通道不需要时模拟输入引脚如何处理?是可以直接悬空,电阻或电容接地?看手册好像没有提到相关方面的处理办法。或者有没有相关参考原理图
    发表于 12-13 13:30

    请问HD3SS215 aux_sel需要如何做设定?是只需要将电压设定为1/2即可使用吗?

    请问HD3SS215 aux_sel需要如何做设定? 是只需要将电压设定为1/2即可使用吗? 例如3.3V 的1/2电位?
    发表于 12-09 07:27

    想如何运放设计一个完全差输出的电路?

    我想用运放设计一个完全差输出的电路。 信号频率范围:1-5K 输出信号幅度:+ - 10V. 这个信号输出差分器。我有时只需要输了半个正
    发表于 08-29 06:47

    增量编码器3大特点,工作模式,精度,输出脉冲信号 一起了解一下吗

    增量编码器3大特点,工作模式,精度,输出脉冲信号...一起了解一下吗?在现代工业自动化和控制系统中,增量编码器扮演着至关重要的角色。它可
    的头像 发表于 08-15 14:20 724次阅读
    增量<b class='flag-5'>式</b>编码器<b class='flag-5'>3</b>大特点,工作模式,精度,输出脉冲<b class='flag-5'>信号</b> 一起<b class='flag-5'>了解</b>一下吗

    为什么Hssl_MulitRead函数需要2个DMA通道,而Hssl_MulitWrite函数只需要1个DMA通道吗?

    有人能解释为什么 Hssl_MulitRead 函数需要 2 个 DMA 通道,而 Hssl_MulitWrite 函数只需要 1 个 DMA 通道吗?
    发表于 05-24 07:38

    使用带编码器三相电阻采样,三相都有参与运算但实际上输入信号只需要UV两相,为什么?

    如题,使用带编码器三相电阻采样,发现三相都有参与运算,但实际上输入信号只需要UV两相,W相输入数据不会影响电机运行效果,请问这是什么原因?出现电磁噪音比较大,有那些方法可以解决?如果都能解决,可以提供有偿服务。
    发表于 04-08 08:08

    勋瑞光电科技:高端显示器首选IPS面板?3分钟带你了解强大之处!

    在当今数字化生活中,显示器的选择对于我们的日常工作和娱乐至关重要。而当提到高端显示器,IPS(In-Plane Switching)屏幕往往备受推崇。 那么,为什么IPS屏幕如此受欢迎?它究竟有何独特之处?跟着勋瑞光电科技一起在3分钟内深入
    的头像 发表于 03-22 09:28 473次阅读

    3分钟快速实现串口PLC远程下载程序操作说明

    3分钟快速实现串口PLC远程下载程序操作说明 搜索蓝蜂物联网官网,即可免费领取样机使用!!先到先得!!! 一. 适用产品型号 其余型号网关此功能正在开发中,敬请期待。 二. 远程下载功能使用流程
    的头像 发表于 02-26 15:38 1068次阅读
    <b class='flag-5'>3</b><b class='flag-5'>分钟</b><b class='flag-5'>快速</b>实现串口PLC远程下载程序操作说明

    分钟了解飞创直线电机运动模组特点、选型及应用-FCL系列

    分钟了解飞创直线电机运动模组特点、选型及应用-FCL系列
    的头像 发表于 02-05 16:28 611次阅读
    三<b class='flag-5'>分钟</b><b class='flag-5'>了解</b>飞创直线电机运动模组特点、选型及应用-FCL系列

    想从电脑下载程序到ADuc845,是不是只需要RXD、TXD两个引脚就够了?

    麻烦问一下: 1、如果想从电脑下载程序到芯片内,是不是只需要RXD、TXD两个引脚就够了? 2、不想把下载程序的电路放到目标板上,想单独做一个板子,通过电脑上的USB口进行下载。中间需要芯片
    发表于 01-12 06:03