信号之函数kill、raise、alarm、pause、sigprocmask、sigpending以及信号集

本文来自个人博客: https://dunkwan.cn

函数killraise

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种情形:

  1. pid > 0 将该信号发送给进程ID为pid的进程。
  2. pid == 0 将该信号发送给与发送进程属于同一进程组的所有进程,而且发送进程具有权限向这些进程发送信号。这里所用的术语“所有进程”不包括实现定义的系统进程集。对于大多数UNIX系统,系统进程集包括内核进程和init(pid为1)。
  3. pid < 0 将该信号发送给其进程组ID等于pid绝对值,而且发送进程具有权限向其发送信号的所有进程。
  4. pid == -1 将该信号发送给发送进程有权限向它们发送信号的所有进程。这里的所有进程不包括系统进程集中的进程。

函数alarmpause

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;若出错,返回-1int 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");
}

结果如下:

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值