Linux-信号的捕捉-信号的阻塞-volatile关键字

本文详细介绍了信号的捕捉流程,包括信号的注册与PCB的关系,信号处理的时机,以及如何在内核态中进行。讨论了信号的阻塞机制,如何设置和解除阻塞,并提供了相关接口。最后,提到了volatile关键字在确保内存可见性方面的作用,以一个实际例子展示了其在信号处理中的应用。
摘要由CSDN通过智能技术生成

1. 信号的捕捉流程

1.1 引导

1.信号的注册,是不是和操作系统维护的进程的PCB有关系呀?
        struct task_struct ==> sigset_t signal
  结论:信号的注册是在操作系统和PCB打交道的,,注册是在进程的PCB当中。

1.2 捕捉流程

问题: 信号什么时候进行处理?
  信号的处理是在内核态完成的
在这里插入图片描述
  1.从内核态切换回用户态的时候,一定会调用do_signal函数来处理进程收到的信号,sig位图当中有信号注册,就执行信号注销的逻辑,sig位图当中没有信号注册,则直接返回用户态
  2.处理信号(信号注销)
    (1)默认处理方式(直接在操作系统内核的代码当中就完成了)
    (2)忽略处理方式(直接在操作系统内核的代码当中就完成了)
    (3)自定义处理方式(调用程序员自己写的函数)
  do_signal函数直接去用户态找sigcallback函数,执行完通过sigreturn函数,切换回内核态,然后再调用一次do_signal函数,判断如果没有信号来,通过sys_sigreturn函数返回到用户态。
  3.流程
    1.在用户执行程序员自己定义的函数
    2.调用sigreturn函数再次回到操作系统内核
    3.再次调用do_signal函数判断是否有信号注册
    4.调用sys_sigreturn函数回到用户态继续去执行代码

问题: 什么时候进入到操作系统内核?
  1.调用系统调用函数的时候
  2.调用C库函数的时候,因为C库函数的内部也是调用了系统调用函数
  3.内存访问越界,访问空指针

2. 信号的阻塞

  1.信号的阻塞不会影响信号的注册
  2.内核的代码

struct task_struct
{
	...
	sigset_t blocked;
	...
};

  blocked位图,作用是阻塞某一个信号的处理,也就是意味着,执行流进入到内核之后,调用do_signal函数判断是否有信号需要处理的时候,发现某一个信号注册了,同时也是需要判断block位图当中该信号对应的bite位是否为1,如果不为1,则暂时不处理该信号(暂时不信号注销),如果为0,则进行处理。

  3.接口
在这里插入图片描述
  how:想让sigprocmask函数做什么
    SIG_BLOCK:设置某个信号为阻塞状态
    SIG_UNBLOCK:设置信号为非阻塞状态
    SIG_SETMASK:设置新的block位图
  set:使用set去设置block位图
    SIG_BLOCK:设置某个信号为阻塞状态
  block(new)= block(old)| set
  eg:block(old): 0101 0000
    set: 0000 1000
    > 0101 1000
  SIG_UNBLOCK:设置信号为非阻塞状态
  block(new)= block(old)&(~set)
  eg:block(old):0101 0000
    &(~set):1011 1111
    
> 0001 0000
  SIG_SETMASK: 设置新的block位图
    block(new)= set
例外:
  信号的阻塞不会将9号信号阻塞掉,也不会阻塞19号信号

问题: 验证非可靠信号的注册,验证可靠信号的注册
  结论:非可靠信号注册一次只添加一个sigqueue结点,可靠信号注册每添加一次,增加一个sigqueue结点

3. volatile

  作用:使变量保证内存可见,通俗的讲cpu在数据计算的时候,为了快,对已经从内存当中拿回来的数据不会再从内存获取,而是直接从寄存器当中获取,加上volatile之后变量的来源一定从内存获取。
  内存–>缓存–>寄存器–>cpu

#include <stdio.h>
#include <signal.h>

volatile int g_val = 1;

void sigcallback(int sig)
{
    g_val = 0;
    printf("sig: %d\n", sig);
}

int main()
{
    signal(2, sigcallback);

    while(g_val)
    {}
    return 0;
}

  上述这段代码执行时,g_val是一个全局变量,它的值为1,因此程序执行到while的时候是死循环,当我们通过计算机发送一个2号信号ctrl+c,执行回调函数,将全局变量g_val的值置为0,为了验证执行过回调函数,打印sig,回调函数执行结束,while循环就结束了。
  -O0,-O1,-O2,-O3:优化级别是由高低到高的。使用g_val的时候一定要从内存获取,不要从寄存器获取。

评论 5
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

天津 唐秙

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值