AskOverflow.Dev

AskOverflow.Dev Logo AskOverflow.Dev Logo

AskOverflow.Dev Navigation

  • 主页
  • 系统&网络
  • Ubuntu
  • Unix
  • DBA
  • Computer
  • Coding
  • LangChain

Mobile menu

Close
  • 主页
  • 系统&网络
    • 最新
    • 热门
    • 标签
  • Ubuntu
    • 最新
    • 热门
    • 标签
  • Unix
    • 最新
    • 标签
  • DBA
    • 最新
    • 标签
  • Computer
    • 最新
    • 标签
  • Coding
    • 最新
    • 标签
主页 / unix / 问题 / 509375
Accepted
Rick
Rick
Asked: 2019-03-29 22:57:55 +0800 CST2019-03-29 22:57:55 +0800 CST 2019-03-29 22:57:55 +0800 CST

什么是中断的系统调用?

  • 772

我正在阅读APUE,而中断的系统调用一章让我感到困惑。

我想根据本书写下我的理解,请指正。

  1. 早期 UNIX 系统的一个特点是,如果进程在“慢”系统调用中被阻塞时捕获到信号,则系统调用会被中断。系统调用返回错误并 errno设置为EINTR。这是在假设发生信号并且进程捕获它的情况下完成的,很有可能发生了一些应该唤醒阻塞的系统调用的事情。

    所以说早期的UNIX 系统有一个特性:如果我的程序使用系统调用,它会被中断/停止,如果程序在任何时候捕捉到一个信号。(默认处理程序也算作捕获吗?)

    例如,如果我有一个read读取 10GB 数据的系统调用,当它读取时,我发送任何一个信号(例如kill -SIGUSR1 pid),然后read会失败并返回。


  1. 为了防止应用程序不得不处理中断的系统调用,4.2BSD 引入了自动重启某些中断的系统调用。自动重新启动的系统调用是ioctl、read、readv、write、writev、wait和waitpid。正如我们所提到的,前五个函数只有在运行速度较慢的设备时才会被信号中断。wait并且waitpid 总是在捕捉到信号时被中断。由于这会导致一些应用程序在操作中断时不希望重新启动操作,因此 4.3BSD 允许进程在每个信号的基础上禁用此功能。

所以在引入自动重启之前,我必须自己处理中断的系统调用。我需要编写如下代码:

中断系统调用的问题是我们现在必须明确地处理错误返回。典型的代码序列(假设一个读取操作并假设我们想要重新启动读取,即使它被中断)将是:

again:
    if ((n = read(fd, buf, BUFFSIZE)) < 0) {
        if (errno == EINTR)
        goto again; /* just an interrupted system call */
        /* handle other errors */
}

但是现在我不用写这种代码了,因为有自动重启机制。


因此,如果我的理解都是正确的,那么我现在应该关心什么/为什么要关心中断的系统调用..?似乎系统/操作系统会自动处理它。

signals system-calls
  • 2 2 个回答
  • 38423 Views

2 个回答

  • Voted
  1. Best Answer
    mtk
    2019-03-31T12:58:19+08:002019-03-31T12:58:19+08:00

    信号处理程序对系统调用的中断仅发生在各种阻塞系统调用的情况下,并且当系统调用被程序员明确建立的信号处理程序中断时发生。

    此外,在阻塞系统调用被信号处理程序中断的情况下,自动系统调用重新启动是一个可选功能。SA_RESTART您可以通过在建立信号处理程序时指定标志来选择自动重新启动系统调用。如(例如)Linux signal(7)手册页中所述:

       If  a  signal  handler  is  invoked while a system call or library
       function call is blocked, then either:
    
       * the call is automatically restarted  after  the  signal  handler
         returns; or
    
       * the call fails with the error EINTR.
    
       Which  of  these two behaviors occurs depends on the interface and
       whether or not  the  signal  handler  was  established  using  the
       SA_RESTART  flag (see sigaction(2)). 
    

    正如上面引用的最后一句话所暗示的,即使您选择使用此功能,它也不适用于所有系统调用,并且它适用的系统调用集因 UNIX 实现而异。Linuxsignal(7)手册页记录了许多在使用该SA_RESTART标志时会自动重新启动的系统调用,但还继续记录了从未重新启动的各种系统调用,即使您在建立处理程序时指定了该标志,包括:

       * "Input" socket interfaces, when a timeout (SO_RCVTIMEO) has been
         set  on  the  socket  using  setsockopt(2):  accept(2), recv(2),
         recvfrom(2), recvmmsg(2) (also with  a  non-NULL  timeout  argu‐
         ment), and recvmsg(2).
    
       * "Output"  socket  interfaces,  when  a timeout (SO_RCVTIMEO) has
         been set on the socket using setsockopt(2): connect(2), send(2),
         sendto(2), and sendmsg(2).
    
       * File   descriptor   multiplexing   interfaces:    epoll_wait(2),
         epoll_pwait(2), poll(2), ppoll(2), select(2), and pselect(2).
    
       * System  V  IPC  interfaces:  msgrcv(2), msgsnd(2), semop(2), and
         semtimedop(2).
    

    对于这些系统调用,使用 APUE 中描述的形式的循环手动重新启动是必不可少的,例如:

    while ((ret = some_syscall(...)) == -1 && errno == EINTR)
        continue;
    if (ret == -1)
        /* Handle error */ ;
    
    • 9
  2. Uncle Billy
    2019-03-30T00:29:25+08:002019-03-30T00:29:25+08:00

    [我没有读过 APUE 的东西,但你引用的东西看起来不太好]

    如果我的程序使用系统调用,如果程序在任何时候捕获到信号,它将被中断/停止。

    没有任何系统调用。只有一些系统调用是可中断的。

    (默认处理程序也算作捕获吗?)

    不。

    例如,如果我有一个读取 10GB 数据的读取系统调用,当它读取时,我发送任何一个信号(例如kill -SIGUSR1 pid),然后读取将失败并返回。

    您的 10GB read() 仅在无法读取单个字节之前被中断时才会返回 EINTR ;否则它将返回它已经读取的数据量(短读=成功,errno 不相关)。

    [这在链接的欺骗中没有解释]

    因此,如果我的理解都是正确的,那么我现在应该关心什么/为什么要关心中断的系统调用..?似乎系统/操作系统会自动处理它。

    因为你可能想在收到信号后做一些事情,而你不能从信号处理程序中做很多事情;任何使用 malloc() 或 stdio(甚至 printf())的东西都是不可能的。因此,您必须在程序的主循环中处理中断,并且为了能够做到这一点,您应该以某种方式从阻塞的 read() 中中断(即使在 poll() 返回 fd 之后,read() 也可以阻塞准备阅读)。

    [这在链接的欺骗中也有解释]

    • 3

