中断产生EINTR错误

https://blog.csdn.net/u011068702/article/details/62069714

1、介绍慢系统调用

该术语适用于那些可能永远阻塞的系统调用。永远阻塞的系统调用是指调用永远无法返回,多数网络支持函数都属于这一类。如:若没有客户连接到服务器上,那么服务器的accept调用就会一直阻塞。

慢系统调用可以被永久阻塞,包括以下几个类别:

(1)读写‘慢’设备(包括pipe,终端设备,网络连接等)。读时,数据不存在,需要等待;写时,缓冲区满或其他原因,需要等待。读写磁盘文件一般不会阻塞。
(2)当打开某些特殊文件时,需要等待某些条件,才能打开。例如:打开中断设备时,需要等到连接设备的modem响应才能完成。
(3)pause和wait函数。pause函数使调用进程睡眠,直到捕获到一个信号。wait等待子进程终止。
(4)某些ioctl操作。
(5)某些IPC操作。



2、EINTR错误产生的原因

如果进程在一个慢系统调用(slow system call)中阻塞时,当捕获到某个信号且相应信号处理函数返回时,这个系统调用被中断,调用返回错误,设置errno为EINTR(相应的错误描述为“Interrupted system call”)。

如下表所示的系统调用就会产生EINTR错误,当然不同的函数意义也不同。


系统调用函数
    
errno为EINTR表征的意义

[html]  view plain  copy
  1. write  
  2. 由于信号中断,没写成功任何数据。  
  3.   
  4. The call was interrupted by a signal before any data was written.  
  5. open  
  6.       
  7. 由于信号中断,没读到任何数据。  
  8. The call was interrupted by a signal before any data was read.  
  9. recv  
  10.       
  11. 由于信号中断返回,没有任何数据可用。  
  12. The receive was interrupted by delivery of a signal before any data were available.  
  13. sem_wait  
  14.       
  15. 函数调用被信号处理函数中断。  
  16. The call was interrupted by a signal handler.  


3、解决办法

既然系统调用会被中断,那么别忘了要处理被中断的系统调用。有三种处理方式:


1 人为重启被中断的系统调用


当碰到EINTR错误的时候,有一些可以重启的系统调用要进行重启,而对于有一些系统调用是不能够重启的。例如:accept、read、write、select、和open之类的函数来说,是可以进行重启的。不过对于套接字编程中的connect函数我们是不能重启的,若connect函数返回一个EINTR错误的时候,我们不能再次调用它,否则将立即返回一个错误。针对connect不能重启的处理方法是,必须调用select来等待连接完成。

理解“重启”?

一些IO系统调用执行时,如 read 等待输入期间,如果收到一个信号,系统将中断read, 转而执行信号处理函数. 当信号处理返回后, 系统遇到了一个问题: 是重新开始这个系统调用, 还是让系统调用失败?早期UNIX系统的做法是, 中断系统调用,并让系统调用失败, 比如read返回 -1, 同时设置 errno 为EINTR中断了的系统调用是没有完成的调用,它的失败是临时性的,如果再次调用则可能成功,这并不是真正的失败,所以要对这种情况进行处理, 典型的方式为如下,我们采用accept函数为例子,代码如下


[html]  view plain  copy
  1. ACCEPT:  
  2.     clifd = accept(srvfd,(struct sockaddr*)&cliaddr,&cliaddrlen);  
  3.   
  4.     if (clifd == -1) {  
  5.         if (errno == EINTR) {  
  6.             goto ACCEPT;  
  7.         } else {  
  8.             fprintf(stderr, "accept fail,error:%s\n", strerror(errno));  
  9.             return -1;  
  10.         }  
  11.     }  



2 安装信号时设置 SA_RESTART属性(该方法对有的系统调用无效)

[html]  view plain  copy
  1. struct sigaction action;    
  2.      
  3. action.sa_handler = handler_func;    
  4. sigemptyset(&action.sa_mask);    
  5. action.sa_flags = 0;    
  6. /* 设置SA_RESTART属性 */    
  7. action.sa_flags |= SA_RESTART;    
  8.      
  9. sigaction(SIGALRM, &action, NULL);    

并不是所有的都有效



3  忽略信号(让系统不产生信号中断)

[html]  view plain  copy
  1. struct sigaction action;    
  2.      
  3. action.sa_handler = SIG_IGN;    
  4. sigemptyset(&action.sa_mask);    
  5.      
  6. sigaction(SIGALRM, &action, NULL);    

所以建议大家用第一种方法,重启。

猜你喜欢

转载自blog.csdn.net/junlon2006/article/details/80403737