Linux 学习之路 - 信号的保存

2024-09-04 07:12
文章标签 linux 学习 保存 信号

本文主要是介绍Linux 学习之路 - 信号的保存,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

前面已经介绍过信号的产生,本文将继续介绍信号的保存与处理。

1、上篇文章的遗留问题

从上篇文章(Linux学习之路 -- 信号概念 && 信号的产生-CSDN博客)中,其实还遗留了一些问题。OS在接受到信号后,大部分的进程的处理方式都是终止进程。但是终止进程的方式有Term和Code两种方式。那么两种进程退出的方式有何区别呢?

Term就表示正常的退出,而code退出会产生核心转储文件。在云服务器中,使用code退出时,默认是将code产生核心转储文件的功能关闭的。而我们如果想要查看该功能是否被打开。我们就可以使用"ulimit -a"选项,对该功能进行查看。

其中的第一项"core file size" 就表示code功能是否被打开。core file size其实也就是表示核心转储文件的大小,如果为零,也就表示没有。我们要恢复该功能,只要将其设置一定大小的正整数即可(单位为KB)。我们可以使用"ulimit -c + 指定大小的整数",如果不想该文件的大小收到限制,就把前面这条命令中的整数改为unlimited即可。

核心转储文件的作用:如果进程运行出现异常时,系统就会将进程在内存中的数据(主要与调试有关)转到磁盘中,形成core、core.pid文件。

该文件主要用于帮助我们进行调试,当我们使用gdb时,能够直接定位到对应出错的行号与文件。我们在打开gdb时,直接输入core-file + core(这里是核心转储的文件名,如果有后缀pid,记得加上)即可。

core dumped 表示形成核心转储文件,这个和前面我们讲述的进程退出时的知识有些许关联。


进程退出时,会返回一个整型。OS以后16个字节对进程的退出状态进行标识。中间其实缺少了一个比特没有介绍,这个比特位就是core dumped标志,用于标识是否形成核心转储文件。

但是核心转储文件虽然能帮我们定位程序出错的位置,但在实际的云服务器上,我们的服务是需要不断运行的。有时候恢复正常服务,需要很长的时间。在此期间,有可能OS会一直输出core文件,导致服务器磁盘被打满,这就会导致服务器也直接挂掉。所以,正常情况下,这个core dump功能是会被默认关闭的。

2、信号的保存

在正式介绍信号保存之前,我们需要把前面的概念修正一下。
<1>实际执行信号的处理动作称为信号递达。(默认、忽略、自定义)
<2>信号从产生到递达之间的状态称为信号未决。(存储信号)
<3>进程是可以阻塞信号的。

1、基本流程

前面我们提到,当OS向进程发送信号,实际上是在向进程写入信号。而进程可能在处理其他事务,所以会使用一个位图来对信号进行保存。在struct task_struct 中就是以一个无符号整型变量对信号进行保存。同理,信号可以被进程阻塞,这个阻塞也是需要先存储信号,而OS也是使用了位图的方式,对其进行保存。当进程接受到信号时,会先保存信号。然后,在需要执行信号的默认处理方式前,判断信号是否被阻塞,如果是,就不处理该信号;不是就执行默认方法。(如果该信号被一直阻塞,那就一直不执行)

除了上述的两张位图,还有一个数组与进程处理信号相关。

这个数组是一个函数指针数组,里面存储了一些对应信号的默认处理方法。当OS接收到信号时,先将信号写进pending位图中,然后判断是否被阻塞。如果没有被阻塞,就找到对应的handler方法,处理信号,该数组的下标就代表了对应的信号。这里可以联想到signal接口处理信号的方式,当我们设定位自定义处理信号时,将该信号在handler_t中对应的默认处理方法替换成自定义函数地址。

2、三张表对应的系统接口和相关操作

1、五个常用的信号集函数

在OS中,我们查看pending位图和block位图,是不会直接传整型的,而是将存储的结构封装成了一个结构体,所以我们在上述的五个接口中均可以看见sigset_t类型的参数。下面一一解释一下对应接口的作用。

