今天在阅读《linux内核完全剖析-基于0.12内核》一书中信号(sigal.c)处理这一小节时发现自己原来对信号机制的理解并不是很到位,通过查阅资料整理记录下来。
创新互联为您提适合企业的网站设计 让您的网站在搜索引擎具有高度排名,让您的网站具备超强的网络竞争力!结合企业自身,进行网站设计及把握,最后结合企业文化和具体宗旨等,才能创作出一份性化解决方案。从网站策划到做网站、成都做网站, 我们的网页设计师为您提供的解决方案。
几个概念:
1)信号是什么:
一个信号就是一个消息,它通知进程一个某种类型的事件已经在系统中发生了;
用户在终端按下某些键,终端驱动程序会发送信号给前台进程;例如ctrl-c产生SIGINT信号;ctrl-\产生SIGQUIT信号;ctrl-z产生SIGSTP信号;
硬件异常产生信号;这些信号由硬件检测到并通知内核,由内核通知当前进程,例如当前进程出现除0错误;
kill系统调用,一个进程可以调用kill发送信号给另一个进程;
内核检测到某种软件设置的条件发生时也可能发信号给一个进程,例如alarm系统调用就会导致出现这样的场景;
2)阻塞信号、待处理信号:
阻塞信号指的是某个进程阻塞了某个或某些信号集;阻塞并不是说进程不接收指定的信号集;而是指信号仍可以被进程接收(放入待处理信号集),但是不处理。从内核具体的实现可以很好的理解;
内核在task_struct中维护进程信号阻塞向量(block)和pending向量(待处理信号集);block与pending一一对应,对于某一个信号:若pending对应bit置位而block复位,标识信号已发生并接收且当前未被进程阻塞,可以立刻处理(默认或用户自定义信号处理函数);若pending对应bit复位而block置位,标识信号还未被接收到,即使接收到了,那么也被当前进程阻塞(即不被立刻处理,需要接触阻塞后才可处理)。。。
3)信号处理细微问题:
待处理信号可能被阻塞:若当前进程捕捉一个SIGINT信号,并且当前正在运行该信号的处理程序,那么如果另一个SIGINT信号传递到这个进程,那么个这个SIGINT将变成待处理的,直到处理程序返回。
待处理信号不会排队等待:这个我的理解是与内核实现有关(内核一般使用位向量标识信号集),好像现在linux中有实时信号集是可以排队的(以后看到了再学习吧。。。)
系统调用可以被中断:一般是对慢速系统调用来说的,被中断的慢速系统调用在信号处理程序返回时不再继续,而是立即返回用户一个错误条件,并将errno置为EINTR;
4)可移植到信号处理程序
可移植信号语义为如下:
只有这个处理程序当前正在处理的那种类型的信号被阻塞;
和所有信号是实现一样,信号不会排队等待;
只要可能,被中断的系统调用会重启;(用户实现:显式循环调用系统调用;内核实现:恢复系统调用参数,让用户栈中的eip重新指向系统调用中断指令;)
一旦设置了信号处理程序,它就一直保持,直到重新设置。