码迷,mamicode.com
首页 > 其他好文 > 详细

中断的系统调用

时间:2015-06-26 12:41:32      阅读:125      评论:0      收藏:0      [点我收藏+]

标签:

转自  http://blog.csdn.net/todd911/article/details/17115365

早期的UNIX系统的一个特性是:如果进程在执行一个低速系统调用而阻塞期间捕捉到一个信号,该系统调用就被终端不再

继续执行。该系统调用返回出错,其errno被设置为EINTR。

为了支持这种特性,将系统调用分成两类:低速系统调用和其他系统调用。低速系统调用是可能会使进程永远阻塞的一类

系统调用:

1.在读某些类型的文件(管道,终端设备以及网络设备)时,如果数据并不存在则可能会使调用者永远阻塞。

2.在写这些类型的文件时,如果不能立即接受这些数据,则会使调用者永远阻塞。

3.打开某些类型的文件,在某些条件发生之前也可能会使调用者阻塞(例如,打开终端设备,它要等待直到所连接的调制

解调器应答了电话)

4.pause函数和wait函数

5.某些ioctl函数

6.某些进程间通信函数。

 

与被中断的系统调用相关的问题是必须显式地处理出错返回。典型的代码如下:

 

[cpp] view plaincopy技术分享技术分享
 
  1. again:  
  2. if((n=read(fd,buf,BUFFSIZE))<0){  
  3.     if(errno == EINTR)  
  4.         goto again;  
  5. }  

为了帮助应用程序使其不必处理被中断的系统调用,4.2BSD引入了某些中断系统调用的自动重启动。自动重启动的系统调用包括:

 

ioctl,read,readv,write,writev,wait和waitpid。其中前5个函数只有对低速设备进行操作时才会被信号终端。而wait和waitpid

在捕捉到信号时总是被中断。

POSIX.1允许实现重启动系统调用,但是这并不是必须的。XSI将SA_RESTART定义为对sigaction的XSI扩展,允许应用程序要求

重启被中断的系统调用。

 

实践:

 

[cpp] view plaincopy技术分享技术分享
 
  1. #include <stdio.h>  
  2. #include <stdlib.h>  
  3. #include <unistd.h>  
  4. #include <stdbool.h>  
  5. #include <signal.h>  
  6. #include <sys/types.h>  
  7. #include <errno.h>  
  8. #include <string.h>  
  9.   
  10. void int_handler (int signum)  
  11. {  
  12.           printf ("int handler %d\n",signum);  
  13. }  
  14.   
  15. int main(int argc, char **argv)  
  16. {  
  17.           char buf[100];  
  18.           ssize_t ret;  
  19.           struct sigaction oldact;  
  20.           struct sigaction act;  
  21.   
  22.           act.sa_handler = int_handler;  
  23.           act.sa_flags=0;  
  24.  //       act.sa_flags |= SA_RESTART;  
  25.           sigemptyset(&act.sa_mask);  
  26.           if (-1 == sigaction(SIGINT,&act,&oldact))  
  27.           {  
  28.                   printf("sigaction failed!\n");  
  29.                   return -1;  
  30.           }  
  31.   
  32.           bzero(buf,100);  
  33.   
  34.           ret = read(STDIN_FILENO,buf,10);  
  35.           if (ret == -1)  
  36.           {  
  37.                   printf ("read error %s\n", strerror(errno));  
  38.   
  39.           }  
  40.           printf ("read %d bytes, content is %s\n",ret,buf);  
  41.           sleep (10);  
  42.           return 0;  
  43. }  

运行结果:

 

root@-virtual-machine:~# ./a.out
^Cint handler 2
read error Interrupted system call
read -1 bytes, content is
root@virtual-machine:~#

可见read直接返回,不会重启。

 

下面我们将

act.sa_flags |= SA_RESTART;

的注释打开,再次运行:

root@virtual-machine:~# ./a.out
^Cint handler 2
^Cint handler 2
^Cint handler 2
^Cint handler 2
^Cint handler 2
^Cint handler 2
123
read 4 bytes, content is 123

root@virtual-machine:~#

按了多次ctrl c,read也没有返回,因为read自动重启了。 //运行的结果跟这个不一样啊...???

中断的系统调用

标签:

原文地址:http://www.cnblogs.com/QingCHOW/p/4601832.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!