相关问题

  • 为什么已经存在 SIGSTOP 时存在 SIGTSTP?

  • 调试的信号编号是 1 到 64 中的哪个数字?

  • 在 C++ 代码中使用 system() 函数与使用源代码一样快速吗?[关闭]

  • 内核模式位

  • 需要一些系统调用

Sidebar

Stats

  • 问题 205573
  • 回答 270741
  • 最佳答案 135370
  • 用户 68524
  • 热门
  • 回答
  • Marko Smith

    模块 i915 可能缺少固件 /lib/firmware/i915/*

    • 3 个回答
  • Marko Smith

    无法获取 jessie backports 存储库

    • 4 个回答
  • Marko Smith

    如何将 GPG 私钥和公钥导出到文件

    • 4 个回答
  • Marko Smith

    我们如何运行存储在变量中的命令?

    • 5 个回答
  • Marko Smith

    如何配置 systemd-resolved 和 systemd-networkd 以使用本地 DNS 服务器来解析本地域和远程 DNS 服务器来解析远程域?

    • 3 个回答
  • Marko Smith

    dist-upgrade 后 Kali Linux 中的 apt-get update 错误 [重复]

    • 2 个回答
  • Marko Smith

    如何从 systemctl 服务日志中查看最新的 x 行

    • 5 个回答
  • Marko Smith

    Nano - 跳转到文件末尾

    • 8 个回答
  • Marko Smith

    grub 错误:你需要先加载内核

    • 4 个回答
  • Marko Smith

    如何下载软件包而不是使用 apt-get 命令安装它?

    • 7 个回答
  • Martin Hope
    user12345 无法获取 jessie backports 存储库 2019-03-27 04:39:28 +0800 CST
  • Martin Hope
    Carl 为什么大多数 systemd 示例都包含 WantedBy=multi-user.target? 2019-03-15 11:49:25 +0800 CST
  • Martin Hope
    rocky 如何将 GPG 私钥和公钥导出到文件 2018-11-16 05:36:15 +0800 CST
  • Martin Hope
    Evan Carroll systemctl 状态显示:“状态:降级” 2018-06-03 18:48:17 +0800 CST
  • Martin Hope
    Tim 我们如何运行存储在变量中的命令? 2018-05-21 04:46:29 +0800 CST
  • Martin Hope
    Ankur S 为什么 /dev/null 是一个文件?为什么它的功能不作为一个简单的程序来实现? 2018-04-17 07:28:04 +0800 CST
  • Martin Hope
    user3191334 如何从 systemctl 服务日志中查看最新的 x 行 2018-02-07 00:14:16 +0800 CST
  • Martin Hope
    Marko Pacak Nano - 跳转到文件末尾 2018-02-01 01:53:03 +0800 CST
  • Martin Hope
    Kidburla 为什么真假这么大? 2018-01-26 12:14:47 +0800 CST
  • Martin Hope
    Christos Baziotis 在一个巨大的(70GB)、一行、文本文件中替换字符串 2017-12-30 06:58:33 +0800 CST

热门标签

linux bash debian shell-script text-processing ubuntu centos shell awk ssh

Explore

  • 主页
  • 问题
    • 最新
    • 热门
  • 标签
  • 帮助

Footer

AskOverflow.Dev

关于我们

  • 关于我们
  • 联系我们

Legal Stuff

  • Privacy Policy

Language

  • Pt
  • Server
  • Unix

© 2023 AskOverflow.DEV All Rights Reserve