嵌入式Linux系统编程 — 6.1 信号的基本概念

news2024/11/24 5:53:10

目录

1 信号的概念和作用

1.1 什么是信号

1.2 信号的目的

1.3 信号如何处理

2 信号的分类

2.1 可靠信号与不可靠信号

2.2 实时信号与非实时信号

3 常见信号与默认行为

3.1 信号本质上是 int 类型数字编号

3.2 常见信号


1 信号的概念和作用

1.1 什么是信号

信号是一种软件中断机制

在Linux系统中,信号是一种软件中断机制,用于通知进程发生了某些事件。信号可以由操作系统内核、用户或另一个进程发送。信号与硬件中断的相似之处在于能够打断程序当前执行的正常流程, 其实是在软件层次上对中断机制的一种模拟。

信号是异步发生的

信号在Linux系统中是异步发生的,这意味着信号的发送和接收是独立于进程的执行流的。当信号发送给进程时,内核会将信号添加到进程的信号队列中,而进程可以在任何时候通过执行信号处理函数来响应这些信号。

由于信号的异步特性,它们可以即时通知进程发生了某些事件,而无需进程进行轮询或等待。这使得信号成为一种快速且有效的进程间通信机制,尤其适用于需要快速响应的情况。

1.2 信号的目的

Linux系统信号的主要目的是提供一种快速、异步的进程间通信机制,用于通知进程发生了某些特定的事件或条件。信号有以下几个主要来源:

  • 硬件异常:硬件发生异常,即硬件检测到错误条件并通知内核,随即再由内核发送相应的信号给相关进程。如浮点错误(如除以零),会由硬件触发并由内核转换为信号(如SIGFPE)发送给进程。

  • 用户生成:用于在终端下输入了能够产生信号的特殊字符。用户可以通过键盘组合键(如Ctrl+C产生SIGINT)来生成断信号,通过这个方法可以终止在前台运行的进程;按下 CTRL + Z 组合按键可以产生暂停信号(SIGCONT),可以暂停当前前台运行的进程。

  • 进程间通信:进程调用 kill()系统调用可将任意信号发送给另一个进程或进程组。 当然对此是有所限制的,接收信号的进程和发送信号的进程的所有者必须相同,亦或者发送信号的进程的所有者是 root 超级用户。

  • 软件条件:软件条件,如进程使用abort函数,会生成SIGABRT信号,进程所设置的定时器已经超时、进程执行的 CPU 时间超限、进程的某个子进程退出等等情况。

  • 系统调用:某些系统调用在特定情况下会触发信号。例如,readwrite操作在对应文件描述符变为非阻塞状态且没有数据可读或写时,可能会触发EAGAIN错误,这可以被看作是一种信号。

  • 内核生成:操作系统内核在检测到某些条件时会生成信号。例如,当进程试图执行非法操作或访问无效内存时,内核会发送SIGILL或SIGSEGV。

  • 自定义信号:应用程序可以定义自己的信号处理逻辑,以响应自定义的信号,如SIGUSR1和SIGUSR2。

1.3 信号如何处理

信号通常是发送给对应的进程,当信号到达后, 该进程需要做出相应的处理措施,可以通过以下几种方式来处理信号:

  • 忽略信号:进程可以选择忽略某些信号,使其不产生任何效果。例如,使用signalsigaction函数将信号处理函数设置为SIG_IGN。事实上,大多数信号都可以使用这种方式进行处理,但有两种信号却决不能被忽略,它们是 SIGKILL 和 SIGSTOP,它们是Linux内核保留的信号,用于立即终止和暂停进程,这两个信号的设计是为了在紧急情况下强制终止或暂停进程,确保系统能够迅速响应严重错误或管理员的干预。

  • 捕获信号:进程可以定义信号处理函数(也称为信号捕获函数或信号处理程序),当信号被发送到进程时,该函数将被调用。

  • 默认操作:如果进程没有特别指定如何处理某个信号,那么信号将执行其默认操作。例如,SIGKILL和SIGSTOP信号的默认操作是终止进程,而SIGCHLD信号的默认操作是忽略。

  • 阻塞信号:进程可以暂时阻止某些信号的传递,直到进程再次允许这些信号。这可以通过sigprocmask函数实现。

