【Linux系统编程二十三】:(信号2)--信号的保存

发布时间:2023年12月24日

一.信号的保存

信号发送本质上是操作系统发送信号,而进程PCB内部有一个位图用来表示是否接收到信号。如果接收到某个信号,该位置上的比特位就会由0置1。该位图我们称为penging表。

信号本质上其实就是一个一到三十一的数字,并且每一种信号都要有它对应的处理方法,即每一个信号都有它自己默认处理方法。所以进程的内核当中呢,还要为每一个信号维护一个叫做handler表。这个handle表它的类型是一个函数指针数组。里面都是信号对应默认处理方法。这里面的默认处理方法可以被捕捉变成自定义方法。
在这里插入图片描述

1.这个信号将来可以充当我们数组的下标,找到某一个位置。所以每个信号都对应着函数指针数组里的方法。
2.所以一旦信号产生了,那么我们直接拿着信号编号缩影一张表,就可以找到处理该信号的方式。

我们为什么要保存信号呢?

因为进程接收到信号之后,可能不会立即处理这个信号,信号不会被立即处理,那么就要有一个时间窗口,信号需要保存起来。

所以我们进程在运行时,一旦接收到信号了,那么对应的信号位图上的比特位就会由0置1,进程会处理这个信号,但是在合适的时候去处理。
所以信号暂时就在位图中保存起来。等进程忙完它的事情后,回来根据位图上的比特位来是否处理。这样信号就保存起来了。

不过信号的保存,不单单是信号位图的保存记录,还有信号阻塞,信号方法的保存。

1.阻塞信号

其实进程可以选择阻塞某个信号。什么意思呢?
进程接收到诸多信号后,可能不会立即处理,而是在适当的时候处理,但最后总归是会处理的。而如果一旦你把某些信号屏蔽了,在该信号没有被解除屏蔽之前,那么即便你收到了该信号。那么对应的信号也不会被操作系统系统进行处理。
还有如果没有产生一个信号,可以被屏蔽吗?可以被阻塞吗?
当然可以。屏蔽是一种状态。和你当前是否产生没关系。
所以在内核里进程还需要维护一张表叫做block表。它也是一张位图,比特位的位置决定的是信号的编号,比特位的内容是为0表示不屏蔽,为1表示屏蔽。那么阻塞也就意味着信号不会被处理。

在这里插入图片描述
理解普通信号时,我们只需要了解清楚三张表就可以,分别叫做pending表、block表和handler表。
pending表用来记录当前进程是否收到了信号,以及收到了哪些信号。
block表用来记录特定信号,那么是否被屏蔽。
handler表描述的是每一种信号所对的处理关系。

所以呢我们有了三张表,即两张位图和一个函数指针数组,就可以用来实现对普通信号的记录,保存相关的管理工作。

2.sigset_t类型(位图)

刚刚说的那pending表、handler表、block它都属于操作系统内部的内核数据结构。操作系统不相信任何用户,他不允许用户直接去修改。
如果用户你想改甚至想读获取这些表,对不起,不能直接读。
所以我们一定未来的访问这三张表是必须得有系统调用接口。

我们要获取这几个位图的时候,那么就注定了要在用户空间和内核空间内核空间和用户空间进行来回的数据拷贝。
所以数据拷贝时,我们就要在接口的参数设计上,要设置我们对应的输出出行参数。理论上获取内核里位图的信息,我们应该也用一个位图变量,将内核里的数据都拷贝到这个位图变量里来,然后由这个位图变量带出来。

如果用户想拿到你这个进程的pending表或block表,那么他怎么获取呢?
所以就必须要求操作系统给我们在应用上设计出一种数据类型。那么这个sigset_t类型呢,它就是我们传说中的位图结构。这是操作系统呢给我们提供了一种类型,叫做信号集类型。它是由操作系统给我们提供,在我们用户层可以直接使用的一种数据类型。

它属于系统级的数据类型,那么我们内部如何去呃存储对应的比特位呢?我们是不能直接用位操作获取位图里的比特位的!
在这里插入图片描述

如果我们将来拿到了进程的penging表、block表,以及我们拿到了我们所谓的sigset_t这样的位图类型,你也绝对不能自己再来进行位操作了啊,不允许你这么做,因为这是操作系统级别的类型。用户不能直接操作。

我们如果要修改对应的这这个sigset_t 位图,我们需要使用下面的这批系统调用。
在这里插入图片描述

3.block表

在这里插入图片描述
在这里插入图片描述
sigprocmask是用来读取或者更改进程的屏蔽信号的
它等价于其实就是阻塞信号器。它可以通过我们传递三个参数来达到对信号屏蔽字。

1.int how 第一个参数:是对block表的操作,你要干啥,是要新增一个屏蔽字还是删除一个屏蔽字呢?总共有三种选择:
①SIG_BLOCK:新增一个屏蔽字
②SIG_UNBLOCK:解除一个屏蔽字
③SIG_SETMAS:覆盖式的设置屏蔽字
2.sigset_t *set 第二个参数:是用来设置屏蔽字的,你要新增或者屏蔽哪个字段都在这个位图里设置,然后它会作为输入型参数配合how,设置到内核里。
3.sigset_t *oset 第三个参数:是将内核里的block位图先保存起来,未来可以还需要使用,保存就是为了恢复。所以作为输出型参数。

4.handler表

handler表本质上就是一个函数指针数组,它里面是信号对应的默认处理方法或者自定义方法。我们利用捕捉就可以获取到handler表里的方法。在这里插入图片描述
捕捉的方法有两种,一个是signal,一个是sigaction。
signal有两个参数,一个参数signum是信号的编号,另一个sighandler_t handler参数就是信号对应的处理方法。可以是操作系统提供的默认方法,也可以是自定义方法。我们应该在一开始就设置捕捉,而不是在最后,因为,捕捉信号只有接收到信号才会捕捉,没有接收到信号是不会捕捉的。如果放在最后,万一接收信号的时候,还没执行到捕捉代码那就不行了。所以捕捉代码最好写在一开头。

捕捉的本质就是去执行信号对应的方法。

在这里插入图片描述

5.pending表

在这里插入图片描述
这个sigpending函数就是用来获取pending位图里的比特位信息的。
它的参数是一个输出型参数,它的作用非常简单,它的作用就是要帮助我们把。调用进程它所对应的pending表给我们以位图的形式带出来。

二.实验验证

我们可以做一个验证:将二号信号屏蔽,然后发送二号信号,捕捉二号信号,再获取进程的pending表,最后再解除二号信号的屏蔽。会发生什么样的过程呢?

在这里插入图片描述

#include <iostream>
#include <signal.h>
#include <unistd.h>
using namespace std;

void PrintPending()
{
    sigset_t set;//位图变量,输出型参数

    sigpending(&set);//将penging表带出来

    for(int signo=1;signo<=31;signo++)
    {
        if(sigismember(&set,signo))//如果signo信号在set表里就为真
        cout<<"1";
        else
        cout<<"0";
    }
    cout<<endl;

}

void handler(int signo)
{
   cout<<"catch a signal "<<signo<<endl;
}
//保存信号主要靠三张表---block表,pending表和handler表

//现在我们想主动的屏蔽一个信号该如何实现呢?利用sigprocmask系统调用

int main()
{
    //0.首先对2号信号进行捕捉
    signal(2,handler);
    //1.对二号信号进行屏蔽
    //首先要信号是在位图里,所以我们需要利用位图来修改信号,需要位图变量。
    sigset_t bset,oset;//输入型参数,输出型参数,保存就是为了恢复
    sigemptyset(&bset);//对这两个位图变量初始化
    sigemptyset(&oset);
    sigaddset(&bset,2);//设置屏蔽字,将二号信号添加到bset位图里
    sigprocmask(SIG_SETMASK,&bset,&oset);//将best覆盖式屏蔽位图里的信号

    int cnt=0;
    //2.重复打印当前进程的pending表
    while(true)
    {
        PrintPending();
        sleep(1);
        cnt++;
        if(cnt==10)//解除2好信号屏蔽
        {
          
          cout<<"unblock signal"<<endl;
         sigprocmask(SIG_SETMASK,&oset,nullptr);//将原来的旧表覆盖
        }
    }
}

它的结果就是当发送2号信号时,进程并不会捕捉2号信号处理,打印pending表示,显示2号信号上的位图由0置1,说明接收到信号。但不处理。说明被屏蔽。
当解除屏蔽时,进程就会立即处理2号信号。

还有一件事,那就是并不是所有信号都可以被屏蔽,19号信号就不能捕捉也不能屏蔽。
在这里插入图片描述

所以信号最重要的就是这三个表了。而对这三个表的操作也很重要。
在这里插入图片描述

三.信号的其他概念

在这里插入图片描述
1.信号递达对应处理的是handler表。信号未决由pending表显示。信号阻塞由block表显示。
2.普通信号只会保存一次,即如果一次性发送许多同一的信号,该信号只会被保存一次。而如果该信号被屏蔽了,当发送许多相同信号,pending位图里只会记录一次。当该信号解除屏蔽,也只会执行一次方法。

文章来源:https://blog.csdn.net/Extreme_wei/article/details/135180689
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。