Linux signal 那些事儿(4)信号的deliver顺序【转】
转自:http://blog.chinaunix.net/uid-24774106-id-4084864.html
上一篇博文提到了,如果同时有多个不同的信号处于挂起状态,kernel如何选择deliver那个信号。

next_signal
负责从挂起信号中选择deliver的signo:当然,有线程显存私有的penging,有线程组共有的pending,对于线程而言,先从自己私有的pending中选,处理完毕私有的才会去处理线程组共有的pending,这个逻辑的代码在:
- int dequeue_signal(struct task_struct *tsk, sigset_t *mask, siginfo_t *info)
- {
- int signr;
- /* We only dequeue private signals from ourselves, we don't let
- * signalfd steal them
- */
- signr = __dequeue_signal(&tsk->pending, mask, info); //线程私有的penging优先
- if (!signr) {
- signr = __dequeue_signal(&tsk->signal->shared_pending,
- mask, info);
- 。。。。
- }
换句话说,如果存在挂起队列中,我们用tkill/tgkill发送的信号会先于用kill发送的信号被deliver,这个我们按下不提,我们看在同一个penging队列中如何挑选下个deliver的signal:
- int next_signal(struct sigpending *pending, sigset_t *mask)
- {
- unsigned long i, *s, *m, x;
- int sig = 0;
- s = pending->signal.sig;
- m = mask->sig;
- /*
- * Handle the first word specially: it contains the
- * synchronous signals that need to be dequeued first.
- */
- x = *s &~ *m;
- if (x) {
- if (x & SYNCHRONOUS_MASK)
- x &= SYNCHRONOUS_MASK;
- sig = ffz(~x) + 1;
- return sig;
- }
- switch (_NSIG_WORDS) {
- default:
- for (i = 1; i < _NSIG_WORDS; ++i) {
- x = *++s &~ *++m;
- if (!x)
- continue;
- sig = ffz(~x) + i*_NSIG_BPW + 1;
- break;
- }
- break;
- case 2:
- x = s[1] &~ m[1];
- if (!x)
- break;
- sig = ffz(~x) + _NSIG_BPW + 1;
- break;
- case 1:
- /* Nothing to do */
- break;
- }
- return sig;
- }
- #define SYNCHRONOUS_MASK \
- (sigmask(SIGSEGV) | sigmask(SIGBUS) | sigmask(SIGILL) | \
- sigmask(SIGTRAP) | sigmask(SIGFPE) | sigmask(SIGSYS))
上一篇博客讲了处于SYNCHRONOUS_MASK里面的信号是优先处理的信号,他们都是一些硬件相关的信号,多是由于异常出错引起。其次是传统信号,[32,64]之间的实时信号,优先级最低。
换句话说所有信号分成三个等级,{SIGILL(4),SIGTRAP(5),SIGBUS(7),SIGFPE(8),SIGSEGV(11),SIGSYS(31)},这是第一等级,传统信号中排除第一等级的信号,就是第二等级的信号,[34,64]之间的信号属于第三等级。如果同一等级内,存在多个信号,按照小信号优先的顺序去deliver。
举个例子:
- kill -10 $signal_pid
- kill -3 $signal_pid
- kill -12 $signal_pid
- kill -11 $signal_pid
- kill -39 $signal_pid
- kill -2 $signal_pid
- kill -5 $signal_pid
- kill -4 $signal_pid
- kill -36 $signal_pid
- kill -24 $signal_pid
- kill -38 $signal_pid
- kill -37 $signal_pid
- kill -31 $signal_pid
- kill -8 $signal_pid
- kill -7 $signal_pid
我们可以看到,我们向同一进程发送多个信号,加入进程阻塞了所有信号(当然SIGKILL/SIGSTOP 不考虑)。这时候,这些个信号,都会位于挂起信号之中,一旦进程解除阻塞,那么考验deliver顺序的时候到了。
我们按照kernel的规则,顺序应该是{4,5,7,8,11,31, 2,3,10,12,24, 36,37,38}这么个顺序。
写个测试程序:
- root@manu-hacks:~/Dropbox/Note/signal# cat signal_delivery_order.c
- #include <stdio.h>
- #include <stdlib.h>
- #include <unistd.h>
- #include <signal.h>
- #include <string.h>
- #include <errno.h>
- static int sig_cnt[NSIG];
- static number= 0 ;
- int sigorder[128]= {0};
- #define MSG "#%d:receiver signal %d\n"
- void handler(int signo)
- {
- sigorder[number++] = signo;
- }
- int main(int argc,char* argv[])
- {
- int i = 0;
- int k = 0;
- sigset_t blockall_mask ;
- sigset_t pending_mask ;
- sigset_t empty_mask ;
- struct sigaction sa ;
- sigfillset(&blockall_mask);
- #ifdef USE_SIGACTION
- sa.sa_handler = handler;
- sa.sa_mask = blockall_mask ;
- sa.sa_flags = SA_RESTART;
- #endif
- printf("%s:PID is %ld\n",argv[0],getpid());
- for(i = 1; i < NSIG; i++)
- {
- if(i == SIGKILL || i == SIGSTOP)
- continue;
- #ifdef USE_SIGACTION
- if(sigaction(i,&sa, NULL)!=0)
- #else
- if(signal(i,handler)== SIG_ERR)
- #endif
- {
- fprintf(stderr,"sigaction for signo(%d) failed (%s)\n",i,strerror(errno));
- // return -1;
- }
- }
- if(argc > 1)
- {
- int sleep_time = atoi(argv[1]);
- if(sigprocmask(SIG_SETMASK,&blockall_mask,NULL) == -1)
- {
- fprintf(stderr,"setprocmask to block all signal failed(%s)\n",strerror(errno));
- return -2;
- }
- printf("I will sleep %d second\n",sleep_time);
- sleep(sleep_time);
- if(sigpending(&pending_mask) == -1)
- {
- fprintf(stderr,"sigpending failed(%s)\n",strerror(errno));
- return -2;
- }
- for(i = 1 ; i < NSIG ; i++)
- {
- if(sigismember(&pending_mask,i))
- printf("signo(%d) :%s\n",i,strsignal(i));
- }
- sigemptyset(&empty_mask);
- if(sigprocmask(SIG_SETMASK,&empty_mask,NULL) == -1)
- {
- fprintf(stderr,"setprocmask to release all signal failed(%s)\n",strerror(errno));
- return -3;
- }
- for( i = 0 ; i < 1000000 ; i++)
- {
- k = random()%1234567;
- }
- }
- for(i = 0 ; i< number ; i++)
- {
- if(sigorder[i] != 0)
- {
- printf("#%d: signo=%d\n",i,sigorder[i]);
- }
- }
- return 0;
- }
注意这个USE_SIGACTION宏包裹的部分是我后来想到的,在我开始的版本中用的是signal(i,handler)来注册函数。handler函数很有意思,是我精心设计的:
- void handler(int signo)
- {
- sigorder[number++] = signo;
- }
按照执行的顺序,我会讲signo的值记录在全局数组中,等到进程退出前,我打印数组的值,就能得到信号deliver的顺序。
这个进程会阻塞所有信号一段时间,在这段时间内,我会向该进程发送一坨信号,待阻塞解除后,打印数组的值,从而获得deliver的顺序。
看下测试程序:
- root@manu-hacks:~/code/c/self/signal_deliver# cat test_order.sh
#!/bin/bashif [ $1 -eq 0 ]
then
./sigaction_delivery_order 30 & #正确的程序
else
./signal_delivery_order 30 & #我最初的程序,信号处理函数执行期间,没有屏蔽其他信号。
fi
signal_pid=$!sleep 2
kill -10 $signal_pid
kill -3 $signal_pid
kill -12 $signal_pid
kill -11 $signal_pid
kill -39 $signal_pid
kill -2 $signal_pid
kill -5 $signal_pid
kill -4 $signal_pid
kill -36 $signal_pid
kill -24 $signal_pid
kill -38 $signal_pid
kill -37 $signal_pid
kill -31 $signal_pid
kill -8 $signal_pid
kill -7 $signal_pid
设计很精巧,设计出这个程序后,我很得意,认为验证deliver传递顺序是水到渠成的事情。
梦想很丰满,无奈现实很骨感,我看了执行结果,那是当头一棒啊:
- root@manu-hacks:~/Dropbox/Note/signal#
- root@manu-hacks:~/Dropbox/Note/signal# ./test_order.sh 1
- ./signal_delivery_order:PID is 31403
- sigaction for signo(32) failed (Invalid argument)
- sigaction for signo(33) failed (Invalid argument)
- I will sleep 30 second
- root@manu-hacks:~/Dropbox/Note/signal# signo(2) :Interrupt
- signo(3) :Quit
- signo(4) :Illegal instruction
- signo(5) :Trace/breakpoint trap
- signo(7) :Bus error
- signo(8) :Floating point exception
- signo(10) :User defined signal 1
- signo(11) :Segmentation fault
- signo(12) :User defined signal 2
- signo(24) :CPU time limit exceeded
- signo(31) :Bad system call
- signo(36) :Real-time signal 2
- signo(37) :Real-time signal 3
- signo(38) :Real-time signal 4
- signo(39) :Real-time signal 5
- #0: signo=39
- #1: signo=38
- #2: signo=37
- #3: signo=36
- #4: signo=24
- #5: signo=12
- #6: signo=10
- #7: signo=3
- #8: signo=2
- #9: signo=31
- #10: signo=11
- #11: signo=8
- #12: signo=7
- #13: signo=5
- #14: signo=4
顺序恰恰相反!!!!
我最初完全解释不通,我google了类似的topic,我发现,我不是第一个发现这个问题的人,绚丽也尘埃在一篇博客中提到:
- 在网上找到这样一段话:
- 信号的优先级:信号实质上是软中断,中断有优先级,信号也有优先级。如果一个进程有多个未决信号,则对于同一个未决的实时信号,内核将按照发送的顺序来递送信号。如果存
在多个未决的实时信号,则值(或者说编号)越小的越先被递送。如果既存在不可靠信号,又存在可靠信号(实时信号),虽然POSIX对这一情况没有明确规
定,但Linux系统和大多数遵循POSIX标准的操作系统一样,将优先递送不可靠信号。 - 经过我反反复复地试验,我发现实验结果和上面描述的刚好相反,信号的编号越大越先被递送,一个进程如果处理SIGQUIT(3),SIGINT(2),SIGHUP(1)(通过”kill -l”可以查看信号的编号),那么先后给该进程发送SIGINT,SIGHUP,SIGQUIT,处理的顺序会是SIGQUIT,SIGINT,SIGHUP,不论改变这个三个信号的发送顺序,处理的顺序都是一样的。
看到了,这位前辈遇到了和我一样的困惑,测试的结果和kernel完全相反。内核不会错,glibc也肯定不会瞎掺合,一定是我的测试程序存在问题:
今天我坐公交车上,突然意识到问题的所在,我的信号处理函数没有屏蔽信号!!!
换句话说,4号信号是先被deliver的,但是还没能handler执行,被5号信号中断掉了,5号信号还没开始执行,被7号信号中断掉了,依次类推,所以我们测试的结果和deliver的结果正好相反。
意识到这一点,我就改进了我的程序,信号执行期间,屏蔽所有信号,这样,就能测试信号deliver的顺序了。对于我的程序而言就是加上-DUSE_SIGACTION选项,让sigaction安装信号时,指明信号处理函数执行期间,屏蔽所有信号。
在我的64位Ubuntu上执行,结果和kernel代码以及手册上的一样。也就是说,并不是手册描述错了,而是我们的老的测试程序,在signal处理期间,没有屏蔽其他信号导致混乱。那么按照正确的方法测试:
- root@manu-hacks:~/code/c/self/signal_deliver# ./test_order.sh 0
- ./sigaction_delivery_order:PID is 3652
- sigaction for signo(32) failed (Invalid argument)
- sigaction for signo(33) failed (Invalid argument)
- I will sleep 30 second
- root@manu-hacks:~/code/c/self/signal_deliver# signo(2) :Interrupt
- signo(3) :Quit
- signo(4) :Illegal instruction
- signo(5) :Trace/breakpoint trap
- signo(7) :Bus error
- signo(8) :Floating point exception
- signo(10) :User defined signal 1
- signo(11) :Segmentation fault
- signo(12) :User defined signal 2
- signo(24) :CPU time limit exceeded
- signo(31) :Bad system call
- signo(36) :Real-time signal 2
- signo(37) :Real-time signal 3
- signo(38) :Real-time signal 4
- signo(39) :Real-time signal 5
- #0: signo=4
- #1: signo=5
- #2: signo=7
- #3: signo=8
- #4: signo=11
- #5: signo=31
- #6: signo=2
- #7: signo=3
- #8: signo=10
- #9: signo=12
- #10: signo=24
- #11: signo=36
- #12: signo=37
- #13: signo=38
- #14: signo=39
和我们预想的完全符合,和内核代码已经手册完全一样。那么这个问题完美解决。
- {4,5,7,8,11,31, 2,3,10,12,24, 36,37,38}
多个挂起信号时,delivery的策略如下:
1 {SIGILL(4),SIGTRAP(5),SIGBUS(7),SIGFPE(8),SIGSEGV(11),SIGSYS(31)}第一等级
2 非实时信号中其他信号是第二等级(SIGKILL SIGSTOP除外)
3 实时信号是第三等级。
存在第一等级的信号挂起,那么优先选择第一等级,
没有第一等级,那么如果存在第二等级的信号,优先选择第二等级内信号。
既没有第一等级,又没有第二等级,那么选择第三等级的信号。
如果同一个等级内都存在多个挂起信号,则小信号优先。
这只是我们用程序测试的结果,其实systemtap提供了signal_deliver这个event让我们monitor,我们可以直观的看到信号传递的顺序:
- root@manu-hacks:~/code/c/self/signal_deliver# cat signal_deliver.stp
- probe kernel.trace("signal_deliver"){
- if(pid() == target())
- {
- printf("signo(%2d) is delivered to PID %8d\n",$sig,pid());
- }
- }
我们可以用test_order.sh 1
,故意用signal那个给我带来困扰的程序测试,我们会看到,传递的顺序依然是对的,这证明了我前面的推测,4是最先deliver的,只不过是因为没有屏蔽其他信号,被5号信号中断了,5又被7号信号中断了,依次类推,导致了我们看到了相反的执行顺序,给我们带来了困扰。
- root@manu-hacks:~/code/c/self/signal_deliver# ./test_order.sh 1
./signal_delivery_order:PID is 4051
sigaction for signo(32) failed (Invalid argument)
sigaction for signo(33) failed (Invalid argument)
I will sleep 30 second
root@manu-hacks:~/code/c/self/signal_deliver# stap -x 4051 signal_deliver.stp
signo(2) :Interrupt
signo(3) :Quit
signo(4) :Illegal instruction
signo(5) :Trace/breakpoint trap
signo(7) :Bus error
signo(8) :Floating point exception
signo(10) :User defined signal 1
signo(11) :Segmentation fault
signo(12) :User defined signal 2
signo(24) :CPU time limit exceeded
signo(31) :Bad system call
signo(36) :Real-time signal 2
signo(37) :Real-time signal 3
signo(38) :Real-time signal 4
signo(39) :Real-time signal 5
#0: signo=39
#1: signo=38
#2: signo=37
#3: signo=36
#4: signo=24
#5: signo=12
#6: signo=10
#7: signo=3
#8: signo=2
#9: signo=31
#10: signo=11
#11: signo=8
#12: signo=7
#13: signo=5
#14: signo=4
signo( 4) is delivered to PID 4051
signo( 5) is delivered to PID 4051
signo( 7) is delivered to PID 4051
signo( 8) is delivered to PID 4051
signo(11) is delivered to PID 4051
signo(31) is delivered to PID 4051
signo( 2) is delivered to PID 4051
signo( 3) is delivered to PID 4051
signo(10) is delivered to PID 4051
signo(12) is delivered to PID 4051
signo(24) is delivered to PID 4051
signo(36) is delivered to PID 4051
signo(37) is delivered to PID 4051
signo(38) is delivered to PID 4051
signo(39) is delivered to PID 4051^Croot@manu-hacks:~/code/c/self/signal_deliver#
话说systemtap提供了很多signal相关的example脚本,非常好用,如下图,收集signal的发送情况:

参考文献
1 Linux实时信号排队2 LKML:
| Subject | [PATCH -tip v4 2/3] tracepoint: Add signal deliver even |
Linux signal 那些事儿(4)信号的deliver顺序【转】的更多相关文章
- Linux signal 那些事儿(2)【转】
转自:http://blog.chinaunix.net/uid-24774106-id-4064447.html 上一篇博文,基本算是给glibc的signal函数翻了个身.现在glibc的sign ...
- Linux signal那些事儿【转】
转自:http://blog.chinaunix.net/uid-24774106-id-4061386.html Linux编程,信号是一个让人爱恨交加又不得不提的一个领域.最近我集中学习了Linu ...
- Linux signal 那些事儿 (3)【转】
转自:http://blog.chinaunix.net/uid-24774106-id-4065797.html 这篇博客,想集中在signal 与线程的关系上,顺带介绍内核signal相关的结构. ...
- linux signal 处理
v/:* {behavior:url(#default#VML);} o/:* {behavior:url(#default#VML);} w/:* {behavior:url(#default#VM ...
- linux signal 用法和注意事项
http://blog.chinaunix.net/uid-9354-id-2425031.html 所以希望能用相同方式处理信号的多次出现,最好用sigaction.信号只出现并处理一次,可以用si ...
- <摘录>linux signal 列表
Linux 信号表 Linux支持POSIX标准信号和实时信号.下面给出Linux Signal的简表,详细细节可以查看man 7 signal. 默认动作的含义如下: 中止进程(Term) 忽略 ...
- Linux系统编程之----》信号
"===信号========================================================================================= ...
- linux signal 列表
Linux 信号表 Linux支持POSIX标准信号和实时信号.下面给出Linux Signal的简表,详细细节可以查看man 7 signal. 默认动作的含义如下: Term 终止进程 ...
- Linux系统编程——进程间通信:信号中断处理
什么是信号? 信号是 Linux 进程间通信的最古老的方式.信号是url=474nN303T2Oe2ehYZjkrggeXCaJPDSrmM5Unoh4TTuty4wSgS0nl4-vl43AGMFb ...
随机推荐
- python之2.x与3.x区别(仅限于基础)
因为看的是python2.x的书籍.用的是python 3.7.所以先把两者的区别记录一下,仅限于基础. 1.input python3.0之后,不区分input()和raw_input(),统一为i ...
- confirm() 方法用于显示一个带有指定消息和 OK 及取消按钮的对话框。系统自带提示
W3C地址::::::: http://www.w3school.com.cn/jsref/met_win_confirm.asp http://www.w3school.com.cn/tiy/t ...
- Fakeapp 入门教程(2):使用篇!
Fakeapp软件的使用主要分成了三个步骤, 使用之前请确保你的电脑配置还可以,推荐配置是:一张显存大于4G的N卡.Fakeapp是有支持CPU选项,但是用CPU跑非常慢. 获取脸部图片 训练模型 生 ...
- Vue钩子函数生命周期实例详解
vue生命周期简介 Vue实例有一个完整的生命周期,也就是从开始创建.初始化数据.编译模板.挂载Dom.渲染→更新→渲染.卸载等一系列过程,我们称这是Vue的生命周期.通俗说就是Vue实例从创建到销毁 ...
- 什么是redis缓存穿透, 缓存雪崩, 缓存击穿
什么是redis? redis是一个非关系型数据库,相对于其他数据库而言,它的查询速度极快,且能承受的瞬时并发量非常的高.所以常常被用来存放网站的缓存,以减少主要数据库(如mysql)的服务器压力. ...
- 字符编码,ASCII、Unicode与UTF-8的理解
首先我们先要明白的两点是:1.计算机中的信息都是由二进制的0和1储存的:2.我们再计算机屏幕上看到的各种字符都是计算机系统按照一定的规则将二进制数字转换而来的. 一.基本概念. 1.字符集(chars ...
- 从头开始学习数据库及ADO.NET——竹子整理
目前为止,学习编程一年有余,写过管理系统,写过商城,写过桌面,接触的多了,乱七八糟的点太多,一堆前段框架,后台类库,纷纷杂杂,更新迭代之快也是令人咋舌.于是我就在想,作为一名程序员,哪些内容是实打实的 ...
- Linux档案与文件系统的压缩与打包
总结: 压缩指令为透过一些运算方法去将原本的档案进行压缩,以减少档案所占用的磁盘容量.压缩前与压缩后的档案所占用的磁盘容量比值,就可以被称为是“压缩比” 压缩的好处是可以减少磁盘容量的浪费,在www网 ...
- Django基础之数据库与ORM
一.数据库配置 1.django默认支持sqlite,mysql, oracle,postgresql数据库. django默认使用sqlite的数据库,默认自带sqlite的数据库驱动 , 引擎名称 ...
- IBOutletCollection 索引获取顺序问题
在sb中绑定了一个IBOutletCollection后,根据索引获取元素发现和自己拖线时的顺序不同,有时又会根据顺序,不知道是xcode的bug还是本身就是无序的. 在使用的时候直接排序: - (v ...