免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
最近访问板块 发新帖
查看: 5157 | 回复: 6
打印 上一主题 下一主题

[Linux] 标准输入使用信号驱动I/O,无限捕捉SIGIO [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2015-03-12 22:58 |只看该作者 |倒序浏览
本帖最后由 McHeaven 于 2015-03-12 23:02 编辑

代码如下,打算设置标准输入为信号驱动I/O,
当输入内容的时候能够触发signal handler进行处理

  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #include <unistd.h>
  4. #include <errno.h>
  5. #include <fcntl.h>
  6. #include <signal.h>


  7. void sigio_handler(int sig)
  8. {
  9.     static int cnt = 0;

  10.     printf("receive SIGIO signal %d\n", ++cnt);
  11.     fflush(stdin);
  12.     fflush(stdout);
  13. }

  14. int main(int argc, char **argv)
  15. {
  16.     struct sigaction sig_act;
  17.     int fl = 0;

  18.     //signal driven I/O setting
  19.     fflush(stdin);
  20.     fflush(stdout);
  21.     fl = fcntl(STDIN_FILENO, F_GETFL, 0);
  22.     fcntl(STDIN_FILENO, F_SETFL, fl | O_NONBLOCK | O_ASYNC );
  23.     fcntl(STDIN_FILENO, F_SETOWN, getpid());
  24.     sigemptyset(&sig_act.sa_mask);
  25.     sig_act.sa_flags = 0;
  26.     sig_act.sa_handler = sigio_handler;
  27.     sigaction(SIGIO, &sig_act, NULL);

  28.     printf("test start\n");
  29.     fflush(stdin);
  30.     fflush(stdout);

  31.     while (1)
  32.     {
  33.         sleep(1);
  34.     }

  35.     return 0;
  36. }
复制代码
运行的结果是signal handler被无限调用,也就是说SIGIO不断产生,结果如下:

  1. test start
  2. receive SIGIO signal 1
  3. receive SIGIO signal 2
  4. receive SIGIO signal 3
  5. receive SIGIO signal 4
  6. receive SIGIO signal 5
  7. receive SIGIO signal 6
  8. receive SIGIO signal 7
  9. receive SIGIO signal 8
  10. receive SIGIO signal 9
  11. receive SIGIO signal 10
  12. receive SIGIO signal 11
  13. receive SIGIO signal 12
  14. receive SIGIO signal 13
  15. receive SIGIO signal 14
  16. receive SIGIO signal 15
  17. receive SIGIO signal 16
  18. receive SIGIO signal 17
  19. receive SIGIO signal 18
  20. ...
复制代码
现在想知道标准输入/输出产生SIGIO的情况是什么
现在遇到的情况是不是因为标准输出也会产生SIGIO信号,不停触发signal handler
可是我绑定时候使用的是STDIN_FILENO,而且每次printf了之后也使用fflush对缓冲区进行了冲洗
请指教,谢谢

论坛徽章:
1
2015年辞旧岁徽章
日期:2015-03-03 16:54:15
2 [报告]
发表于 2015-03-23 17:07 |只看该作者
本帖最后由 羽剑天涯 于 2015-03-23 17:08 编辑

回复 1# McHeaven


    感觉\n和fflush(stdout)(在有输出缓冲时)都会在stdin中产生输入(回车输入?),你的中断函数中printf有\n(同时有fflush(stdout)),则此时又触发一次中断,如此进入了恶性循环,从而SIGIO无限触发。你把这两处去掉,计数放在程序退出前打印,就不会这样了,修改后的代码:
  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #include <unistd.h>
  4. #include <errno.h>
  5. #include <fcntl.h>
  6. #include <signal.h>
  7. #include <time.h>

  8. static int cnt = 0;
  9. void sigio_handler(int sig)
  10. {

  11.     printf("receive SIGIO signal %d", ++cnt);
  12.     fflush(stdin);
  13.     //fflush(stdout);
  14. }

  15. int main(int argc, char **argv)
  16. {
  17.     struct sigaction sig_act;
  18.     int fl = 0;

  19.     //signal driven I/O setting
  20.     fflush(stdin);
  21.     fflush(stdout);
  22.     fl = fcntl(STDIN_FILENO, F_GETFL, 0);
  23.     fcntl(STDIN_FILENO, F_SETFL, fl | O_NONBLOCK | O_ASYNC );
  24.     fcntl(STDIN_FILENO, F_SETOWN, getpid());
  25.     sigemptyset(&sig_act.sa_mask);
  26.     sig_act.sa_flags = 0;
  27.     sig_act.sa_handler = sigio_handler;
  28.     sigaction(SIGIO, &sig_act, NULL);

  29.     printf("test start\n");
  30.     fflush(stdin);
  31.     fflush(stdout);

  32.         time_t st=time(NULL);
  33.     while (time(NULL) < st+5)
  34.     {
  35.         sleep(1);
  36.     }
  37.     printf("\ncnt=%d\n",cnt);

  38.     return 0;
  39. }
复制代码

论坛徽章:
0
3 [报告]
发表于 2015-03-24 10:13 |只看该作者
我自己也做了几个实验,得到这样的结论:

对于stdin绑定了信号驱动I/O的进程,在stdout上的输出能触发SIGIO信号需要满足:stdout缓冲区中有数据,stdout缓冲区必须被冲洗;而与缓冲区中数据大小无关。在Linux 2.6上,每键入一个字符就会触发一次SIGIO信号,包括回车键;而在Linux 3.13上,只有敲入了回车键之后才会触发一次SIGIO信号。

这是我写的总结,有兴趣可以帮我看看有什么疏漏的:
blog.csdn.net/mcheaven/article/details/44303753

论坛徽章:
1
2015年辞旧岁徽章
日期:2015-03-03 16:54:15
4 [报告]
发表于 2015-03-24 15:58 |只看该作者
回复 3# McHeaven


    我觉得应该是这样:标准输入和标准输出(包括错误输出)都是关联在同一个”设备“(如pts、tty等)上,则这个”设备“上有IO输入输出时都会产生SIGIO(在这个”设备“的fd(s)上有开启O_ASYNC的情况下)。
    你总结中做的那些实验,都是因为确实产生了输出(如果缓冲了,就没有产生输出),才会产生SIGIO,你可以尝试关闭输出的缓冲setbuf(stdout,NULL),这样的话,只要有输出(不必一定有换号和刷新缓冲),就会立刻显示出来,同时就会产生SIGIO信号。
    另外,我认为,试验中的按键输入产生的中断,在2.6和3.13中不同,也是因为输入缓冲实现不同(2.6下没有这个缓冲?估计3.13也有什么设置可以关闭这个缓冲),如果进入缓冲(此时驱动认为不需要通知应用),则不会产生SIGIO,这样也可以解释,你输入4、回车、5,最后4和回车被程序捕获了,而5最后遗留在外边,测试如图:

    对应的代码:
  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #include <unistd.h>
  4. #include <errno.h>
  5. #include <fcntl.h>
  6. #include <signal.h>
  7. #include <time.h>

  8. static int cnt = 0;
  9. void sigio_handler(int sig)
  10. {
  11.     char buf[100];
  12.     cnt++;
  13.     read(STDIN_FILENO, buf, 100);
  14. }

  15. int main(int argc, char **argv)
  16. {
  17.     struct sigaction sig_act;
  18.     int fl = 0;
  19.     setbuf(stdin, NULL);
  20.     setbuf(stdout,NULL);
  21.     fl = fcntl(STDIN_FILENO, F_GETFL, 0);
  22.     fcntl(STDIN_FILENO, F_SETFL, fl | O_NONBLOCK | O_ASYNC );
  23.     fcntl(STDIN_FILENO, F_SETOWN, getpid());
  24.     sigemptyset(&sig_act.sa_mask);
  25.     sig_act.sa_flags = 0;
  26.     sig_act.sa_handler = sigio_handler;
  27.     sigaction(SIGIO, &sig_act, NULL);

  28.     time_t st=time(NULL);
  29.     while (time(NULL) < st+5)
  30.     {
  31.         sleep(1);
  32.     }
  33.     printf("\ncnt=%d\n", cnt);

  34.     return 0;
  35. }
复制代码
我认为,其实,SIGIO信号的产生完全由对应”设备“的驱动程序实现决定,它什么时候调用kill_fasync,应用就什么时候收到对应信号,如果驱动中认为不需要发送信号(如缓冲掉),则程序就不会收到对应的信号。

论坛徽章:
1
2015年辞旧岁徽章
日期:2015-03-03 16:54:15
5 [报告]
发表于 2015-03-24 17:15 |只看该作者
另外3.13中缓冲的事情也可以解决,你把输入输出的终端设置为非标准模式(non-canonical mode),此时也可以按键就触发SIGIO,代码如下:
  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #include <unistd.h>
  4. #include <errno.h>
  5. #include <fcntl.h>
  6. #include <signal.h>
  7. #include <time.h>
  8. #include <termios.h>

  9. static int cnt = 0;
  10. char buf[100];
  11. int l=0;
  12. void sigio_handler(int sig)
  13. {
  14.     cnt++;
  15.     while (l<100 && read(STDIN_FILENO, buf+l, 1) == 1) {
  16.                 l++;
  17.     }

  18. int main(int argc, char **argv)
  19. {
  20.     int fl = 0;
  21.     struct termios o={0},n={0};

  22.         setbuf(stdin, NULL);
  23.         setbuf(stdout,NULL);

  24.         cfmakeraw(&n);
  25.         tcgetattr(STDIN_FILENO,&o);
  26.         tcsetattr(STDIN_FILENO,TCSANOW,&n);
  27.     fl = fcntl(STDIN_FILENO, F_GETFL, 0);
  28.     fcntl(STDIN_FILENO, F_SETFL, fl | O_NONBLOCK | O_ASYNC );
  29.     fcntl(STDIN_FILENO, F_SETOWN, getpid());
  30.     fflush(stdin);
  31.     fflush(stdout);
  32.     signal(SIGIO, sigio_handler);

  33.         time_t st=time(NULL);
  34.     while (time(NULL) < st+5)
  35.     {
  36.         sleep(1);
  37.     }
  38.     signal(SIGIO, SIG_IGN);
  39.     printf("\ncnt=%d,l=%d\n", cnt,l);
  40.     int i=0;

  41.     for (i=0; i<l; i++) {
  42.                 printf("%d,",buf[i]);
  43.     }
  44.     printf("\n");
  45.         tcsetattr(STDIN_FILENO, TCSANOW, &o);

  46.     return 0;
  47. }
复制代码

论坛徽章:
0
6 [报告]
发表于 2015-03-24 23:09 |只看该作者
感谢回复,你的思考让我加深了理解
关于键入字符触发SIGIO,两个版本表现不同应该就是终端设置的原因

让我费解的是:
程序里面O_ASYNC标志绑定的是标准输入的描述符
为什么标准输出上的成功输出会产生SIGIO信号
两者描述符是不同的
难道是驱动程序认为只要指定了0/1/2中其中一个是O_ASYNC,其他的也都是?
或者两者共用了什么东西的缘故?

论坛徽章:
1
2015年辞旧岁徽章
日期:2015-03-03 16:54:15
7 [报告]
发表于 2015-03-25 12:32 |只看该作者
回复 6# McHeaven


    0-2其实用的是同一个设备(驱动理论上肯定是一个),如下图所示
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

北京盛拓优讯信息技术有限公司. 版权所有 京ICP备16024965号-6 北京市公安局海淀分局网监中心备案编号:11010802020122 niuxiaotong@pcpop.com 17352615567
未成年举报专区
中国互联网协会会员  联系我们:huangweiwei@itpub.net
感谢所有关心和支持过ChinaUnix的朋友们 转载本站内容请注明原作者名及出处

清除 Cookies - ChinaUnix - Archiver - WAP - TOP