免费注册 查看新帖 |

Chinaunix

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

请教信号中断之后重启的问题 [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2007-07-24 17:11 |只看该作者 |倒序浏览
有时候在IO读取时需要设置超时, 但是Linux下面的write ,read 系统调用是自动重启的, 所以下面的代码不能控制超时:

  1. void sig_alrm(int signo)
  2. {
  3.         return ;
  4. }

  5. int main()
  6. {
  7.         int n;
  8.         char line[100] = {0};
  9.         if (signal(SIGALRM, sig_alrm) == SIG_ERR)
  10.         {
  11.                 perror("signal\n");
  12.                 exit(1);
  13.         }

  14.         alarm(5);
  15.         if ((n = read(STDIN_FILENO, line, 100)) < 0)
  16.         {
  17.                 if (errno == EINTR)
  18.                 {
  19.                         printf("yes\n");
  20.                 }
  21.                 perror("read\n");
  22.                 exit(1);
  23.         }
  24.         alarm(0);
  25.         return 0;
  26. }
复制代码

后来改用了sigaction就可以处理这个问题了, 即:

  1. void sig_alrm(int signo)
  2. {
  3.         return ;
  4. }

  5. int main()
  6. {
  7.         int n;
  8.         char line[100];
  9.         struct sigaction act, oact;
  10.         act.sa_handler = sig_alrm;
  11.         sigemptyset(&act.sa_mask);
  12.       
  13.         act.sa_flags |= SA_INTERRUPT;

  14.         if (sigaction(SIGALRM, &act, &oact) < 0)
  15.         {
  16.                 perror("sigaction\n");
  17.                 exit(1);
  18.         }
  19.         alarm(4);
  20.         if ((n = read(STDIN_FILENO, line, 100)) < 0)
  21.         {
  22.                 if (errno == EINTR)
  23.                 {
  24.                         printf("yes\n");
  25.                 }
  26.                 perror("read\n");
  27.                 exit(1);
  28.         }

  29.         printf("done\n");
  30.         return 0;
  31. }

复制代码

就是设置成SA_INTERRUPT, 这样就可以打断系统调用read了. 但是我想请教的问题是为什么我把act.sa_flags != SA_INTERRUPT; 这一行去掉之后, 仍然可以进行超时控制呢?就是说仍然工作正常, 设不设置这一位, 系统调用都不会重启. 而之前的signal却总是会重启的. 请教一下是怎么回事? 谢谢大家!

论坛徽章:
0
2 [报告]
发表于 2007-07-24 18:16 |只看该作者
这是否说明了:
使用signal注册的信号处理函数, 对于read, write这些系统调用默认情况下是会重启被中断的系统调用,
而使用sigaction注册的信号处理函数, 默认情况下是不会重启被中断的系统调用, 如果需要使用其能够自动重启, 那么就设置
SA_RESTART位呢?
谢谢大家指教!

论坛徽章:
0
3 [报告]
发表于 2007-07-24 19:06 |只看该作者
首先,局部变量的值是不会自动初始化的,你怎知道原来的值中没有SA_INTERRUPT?
其次,Linux系统中的signal函数遵从的是BSD风格(所以重启),而sigaction遵从的是POSIX。signal是有歧义的,今后永远不应该再使用,全部改用sigaction。
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

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

清除 Cookies - ChinaUnix - Archiver - WAP - TOP