<1>sigemptyset

该接口用于将位图初始化为零,一般我们需要自己定义一个sigset_t变量,把这个变量用于存储系统中对应位图信息。而这个变量定义时不初始化,所以可能会出现随机数现象,该接口就是用于将变量中的数据全部清零。

<2>sigfillset

该接口用于将所有定义的信号,设进set这个参数信号集中。

<3>sigaddset 

该接口用于将signum信号添加进set这个信号集。

<4>sigdelset

该接口用于将signum信号从set信号集中删除。

<5>sigismember

该接口用于判断信号signum是否在set这个信号集中。

前四个接口调用成功返回零,失败返回-1。最后一个接口成功返回true,失败返回false。

2、sigprocmask

该接口可以读取或更改系统中对应的信号屏蔽集(block位图).

第一个参数表示所要对信号屏蔽集做的动作,第二个参数表示新的信号屏蔽字,第三个参数表示原来的信号屏蔽集。

第一个参数一般常用的有三个选择:SIG_BLOCK 、SIG_UNBLOCK、SIG_SETMASK。第一个选项表示将set这个信号集中的屏蔽的信号添加进现有的信号屏蔽集中;第二个选项表示将set这个信号集中屏蔽的信号从原有的信号删除;第三个选项表示将set这个信号集直接替换当前的信号屏蔽集。

调用成功返回0,失败返回-1。

3、sigpending 

该接口用于读取系统中的pending信号集(pending位图)。成功返回0,失败返回-1。

4、示例:

我们以下面的场景为例,演示一下上述的相关的接口的使用方式。

场景:我们当前将2号信号阻塞,然后不断向当前进程发送2号信号,观察pending表是否一直都保存着2号信号。(pending表中一般就一个信号,当收到新的不同的信号时,原有的信号会被消除)

#include <iostream>
#include <unistd.h>
#include <signal.h>void Print(sigset_t &pe)
{std::cout << "pending map: ";for (int i = 31; i >= 1; i--){if (sigismember(&pe, i)){std::cout << "1";}else{std::cout << "0";}}std::cout << std::endl;
}
int main()
{sigset_t block, oblock;sigemptyset(&block);sigemptyset(&oblock);sigaddset(&block, 2);int n = sigprocmask(SIG_BLOCK, &block, &oblock);if (n == 0){while (true){sigset_t pending;sigemptyset(&pending);sigpending(&pending);sleep(1);Print(pending);}}return 0;
}

运行结果:

我们可以看见,当我们将2号信号阻塞时,pending中的2号信号一直没有被处理。需要说明的是,我们不能直接打印pending表,OS是以sigset_t的类型对信号进行存储,里面可能包含有别的数据,所以直接打印是错误的。在代码中,需要使用sigismember来判断信号是否存储。

上面的示例,我们屏蔽了2号信号,那我们能不能屏蔽1到31号信号呢?如果不能,那么哪些信号不会被屏蔽呢?下面通过代码对该问题进行验证。

#include <iostream>
#include <unistd.h>
#include <signal.h>void Print(sigset_t &pe)
{std::cout << "pending map: ";for (int i = 31; i >= 1; i--){if (sigismember(&pe, i)){std::cout << "1";}else{std::cout << "0";}}std::cout << std::endl;
}
int main()
{sigset_t block, oblock;sigemptyset(&block);sigemptyset(&oblock);for(int i = 1; i < 32; i++){sigaddset(&block, i);}sigaddset(&block, 2);int n = sigprocmask(SIG_BLOCK, &block, &oblock);if (n == 0){while (true){sigset_t pending;sigemptyset(&pending);sigpending(&pending);sleep(1);Print(pending);}}return 0;
}

运行结果

运行至九号信号时,进程结束。说明九号进程不可被屏蔽。后面的信号就不做实验了,在这31个信号中,还有19号信号也不可被屏蔽,其中我们将18号信号屏蔽时,附近的几个信号会在信号屏蔽集中被剔除。这两个信号被禁止屏蔽其实也就是为了防止异常进程不断运行,影响OS的正常运转或阻止异常进程读取核心数据。

