本文来自个人博客: https://dunkwan.cn
函数kill
和raise
kill
函数将信号发送给进程或进程组。raise
函数则允许进程向自身发送信号。
#include <signal.h>
int kill(pid_t pid, int signo);
int raise(int signo);
两个函数返回值:若成功,返回0;若出错,返回-1。
调用raise(signo);
等价于kill(getpid(), signo);
。
kill
函数的pid
参数有以下4种情形:
pid
> 0 将该信号发送给进程ID为pid
的进程。pid
== 0 将该信号发送给与发送进程属于同一进程组的所有进程,而且发送进程具有权限向这些进程发送信号。这里所用的术语“所有进程”不包括实现定义的系统进程集。对于大多数UNIX系统,系统进程集包括内核进程和init
(pid为1)。pid
< 0 将该信号发送给其进程组ID等于pid
绝对值,而且发送进程具有权限向其发送信号的所有进程。pid
== -1 将该信号发送给发送进程有权限向它们发送信号的所有进程。这里的所有进程不包括系统进程集中的进程。
函数alarm
和pause
alarm
函数用于设置一个定时器(闹钟时间),在将来的某个时刻该定时器会超时。当定时器超时时,产生SIGALRM
信号。如果忽略或不捕捉此信号,则其默认动作是终止调用该alarm
函数的进程。
pause
函数用于使调用进程挂起直至捕捉到一个信号。
#include <unistd.h>
unsigned int alarm(unsigned int seconds);
返回值:0或以前设置的闹钟时间的余留秒数。
int pause(void);
返回值:-1,errno设置为EINTR。
信号集
不同的信号的编号可能超过一个整型量所包含的位数,所以一般而言,不能用整型量中的一位来代表一种信号,也就是不能用一个整型量表示信号集。POSIX.1定义数据类型sigset_t
以包含一个信号集,并且定义了下列5个处理信号集的函数。
#include <signal.h>
int sigemptyset(sigset_t *set);
int sigfillset(sigset_t *set);
int sigaddset(sigset_t *set, int signo);
int sigdelset(sigset_t *set, int signo);
4个函数返回值:若成功,返回0;若出错,返回-1。
int sigismember(const sigset_t *set, int signo);
返回值:若真,返回1;若假,返回0。
以下是上述函数的具体实现:
#include <signal.h>
#include <errno.h>
#define SIGBAD(signo) ((signo) <= 0 || (signo) >= NSIG)
int sigaddset(sigset_t *set, int signo)
{
if(SIGBAD(signo)){
errno = EINVAL;
return (-1);
}
(*set) |= 1 << (signo - 1);
return (0);
}
int sigdelset(sigset_t *set, int signo)
{
if(SIGBAD(signo)){
errno = EINVAL;
return (-1);
}
*set &= ~(1 <<(signo -1));
return (0);
}
int sigismember(const sigset_t *set, int signo)
{
if(SIGBAD(signo)){
errno = EINVAL;
return (-1);
}
return ((*set & (1 << (signo -1))) != 0);
}
函数sigprocmask
sigprocmask
函数用于检测或更改,或同时检测和更改进程的信号屏蔽字。
#include <signal.h>
int sigprocmask(int how, const sigset_t *restrict set, sigset_t *restrict oset);
返回值:若成功,返回0;若出错,返回-1。
若
oset
是非空指针,那么进程的当前信号屏蔽字通过oset
返回。若
set
是非空指针,则参数how
指示如何修改当前信号屏蔽字。若
set
是空指针,则不改变该进程的信号屏蔽字,how
值此时无意义。
how
的取值如下表:
函数sigpending
sigpending
函数返回一信号集,对于调用进程而言,其中的各信号是阻塞不能递送的,因而也一定是当前未决的。该信号集通过set
参数返回。
#include <signal.h>
int sigpending(sigset_t *set);
返回值:若成功,返回0;若出错,返回-1。
测试示例:
#include "../../include/apue.h"
static void sig_quit(int);
int main(void)
{
sigset_t newmask, oldmask, pendmask;
if(signal(SIGQUIT, sig_quit) == SIG_ERR)
err_sys("can't catch SIGQUIT");
sigemptyset(&newmask);
sigaddset(&newmask, SIGQUIT);
if(sigprocmask(SIG_BLOCK, &newmask, &oldmask) < 0)
err_sys("SIG_BLOCK error");
sleep(5);
if(sigpending(&pendmask) < 0)
err_sys("sigpending error");
if(sigismember(&pendmask, SIGQUIT))
printf("\nSIGQUIT pending\n");
if(sigprocmask(SIG_SETMASK, &oldmask, NULL) < 0)
err_sys("SIG_SETMASK error");
printf("SIGQUIT unblocked\n");
sleep(5);
return 0;
}
static void sig_quit(int signo)
{
printf("caught SIGQUIT\n");
if(signal(SIGQUIT, SIG_DFL) == SIG_ERR)
err_sys("can't reset SIGQUIT");
}
结果如下: