《UNIX环境高级编程》笔记--中断的系统调用

原创 2013年12月04日 16:50:43

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

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

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

系统调用:

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

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

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

解调器应答了电话)

4.pause函数和wait函数

5.某些ioctl函数

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


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

again:
if((n=read(fd,buf,BUFFSIZE))<0){
	if(errno == EINTR)
		goto again;
}
为了帮助应用程序使其不必处理被中断的系统调用,4.2BSD引入了某些中断系统调用的自动重启动。自动重启动的系统调用包括:

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

在捕捉到信号时总是被终端。

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

重启被中断的系统调用。


实践:

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <stdbool.h>
#include <signal.h>
#include <sys/types.h>
#include <errno.h>
#include <string.h>

void int_handler (int signum)
{
          printf ("int handler %d\n",signum);
}

int main(int argc, char **argv)
{
          char buf[100];
          ssize_t ret;
          struct sigaction oldact;
          struct sigaction act;

          act.sa_handler = int_handler;
          act.sa_flags=0;
 //       act.sa_flags |= SA_RESTART;
          sigemptyset(&act.sa_mask);
          if (-1 == sigaction(SIGINT,&act,&oldact))
          {
                  printf("sigaction failed!\n");
                  return -1;
          }

          bzero(buf,100);

          ret = read(STDIN_FILENO,buf,10);
          if (ret == -1)
          {
                  printf ("read error %s\n", strerror(errno));

          }
          printf ("read %d bytes, content is %s\n",ret,buf);
          sleep (10);
          return 0;
}
运行结果:

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自动重启了。

慢系统调用与信号中断(转)

早期的Unix系统,如果进程在一个‘慢’系统调用中阻塞时,捕获到一个信号,这个系统调用被中断,调用返回错误,设置errno为EINTR。系统调用被分为慢系统调用和其他两大类别。     慢系统调用可...

什么是低速系统调用

可能使进程永远阻塞的系统调用就是低速系统调用。

UNIX中断系统调用和可重入函数

可中断的系统调用指的是进程阻塞在这个系统调用期间接受到一个信号,该系统调用会返回并返回出错信息errno=ENTER(注意:信号都是内核接收,并在系统从内核态返回到用户态的时候执行信号处理程序。信号中...

Unix环境高级编程(阅读笔记)----中断的系统调用,自动重启动的函数

早期的UNIX系统的一个特性是:如果进程在执行一个低速系统调用而阻塞期间捕捉到一个信号,该系统调用就被终端不再 继续执行。该系统调用返回出错,其errno被设置为EINTR。 为了支持这种特性,将系统...

UNIX环境高级编程——Linux系统调用列表

以下是Linux系统调用的一个列表,包含了大部分常用系统调用和由系统调用派生出的的函数。这可能是你在互联网上所能看到的唯一一篇中文注释的Linux系统调用列表,即使是简单的字母序英文列表,能做到这么完...

Unix环境编程学习笔记------查看系统调用号文件

系统调用号:   系统调用号写在: /arch/x86/include/asm/unistd_32.h 文件中 简要之,即:系统调用号写在  unistd_32.h  文件中。   下面是我...
  • H002399
  • H002399
  • 2015年09月29日 09:49
  • 257

unix环境多进程编程----用到的32个系统调用祥解

转载)unix环境多进程编程----用到的32个系统调用祥解一.多进程程序的特点 由于UNIX系统是分时多用户系统, CPU按时间片分配给各个用户使用, 而在实质上应该说CPU按时间片分配给各个进程使...

unix环境多进程编程----用到的32个系统调用详解

一.多进程程序的特点代码:  由于UNIX系统是分时多用户系统, CPU按时间片分配给各个用户使用, 而在实质上应该说CPU按时间片分配给各个进程使用, 每个进程都有自己的运行环境以使得在CPU做进程...

linux系统数据文件和信息--unix环境高级编程读书笔记

linux中对文件passwd,group,utmp,wtmp的操作,以及uname函数和时间操作的一些介绍,总的来说,是对linunx中的一些系统文件操作的相关介绍。...

《UNIX环境高级编程》六系统数据文件和信息读书笔记

UNIX系统的正常运作需要使用大量与系统有关的数据文件,例如,口令文件/etc/passwd和组文件/etc/group等。 1、口令文件 口令文件的各字段包含在pwd.h>中定义的passwd结...
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:《UNIX环境高级编程》笔记--中断的系统调用
举报原因:
原因补充:

(最多只允许输入30个字)