2 信号的分类

2.1 可靠信号与不可靠信号

在Linux系统中,信号可以分为可靠信号和不可靠信号,这种分类主要是基于信号的传递和处理机制:

  • 不可靠信号(Unreliable Signals):这类信号可能在发送给进程时丢失,尤其是在高负载或大量信号同时发送的情况下。它们通常用于通知进程发生了某些事件,但不保证进程一定会接收到这些信号。

  • 可靠信号(Reliable Signals)可靠信号确保一旦发送,就会传递给进程,并且会被进程接收。这些信号通常用于控制进程的生命周期,如终止或暂停进程。

在 Linux 系统下使用"kill -l"命令可查看到所有信号,如下所示:

括号" ) "前面的数字对应该信号的编号,编号 1~31 所对应的是不可靠信号,编号 34~64 对应的是
可靠信号,从图中可知,可靠信号并没有一个具体对应的名字,而是使用了 SIGRTMIN+N 或 SIGRTMAXN 的方式来表示。

2.2 实时信号与非实时信号

实时信号与非实时信号其实是从时间关系上进行的分类,与可靠信号与不可靠信号是相互对应的, 非实时信号都不支持排队,都是不可靠信号;实时信号都支持排队,都是可靠信号。

  • 非实时信号:也称为标准信号或传统信号,它们是Linux系统中默认的信号类型。这些信号的编号从1到31,包括常见的信号如SIGKILLSIGTERMSIGINT等。非实时信号的优先级较低,它们可以被进程忽略或捕获,并由进程定义的处理函数来处理。

  • 实时信号:实时信号是POSIX.1b标准的一部分,提供了比非实时信号更高的优先级。这些信号的编号从32开始,由SIGRTMINSIGRTMAX定义,具体数量依赖于系统实现。实时信号不能被进程忽略,必须通过信号处理函数来处理,或者使用sigaction设置为忽略。它们通常用于需要快速响应的场景,如多媒体应用或实时控制系统。

常见信号与默认行为

3.1 信号本质上是 int 类型数字编号

信号在Linux系统中确实是以整型数字编号来标识的,每个信号都有一个唯一的数字,通常从1开始递增。例如,SIGINT信号编号为2,SIGTERM信号编号为15。这些整型编号是信号的标识符,用于在程序中引用和操作信号。

然而,除了整型编号,信号还具有对应的宏定义,这些宏定义通常以SIG为前缀,后面跟着信号名称的缩写。例如,SIGINT代表中断信号,SIGTERM代表终止信号。使用宏定义而不是直接使用整型编号可以使代码更易读和维护。

#define SIGHUP 1 /* Hangup (POSIX). */
#define SIGINT 2 /* Interrupt (ANSI). */
#define SIGQUIT 3 /* Quit (POSIX). */
#define SIGILL 4 /* Illegal instruction (ANSI). */
#define SIGTRAP 5 /* Trace trap (POSIX). */
#define SIGABRT 6 /* Abort (ANSI). */
#define SIGIOT 6 /* IOT trap (4.2 BSD). */
#define SIGBUS 7 /* BUS error (4.2 BSD). */
#define SIGFPE 8 /* Floating-point exception (ANSI). */
#define SIGKILL 9 /* Kill, unblockable (POSIX). */
#define SIGUSR1 10 /* User-defined signal 1 (POSIX). */
#define SIGSEGV 11 /* Segmentation violation (ANSI). */
#define SIGUSR2 12 /* User-defined signal 2 (POSIX). */
#define SIGPIPE 13 /* Broken pipe (POSIX). */
#define SIGALRM 14 /* Alarm clock (POSIX). */
#define SIGTERM 15 /* Termination (ANSI). */
#define SIGSTKFLT 16 /* Stack fault. */
#define SIGCLD SIGCHLD /* Same as SIGCHLD (System V). */
#define SIGCHLD 17 /* Child status has changed (POSIX). */
#define SIGCONT 18 /* Continue (POSIX). */
#define SIGSTOP 19 /* Stop, unblockable (POSIX). */
#define SIGTSTP 20 /* Keyboard stop (POSIX). */
#define SIGTTIN 21 /* Background read from tty (POSIX). */
#define SIGTTOU 22 /* Background write to tty (POSIX). */
#define SIGURG 23 /* Urgent condition on socket (4.2 BSD). */
#define SIGXCPU 24 /* CPU limit exceeded (4.2 BSD). */
#define SIGXFSZ 25 /* File size limit exceeded (4.2 BSD). */
#define SIGVTALRM 26 /* Virtual alarm clock (4.2 BSD). */
#define SIGPROF 27 /* Profiling alarm clock (4.2 BSD). */
#define SIGWINCH 28 /* Window size change (4.3 BSD, Sun). */
#define SIGPOLL SIGIO /* Pollable event occurred (System V). */
#define SIGIO 29 /* I/O now possible (4.2 BSD). */
#define SIGPWR 30 /* Power failure restart (System V). */
#define SIGSYS 31 /* Bad system call. */
#define SIGUNUSED 31

