当前位置: 首页 > article >正文

信号的产生和保存

信号的产生

信号就是操作系统对用户操作做出的反应,但它的本质就是往操作系统写入信号,这是由操作系统的结构决定的。通过修改比特位来告诉操作系统接收信号和传了几号信号。

也正是因为我们身为用户无法亲自修改内核数据,所以我们需要通过操作系统提供的系统调用来让操作系统帮我们做修改操作。

kill -l

我们可以通过kill -l来查看系统提供的命令列表

但是并非所有信号我们都能用到,一般情况下,我们只会用1-31号信号,这也是为什么我们程序正常结束是返回0了,因为0号信号不存在,也就不会报错。

signal

如果我们想要自定义信号的转化方式,那么必须是这样空返回值,一个整型参数的函数

这个整型参数就是用来接收转化前收到的信号的。

这样当我们遇到了sigint时就会转化成对我们handlersig的调用

那么当我们运行这个文件时,一旦我们想要ctrl+c终止时,就会被转化成对handlersig的调用

可是新的问题来了,我们要是所有信号都自定义了,那么会不会永远退不出来呢,其实不会的,因为操作系统为了防止恶意进程专门设置了一个kill -9信号,它无法被自定义,所以我们可以用它来终止进程。

这样进程依旧被我们终止了。

我们也可以重写一个直接用来删除的文件

上面这一类我们都可以归类于是键盘产生的信号。

系统调用信号

这个理解起来挺简单的,我们调用的每一个系统函数都会有信号产生,不过我们可以通过系统调用了解一下系统的信号是怎么产生的。

raise

abort

这个系统函数会强制让信号执行它的默认操作,这样即使我们捕捉了对应信号想要执行自定义操作也不行了。

异常产生信号

信号的产生还有一种,就是异常。如果我们的进程有什么问题,也会触发异常信号,比如说 /0 或者说野指针问题。

可以看到 /0的异常是8号信号,而野指针则是11号信号。

那么操作系统是怎么知道这些问题的呢。

首先操作系统管理着软硬件,而硬件里的寄存器会记录当前进程的状态,那么一旦状态出了问题,它就会通过上下文和task_struct找到进程然后发送对应的异常信号。

野指针问题也差不多,它是属于CR3存储的目录地址和虚拟地址给MMU转化后没有对应的映射关系,所以失败了,那么寄存器同样记录异常状态然后返回给进程。

alarm以及pause

alarm就单纯是个闹钟而已。

通过捕捉闹钟信号,实现个闹钟死循环。

pause则是只有信号返回时它才有反应,否则就一直暂停着。

这俩单看都没啥用的感觉,但是结合起来,它就接近操作系统的本质了。

操作系统本身也是一个死循环,只有收到对应信号才会做出对应的操作,是不是没事的时候就像上面的pause暂停,有事的时候就像接收到闹钟信号做出反应。不过操作系统里的闹钟可能是一个结构体,存储着对应任务的触发时间和优先级,再由堆管理起来,这样操作系统拿堆顶的进程开始运行,这样就知道什么时候运行上面进程又返回什么信号。

以上就是信号产生的过程。

信号的保存

信号的保存又分为三种情况 递达、未决、阻塞

递达

递达就是操作系统成功接收之后的状态。

递达又分为自定义、默认、忽略

自定义就是我们之前的signal把捕捉的信号自定义成我们想要的方法。

默认就是不捕捉,按照信号表里的方法执行信号操作。

忽略则是虽然我接收了信号,但是无视你 你哪里凉快哪里待着。

未决

未决指的是信号产生到递达前的状态。

阻塞(屏蔽)

阻塞是我专门在你递达之前卡住你,等我有空处理你的时候再让你递达。

它和忽略是前后者的关系,并不相同,忽略是已递达但是不处理,阻塞是我可能暂时没空接收你,你先别等着。

block pending handler

操作系统用三个表来标识上面的状态

横着看来判断X号信号属于什么情况。

比如一号信号已被接收未阻塞执行方法为默认。

既然有这张表那么操作系统就为我们提供了一个函数来修改它.

sigprocmask

这个函数可以修改block表内的数据

第一个是我们用什么方法修改,第二个则是根据第一个参数给,第三个是为了防止你改错了改不回来,所以需要提供一个容器存旧数据。

第一个参数block是直接新增信号到block表里。

第二个则是通过先取反再与的操作修改信号。

第三个是我们直接提供整张修改后的block表覆盖旧表。

实际上用起来第三种才方便。

这里的sigset_t 就是容器的类型,它是一个系统定义的结构体。我们把它初始化后作为参数传入

sigprocmask里,这样就设置出了一个全部不阻塞的block表,但是这些操作都只在当前作用域

所以我们需要使用sigaddset把它添加到信号集内,我们可以理解为上传存档。

然后我们就可以用sigpending查看它的数据变化了,一旦我们传入任何一个信号

那么它的全0就会被改变。

最后我们输入2号信号,我们可以看到一开始为全为0的pending表第二位变成了1,这就代表它成功接收到了我们的2号信号。

sigpending

pending表只支持查看,我们同样传入一个容器,它会把数据拷贝进容器然后返回,至于为什么不支持修改,因为我们的信号产生那么多种方式,我们大可以直接用键盘异常这样的修改。


http://www.kler.cn/a/599182.html

相关文章:

  • 数据预处理习题
  • Shiro漏洞攻略
  • FFmepg入门:最简单的视频重编码工具
  • MyBatis基础一
  • 无人船 | 基于ROS的轻量级多无人艇自主导航仿真框架
  • Git 钩子:特定操作脚本
  • GithubPages+自定义域名+Cloudfare加速+浏览器收录(2025最新排坑)
  • unix网络编程
  • 【XPipe】一款好用的SSH工具
  • 丐版插入selectdb模拟
  • Debian,Ubuntu,设置/etc/vim/vimrc.tiny解决:上下左右变成ABCD,backspace退格键失效的问题
  • netplan是如何操控systemd-networkd的? 笔记250324
  • 常见框架漏洞攻略-ThinkPHP篇
  • 搜广推校招面经五十七
  • C语言入门教程100讲(40)文件定位
  • search_fields与filterset_fields的使用
  • 【参考资料 II】C 运算符大全:算术、关系、赋值、逻辑、条件、指针、符号、成员、按位、混合运算符
  • 多线程编程
  • 模糊数学 | 模型 / 集合 / 关系 / 矩阵
  • endnote相关资料记录