如何理解SA_RESTART”被信号中断的系统调用自动重启“?

"被信号中断的系统调用自动重启"是 Linux 信号处理中的一个重要机制,由 SA_RESTART 标志控制。以下是深入解释:

核心概念

当进程执行阻塞型系统调用 (如 read(), write(), accept(), wait() 等)时:

  1. 如果系统调用执行期间收到信号
  2. 该信号设置了信号处理函数(非忽略/默认终止)
  3. 系统调用会被中断,立即返回用户空间执行信号处理函数

此时系统调用有两种后续处理方式:

  • 不自动重启 :系统调用返回 -1,并设置 errno = EINTR(被中断)
  • 自动重启 (设置 SA_RESTART):内核自动重新执行被中断的系统调用

工作流程对比

有信号 已设置 未设置 阻塞系统调用开始 信号到达 中断系统调用 执行信号处理函数 SA_RESTART 标志 内核自动重启系统调用 返回用户空间 errno=EINTR 系统调用完成 应用程序处理 EINTR

实际代码示例

未设置 SA_RESTART 的情况

c 复制代码
#include <stdio.h>
#include <unistd.h>
#include <signal.h>

void handler(int sig) {
    printf("Received signal %d\n", sig);
}

int main() {
    struct sigaction sa;
    sa.sa_handler = handler;
    sigemptyset(&sa.sa_mask);
    sa.sa_flags = 0; // 不设置 SA_RESTART
    
    sigaction(SIGINT, &sa, NULL);
    
    char buf[256];
    ssize_t n = read(STDIN_FILENO, buf, sizeof(buf)); // 阻塞读
    
    if (n == -1 && errno == EINTR) {
        printf("Read interrupted!\n");
    }
    return 0;
}

当按下 Ctrl+C(发送 SIGINT)时:

  1. read() 被中断
  2. 执行信号处理函数
  3. read() 返回 -1errno = EINTR

设置 SA_RESTART 的情况

c 复制代码
sa.sa_flags = SA_RESTART; // 关键设置

同样按下 Ctrl+C 后:

  1. read() 被中断
  2. 执行信号处理函数
  3. 内核自动重启 read() 调用
  4. 程序继续阻塞等待输入

技术细节

  1. 哪些系统调用支持重启

    • 文件I/O:read, write, open
    • 进程控制:wait, waitpid
    • 网络操作:accept, connect, recv
    • 不支持的:poll, select, epoll_wait(需手动处理 EINTR)
  2. 重启的实现位置

    在 Linux 内核源码 kernel/signal.cget_signal() 函数中:

    c 复制代码
    if (sig_handler_ignored(...) || 
        (sa->sa_flags & SA_RESTART)) {
        restart_syscall(); // 重启逻辑
    }
  3. 编程最佳实践

    • 对需要可靠性的服务程序,建议设置 SA_RESTART

    • 对需要快速响应信号的程序(如监控),避免使用 SA_RESTART

    • 即使使用 SA_RESTART,也应处理关键操作的 EINTR:

      c 复制代码
      do {
          ret = accept(sockfd, ...);
      } while (ret == -1 && errno == EINTR); // 安全重启

典型应用场景

  1. 网络服务器

    c 复制代码
    sa.sa_flags = SA_RESTART; // 确保accept不会因信号失败
    sigaction(SIGCHLD, &sa, NULL); // 处理子进程退出

    当子进程退出发送 SIGCHLD 时,accept() 自动重启

  2. 交互式程序

    c 复制代码
    sa.sa_flags = 0; // 不设置SA_RESTART
    sigaction(SIGALRM, &sa, NULL); // 超时信号

    read()SIGALRM 中断时,可执行超时处理逻辑

这种机制平衡了系统调用的可靠性和信号的响应能力,是 Linux 信号处理体系的核心设计之一。

相关推荐
j_xxx404_25 分钟前
Linux:自主shell命令行解释器附源码
linux·运维·服务器
开开心心就好1 小时前
开源免费高速看图工具,支持漫画大图秒开
linux·运维·服务器·安全·ruby·symfony·1024程序员节
花间相见1 小时前
【AI开发】—— Ubuntu系统使用nvm管理Node.js多版本,版本切换一键搞定(实操完整版)
linux·ubuntu·node.js
PPPPPaPeR.1 小时前
从零实现一个简易 Shell:理解 Linux 进程与命令执行
linux·开发语言·c++
Byte不洛1 小时前
Linux 多线程:生产者消费者模型、阻塞队列与条件变量详解
linux·多线程·并发编程·pthread·生产者消费者模型
小Pawn爷1 小时前
13.virtualbox安装ubuntu
linux·运维·ubuntu
VekiSon1 小时前
Linux内核驱动——Ubuntu 网络启动环境配置与操作
linux·arm开发·嵌入式硬件·ubuntu
YYYing.2 小时前
【Linux/C++进阶篇(二) 】超详解自动化构建 —— 日常开发中的“脚本” :Makefile/CMake
linux·c++·经验分享·ubuntu
wdfk_prog2 小时前
[Linux]学习笔记系列 -- [drivers][gpio[[gpiolib]
linux·笔记·学习
砚上有墨2 小时前
问题记录:云平台计算节点内存故障,热迁移失败导致系统重启。
linux·运维·云计算