通过上述的接口,我们可以实现信号的屏蔽,也可以实现信号屏蔽的解除。这里就不做演示了,但需要注意的是,一旦信号的屏蔽被解除后,pending中的信号是先被清零,再被递达。

这篇关于Linux 学习之路 - 信号的保存的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1135349

相关文章

Linux命令之firewalld的用法

《Linux命令之firewalld的用法》:本文主要介绍Linux命令之firewalld的用法,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录linux命令之firewalld1、程序包2、启动firewalld3、配置文件4、firewalld规则定义的九大

Linux之计划任务和调度命令at/cron详解

《Linux之计划任务和调度命令at/cron详解》:本文主要介绍Linux之计划任务和调度命令at/cron的使用,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录linux计划任务和调度命令at/cron一、计划任务二、命令{at}介绍三、命令语法及功能 :at

Linux下如何使用C++获取硬件信息

《Linux下如何使用C++获取硬件信息》这篇文章主要为大家详细介绍了如何使用C++实现获取CPU,主板,磁盘,BIOS信息等硬件信息,文中的示例代码讲解详细,感兴趣的小伙伴可以了解下... 目录方法获取CPU信息:读取"/proc/cpuinfo"文件获取磁盘信息:读取"/proc/diskstats"文

Linux内核参数配置与验证详细指南

《Linux内核参数配置与验证详细指南》在Linux系统运维和性能优化中,内核参数(sysctl)的配置至关重要,本文主要来聊聊如何配置与验证这些Linux内核参数,希望对大家有一定的帮助... 目录1. 引言2. 内核参数的作用3. 如何设置内核参数3.1 临时设置(重启失效)3.2 永久设置(重启仍生效

kali linux 无法登录root的问题及解决方法

《kalilinux无法登录root的问题及解决方法》:本文主要介绍kalilinux无法登录root的问题及解决方法,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,... 目录kali linux 无法登录root1、问题描述1.1、本地登录root1.2、ssh远程登录root2、

Linux ls命令操作详解

《Linuxls命令操作详解》通过ls命令,我们可以查看指定目录下的文件和子目录,并结合不同的选项获取详细的文件信息,如权限、大小、修改时间等,:本文主要介绍Linuxls命令详解,需要的朋友可... 目录1. 命令简介2. 命令的基本语法和用法2.1 语法格式2.2 使用示例2.2.1 列出当前目录下的文

Linux中的计划任务(crontab)使用方式

《Linux中的计划任务(crontab)使用方式》:本文主要介绍Linux中的计划任务(crontab)使用方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、前言1、linux的起源与发展2、什么是计划任务(crontab)二、crontab基础1、cro

Linux换行符的使用方法详解

《Linux换行符的使用方法详解》本文介绍了Linux中常用的换行符LF及其在文件中的表示,展示了如何使用sed命令替换换行符,并列举了与换行符处理相关的Linux命令,通过代码讲解的非常详细,需要的... 目录简介检测文件中的换行符使用 cat -A 查看换行符使用 od -c 检查字符换行符格式转换将

Linux系统配置NAT网络模式的详细步骤(附图文)

《Linux系统配置NAT网络模式的详细步骤(附图文)》本文详细指导如何在VMware环境下配置NAT网络模式,包括设置主机和虚拟机的IP地址、网关,以及针对Linux和Windows系统的具体步骤,... 目录一、配置NAT网络模式二、设置虚拟机交换机网关2.1 打开虚拟机2.2 管理员授权2.3 设置子

Linux系统中卸载与安装JDK的详细教程

《Linux系统中卸载与安装JDK的详细教程》本文详细介绍了如何在Linux系统中通过Xshell和Xftp工具连接与传输文件,然后进行JDK的安装与卸载,安装步骤包括连接Linux、传输JDK安装包... 目录1、卸载1.1 linux删除自带的JDK1.2 Linux上卸载自己安装的JDK2、安装2.1