3.2 常见信号

前面说到, Linux 下对标准信号(不可靠信号、 非实时信号) 的编号为 1~31,如示例代码 8.1.1 所示,接下来将介绍这些信号以及这些信号所对应的系统默认操作。

编号信号名称描述系统默认操作
2SIGINT终端中断符,当用户在终端按下中断字符(通常是 CTRL + C)时,内核将发送 SIGINT 信号给前台进程组中的每一个进程。term
3SIGQUIT终端退出符,当用户在终端按下退出字符(通常是 CTRL + \)时,内核将发送 SIGQUIT 信号给前台进程组中的每一
个进程。
term+core
4SIGILL非法硬件指令,如果进程试图执行非法(即格式不正确)的机器语言指令,系统将向进程发送该信号。term+core
6SIGABRT异常终止(abort),如果进程试图执行非法(即格式不正确)的机器语言指令,系统将向进程发送该信号。term+core
7SIGBUS内存访问错误,产生该信号(总线错误, bus error)表示发生了某种内存访问错误。term+core
8SIGFPE算术异常,该信号因特定类型的算术错误而产生,譬如除以 0。term+core
9SIGKILL终极终止信号,SIGKILL此信号为“必杀(sure kill)”信号,用于杀死进程的终极办法,此信号无法被进程阻塞、忽略或者捕获,故而“一击必杀”,总能终止进程。term
10SIGUSR1用户自定义信号 1,该信号和 SIGUSR2 信号供程序员自定义使用,内核绝不会为进程产生这些信号,在我们的程序中,可以使用这些信号来互通通知事件的发生,或是进程彼此同步操作。term
11SIGSEGV无效的内存引用,这一信号非常常见,当应用程序对内存的引用无效时,操作系统就会向该应用程序发送该信号。term+core
12SIGUSR2用户自定义信号 2,与 SIGUSR1 信号相同。term
13SIGPIPE管道关闭,涉及到管道和 socket,当进程向已经关闭的管道、 FIFO 或套接字写入信息时,那么系统将发送该信号
给进程。
term
14SIGALRM定时器超时(alarm),应用程序中可以调用 alarm()或 setitimer()函数来设置一个定时器,当定时器定时时间到,那么内核将会发送 SIGALRM 信号给该应用程序term
15SIGTERM终止进程,SIGTERM通常用于请求进程正常终止。它是一种较为温和的终止信号,允许进程在终止前进行清理工作,比如保存状态、关闭文件描述符或释放资源。与SIGKILL(信号编号9)不同,用于立即强制终止进程。term
17SIGCHLD/SIGCLD子进程终止或停止,当父进程的某一个子进程终止时,内核会向父进程发送该信号。ignore
18SIGCONT使停止状态的进程继续运行,将该信号发送给已停止的进程,进程将会恢复运行。cont
19SIGSTOP停止进程,这是一个“必停”信号,用于停止进程(注意停止不是终止,停止只是暂停运行、进程并没有终止)。stop
20SIGTSTP终端停止符,也是一个停止信号,当用户在终端按下停止字符(通常是 CTRL + Z),那么系统会将 SIGTSTP 信号
发送给前台进程组中的每一个进程,使其停止运行。
stop
24SIGXCPU超过 CPU 限制,当进程的 CPU 时间超出对应的资源限制时,内核将发送此信号给该进程。term+core
26SIGVTALRM虚拟定时器超时,应用程序调用 setitimer()函数设置一个虚拟定时器,当定时器定时时间到时,内核将会发送该信号给进程。term
28SIGWINCH终端窗口尺寸发生变化,在窗口环境中,当终端窗口尺寸发生变化时(譬如用户手动调整了大小,应用程序调用 ioctl()设置了大小等),系统会向前台进程组中的每一个进程发送该信号。ignore
29SIGPOLL/SIGIO异步 I/O,用于提示一个异步 IO 事件的发生,譬如应用程序打开的文件描述符发生了 I/O 事件时,内核会向应用程序发送 SIGIO 信号。term/ignore
31SIGSYS无效系统调用,如果进程发起的系统调用有误,那么内核将发送该信号给对应的进程。term+core

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1885398.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

