- 论坛徽章:
- 1
|
本帖最后由 羽剑天涯 于 2015-04-01 15:22 编辑
回复 4# McHeaven
每个信号生命周期有三个阶段:1.generation产生->2.pending未决->3.delivery递送。
每个信号的指向有:1,发向进程的;2,发向线程的。
参照man 7 signal中的说法:
……
A thread can obtain the set of signals that it currently has pending using sigpending(2). This set will consist of the union of the set of pending process-directed signals and the set of signals pending for the calling thread.
……
也就是说表述2其实表达的是返回的信号集是发向进程的未决信号集和发向线程的未决信号集的合集。
我的理解是信号屏蔽字是线程独立的,如果是单线程,那么该线程的信号屏蔽字也可以称为进程屏蔽字;如果是多线程,几个线程就有几个信号屏蔽字,此时没有可以称为进程屏蔽字的(从sigprocmask在多线程时是不确定的(The use of the sigprocmask() function is unspecified in a multi-threaded process)推测)。
未决信号集分为发向进程的未决信号集和发向线程的未决信号集(各线程独立)。发向线程的信号,在pending状态就在该线程的未决信号集;发向进程的信号,在pending状态就在进程的未决信号集(时间不一定短,但凡delivery状态之前都在该信号集中),然后系统会选择一个合适的线程,递送该信号(注,在该线程响应该信号前,该信号还是处于pending状态)。
另外对于信号的处理,是进程共用的,也就是任何时刻,同一信号在同一进程中响应函数是同一个,无非是响应函数是打断了哪个线程的运行上下文,然后该线程处于其中断上下文罢了。
另附测试代码和结果如下:- #define _GNU_SOURCE
- #define T_NUM 3
- #include <stdio.h>
- #include <stdlib.h>
- #include <unistd.h>
- #include <errno.h>
- #include <fcntl.h>
- #include <signal.h>
- #include <time.h>
- #include <errno.h>
- #include <pthread.h>
- #include <sys/types.h>
- #include <sys/syscall.h>
- pthread_mutex_t plock = PTHREAD_MUTEX_INITIALIZER;
- void thread_func(sigset_t *psigs)
- {
- int i = 0;
- printf("thread[%ld] is created\n", syscall(SYS_gettid));
- pthread_sigmask(SIG_SETMASK, psigs, NULL);
- while (1) {
- pthread_mutex_lock(&plock);
- printf("thread[%ld]'s pending sigs are ", syscall(SYS_gettid));
- sigpending(psigs);
- for (i=1; i<SIGUNUSED; i++) {
- if (sigismember(psigs, i)) {
- printf("%d,", i);
- }
- }
- printf("\n");
- pthread_mutex_unlock(&plock);
- sleep(1);
- }
- }
- int main(int argc, char **argv)
- {
- int i = 0, j = 0;
- pthread_t tids[T_NUM-1];
- sigset_t tsets[T_NUM-1];
- sigfillset(tsets);
- pthread_sigmask(SIG_SETMASK, tsets, NULL);
- printf("\n");
- for (i=0; i< T_NUM-1; i++) {
- signal(SIGHUP+i, SIG_IGN);
- sigfillset(tsets+i);
- sigdelset(tsets+i, SIGHUP+i);
- if (pthread_create(tids+i, NULL, (void*(*)(void*))thread_func,(void*) (tsets+i)) != 0) {
- printf("Failed to creat thread,errno=%d\n", errno);
- return 1;
- }
- }
- sleep(1);
- while (1) {
- for (i=0; i<T_NUM-1; i++) {
- for (j=0; j<T_NUM-1; j++) {
- pthread_kill(tids[i], SIGHUP+j);
- sleep(1);
- }
- }
- }
- return 0;
- }
复制代码 在外部执行kill -3 进程号和kill -4 进程号,结果如下- thread[4451] is created
- thread[4451]'s pending sigs are
- thread[4452] is created
- thread[4452]'s pending sigs are
- thread[4451]'s pending sigs are
- thread[4452]'s pending sigs are
- thread[4452]'s pending sigs are
- thread[4451]'s pending sigs are
- thread[4452]'s pending sigs are
- thread[4451]'s pending sigs are 2,
- thread[4451]'s pending sigs are 2,
- thread[4452]'s pending sigs are 1,
- thread[4452]'s pending sigs are 1,
- thread[4451]'s pending sigs are 2,
- thread[4452]'s pending sigs are 1,
- thread[4451]'s pending sigs are 2,
- thread[4452]'s pending sigs are 1,
- thread[4451]'s pending sigs are 2,3,
- thread[4452]'s pending sigs are 1,3,
- thread[4451]'s pending sigs are 2,3,
- thread[4452]'s pending sigs are 1,3,
- thread[4452]'s pending sigs are 1,3,4,
- thread[4451]'s pending sigs are 2,3,4,
复制代码 从楼主抛出的问题,通过查阅相关资料,加深了我对Linux信号处理的理解。以上理解如有不合理的地方欢迎讨论。 |
|