Linux下的信号编程小结

1 概念
1.1 产生条件
- 用户按下了某些终端键(如Ctrl-c)时
- 硬件异常产生信号
- 进程用kill(2)函数给另一个进程或进程组发送信号
- 用户用kill(1)命令发送信号给进程
- 某种软件条件发生(如SIGPIPE等信号)
1.2 动作
- 忽略此信号
- 捕捉信号
- 执行系统默认动作
1.3 信号分类
- 不可靠信号,即普通信号。如果一个进程接收到一个普通信号,而进程的未决信号集(pending signals)中存在相同的信号,则这个新发送的信号丢失。即同一时间进程的未决信号集中只可能有一个普通信号。文件/usr/include/bits/signum.h中定义了所有普通信号。
- 可靠信号,即实时信号。信号值在SIGRTMIN(32)和SIGRTMAX(64)之间的信号。实时信号每次都会被加入到未决信号集中。

2 signal函数
2.1 函数原形
    typedef void (*sighandler_t)(int);
    sighandler_t signal(int signum, sighandler_t handler);
2.2 说明
handler的值可以是SIG_IGN(忽略信号),SIG_DFL(默认行为)或者是用户自定义的一个信号处理函数。返回的是之前的信号处理函数指针或者SIG_ERR。

3 中断的系统调用
如果进程执行一个低速系统调用而阻塞期间捕捉到一个信号,该系统调用会被中断返回出错,errno设为EINTR。4.2BSD引入了自动再启动的系统调用包括:read、readv、write、writev、ioctl、wait和waitpid。

4 可再入函数
不可再入函数一般是指:a)使用了静态数据结构,b)调用了malloc或free或c)标准I/O函数。如果在信号处理函数种调用不可再入函数,则可能会引起错误。
即使是可载入函数,因为errno每个进程只有一个,也有可能引起错误。因此一般调用使用errno的系统调用之前都要保存errno,在调用后再恢复。

5 发送等待信号
5.1 相关函数
    int kill(pid_t pid, int sig);
    int raise(int sig);
    unsigned int alarm(unsigned int seconds);
    int pause(void);
5.2 说明
kill可以给人以进程发送信号,raise则给当前进程发送信号。
alarm则设置一个闹钟值,所设置的时间超过后,产生一个SIGALARM信号,其默认动作时终止该进程。
pause是当前进程挂起直到接收到一个信号。
实例:
- sleep的不完整的实现:signals/sleep1.c

6 信号集
6.1 相关函数
    int sigemptyset(sigset_t *set);
    int sigfillset(sigset_t *set);
    int sigaddset(sigset_t *set, int signum);
    int sigdelset(sigset_t *set, int signum);
    int sigismember(const sigset_t *set, int signum);
6.2 说明
sigset是表示多个信号的数据类型。sigemptyset函数使信号集set不包含任何信号,sigfillset使set包含所有的信号。sigaddset和sigdelset添加和删除一个信号。sigismember检测信号集是否包含特定的信号。

7 POSIX信号处理函数
7.1 相关函数
    int sigaction(int signum, const struct sigaction *act, struct sigaction *oldact);
    int sigprocmask(int how, const sigset_t *set, sigset_t *oldset);
    int sigpending(sigset_t *set);
    int sigsuspend(const sigset_t *mask);
7.2 说明
用sigprocmask设定进程信号屏蔽字,参数how有三个可选项:SIG_BLOCK,SIG_UNBLOCK和SIG_SETMASK。
sigpending返回对于调用进程被阻塞的和当前未决的信号集。
sigaction取代了早期的signal函数,用于设定与信号相关联的处理动作。参数act的类型为:
    struct sigaction {
        void (*sa_handler)(int);
        void (*sa_sigaction)(int, siginfo_t *, void *);
        sigset_t sa_mask;
        int sa_flags;
    }
因为可能会用union实现,只能给两个信号处理函数指针sa_handler和sa_sigaction中的一个赋值。sa_mask是信号处理函数执行时应该被屏蔽的信号。flag则是一些标识,要使用sa_sigaction则必须设定SA_SIGINFO,这样信号的一些信息及传递的变量才会被设置到sa_sigaction的第二个参数中。关于sa_sigaction第二个参数的使用,参看man文档。
sigsuspend以参数mask为屏蔽字是进程挂起直到接收到一个信号。相当于以原子操作实现:
    sigprocmask(SIG_SETMASK, &mask, &oldmask);
    pause();
    sigprocmask(SIG_SETMASK, &oldmask, NULL);
实例:
- 父子进程同步的实现:lib.rhlin /tellwait.c
- 用sigsuspend保护临界区:signals/suspend1.c
- 用sigsuspend等待一个全局变量被设置:signals/suspend2.c
- abort的实现:signals/abort.c
- system的POSIX.2实现:signals/system.c
- 处理SIGTSTP信号(Ctrl-z):signals/sigtstp.c

8 非局部跳转
8.1 相关函数
    int sigsetjmp(sigjmp_buf env, int savesigs);
    void siglongjmp(sigjmp_buf env, int val);
8.2 说明
这两个函数和setjmp、longjmp之间的唯一区别是sigsetjmp增加了一个参数。savesigs为非0,则sigsetjmp在env中保存进程的当前屏蔽字。
实例:
- 信号屏蔽及siglongjmp的使用:signals/mask.c

你可能感兴趣的:(数据结构,linux,struct,kill,System,Signal)