为什么要学习大模型应用开发?原因80%的人都不知道

0 prompt engineer 就是prompt工程师它的底层透视。 1 学习大模型的重要性 底层逻辑 人工智能大潮已来,不加入就可能被淘汰。就好像现在职场里谁不会用PPT和excel一样,基本上你见不到。你问任何一个人问他会不会用PPT,他都会说会用&#x…

Python安装,几个步骤轻松实现(超详细)

目录 步骤一、进入官网 步骤二、将鼠标放在Downloads上,选择All releases 步骤三、 下滑找到Looking for a specific release,选择python版本点击Download安装 步骤四、下滑找到Files,以windows64为例,点击下载 步骤五、安装Py…

RTL8305NB从电口模式切换为光口模式

#if 1//下面是参考案例 //RTL8305NB#define PORT2_PHY_ADDR 0x05 // SFP Port2 PHY地址 #define STATUS_REG_ADDR 0x01 // 状态寄存器地址#define MDC_PIN GPIO_Pin_13 //MDC (PC13) #define MDIO_PIN GPIO_Pin_6 //MDIO (PE6)#define MDIO_DELAY 10 // us #defin…

秋招Java后端开发冲刺——基础篇5(String集合)

一、String String类是Java中字符串操作类,位于java.lang包下String类型对象的底层使用字符数组char[]存储字符串,由final修饰且没有提供公共的修改方法,因此String对象是不可变的。常见方法 方法名作用trim()去掉字符串首尾空字符split(分…

[AI Perplexica] AI驱动的开源搜索引擎

之前,我们有介绍过 Perplexcity 现在,开源市场上,也有一款对标产品 Perplexica ,我们来看下 界面很像 介绍 Perplexica是一个开源的、由AI驱动的搜索工具或搜索引擎,它深入互联网寻找答案。受到Perplexity AI的启发…

警惕!帕金森老人身体恶化七大征兆,家有老人必看!

帕金森病,这个在老年人群中越来越常见的神经系统疾病,以其独特的“静止性震颤、运动迟缓、肌强直和姿势平衡障碍”等四大症状,成为许多家庭心中的痛。然而,帕金森病的进展并非一蹴而就,而是有着一系列的身体变坏征兆。…

vue3源码(六)渲染原理-runtime-dom

1、从入口文件看实现 项目入口文件 import { createApp } from vue import ./style.css import App from ./App.vuecreateApp(App).mount(#app)文件位置core\packages\runtime-dom\src\index.ts 保证了render的唯一性 // // rendererOptions 是patchProp 和nodeOps的合集&a…

材料科学SCI期刊,IF=6+,超高录用率,2个月录用

一、期刊名称 Advanced Electronic Materials 二、期刊简介概况 期刊类型:SCI 学科领域:材料科学 影响因子:5.3 中科院分区:2区 三、期刊简介 Advanced Electronic Materials 是一个跨学科论坛,旨在为材料科学、…

YOLOv8-对注意力机制模型进行通道剪枝-同时实现涨点和轻量化【附代码】

文章目录 前言视频效果文章概述必要环境一、训练自己的模型1、 训练命令2、 训练参数解析 二、模型剪枝1、 对训练好的模型将进行剪枝2、 剪枝代码详解1.解析命令行参数2. 定义剪枝函数3. 定义剪枝结构4. 更新注意力机制5. 保存更新后的模型6. 主函数 三、剪枝后的训练运行命令…

Windows 11 安装 安卓子系统 (WSA)

How to Install Windows Subsystem for Android (WSA) on Windows 11 新手教程:如何安装Windows 11 安卓子系统 说明 Windows Subsystem for Android 或 WSA 是由 Hyper-V 提供支持的虚拟机,可在 Windows 11 操作系统上运行 Android 应用程序。虽然它需…

c++读取文件时出现中文乱码

原因:UTF-8格式不支持汉字编码 解决:改成ANSI,因为ANSI编码支持汉字编码

生成式人工智能将如何改变网络可访问性

作者:Matthew Adams 受 Be My Eyes 和 OpenAI 启发的一项实验,尝试使用 ChatGPT 4o 实现网页无障碍 在 Elastic,我们肩负着一项使命,不仅要构建最佳的搜索驱动型 AI 平台,还要确保尽可能多的人喜欢使用该平台。我们相…

深入剖析vLLM:大模型计算加速系列之调度器策略探索

原文: 图解大模型计算加速系列:vLLM源码解析2,调度器策略(Scheduler) 目录 收起 前期提要与本期导览 一、入口函数 二、SequenceGroup 2.1 原生输入 2.2 SequenceGroup的作用 2.3 SequenceGroup的结构 三、add_request()&#xff1a…

[python][Anaconda]使用jupyter打开F盘或其他盘文件

jupyter有一个非常不好的体验,就是不能在界面切换到其他盘来打开文件。 使用它,比较死板的操作是要先进入文件目录,再运行jupyter。 以Windows的Anaconda安装了jupyter lab或jupyter notebook为例。 1,先运行Anaconda Prompt 2&…

儿童房间灯哪个牌子的好?几款儿童房间灯具品牌分享

对于视力正处于发育阶段的儿童而言,台灯已不仅仅是一个简单的照明工具。它不仅驱散夜幕下的阴霾,还能为儿童的眼部保驾护航。一款优质的护眼台灯更是不可或缺的守护者。然而,面对市场上琳琅满目的选择,怎样选出一款合适的护眼台灯…

​Stable Diffusion史上最全插件,已打包整理,12个常用插件你肯定用得上!

还在于有丰富的第三方插件,即我们在安装部署之后安装汉化插件的界面 插件安装方式可以是“可下载->加载扩展列表”,然后从列表选择或搜索插件下载,或直接选择“从网站安装”,填写插件的git仓库地址。一般我们从扩展列表搜索即可…

【Python】已解决:pymssql._pymssql.OperationalError 关于关键字‘distinct’的语法错误

文章目录 一、分析问题背景二、可能出错的原因三、错误代码示例四、正确代码示例五、注意事项 已解决:pymssql._pymssql.OperationalError 关于关键字‘distinct’的语法错误 一、分析问题背景 在使用pymssql库与SQL Server数据库进行交互时,有时会遇到…

WPF在.NET9中的重大更新:Windows 11 主题

在2023年的2月20日,在WPF的讨论区,WPF团队对路线的优先级发起了一次讨论。 对三个事项发起了投票。 第一个是Windows 11 主题 第二个是更新的控件 第三个是可空性注释 最终Windows 11 主题得票最高,WPF团队2023-2024的工作优先级就是Windows…

UE4_材质_水体的反射与折射制作_Ben教程

在这个教程中,将制作水的反射和折射,上个教程,我们主要讲了制作水涟漪(水面波纹)和水滴法线混合,水深计算,我们首先要谈的是反射和产生折射的问题。我们将所有从干扰从场景中分离出去&#xff0…

微信小程序 canvas 处理图片的缩放移动旋转问题

这里使用到了一个插件&#xff0c;canvas-drag&#xff0c;来实现大部分功能的 上效果 直接上代码吧~ wxml <div class"container"><canvas-drag id"canvas-drag" graph"{{graph}}" width"700" height"750" ena…