使用sigprocmask设置掩码仅取决于调用的位置

时间:2014-04-24 15:26:54

标签: linux linux-kernel signals sigprocmask

我有一个奇怪的行为,其中manpage和google没有帮助。

在我的代码中,我希望在发送SIGUSR2时阻止/取消阻止SIGINT。为此,我安装信号处理程序并在函数中准备掩码集:

void installSignallHandler(){

    sBlock.sa_handler = handleBlock;
    sigemptyset(&sBlock.sa_mask);
    sigaddset(&sBlock.sa_mask, SIGUSR2);
    sBlock.sa_flags = 0;
    sigaction(SIGUSR2, &sBlock, NULL);

    sigemptyset(&signals_protected);
    sigaddset(&signals_protected, SIGINT);

    // For some reason sigprocmask only works if it is called here the first time
    // sigprocmask(SIG_BLOCK, &signals_protected, NULL);
    // sigintBlocked = true;
}

然后,如果发送SIGUSR2,则调用此函数:

void handleBlock(int signal){
    if(sigintBlocked){
        printf("Unblocking SIGINT ...\n");
        sigprocmask(SIG_UNBLOCK, &signals_protected, NULL);
        sigintBlocked = false;
    }
    else{
        printf("Blocking SIGINT ...\n");
        sigprocmask(SIG_BLOCK, &signals_protected, NULL);
        sigintBlocked = true;
    }
}

为了进行测试,我这样称呼它:

int main(int argc, char **argv) {
    installSignallHandler();
    while(1){
        printf("processing...\n");
        sleep(1);
    }
    return EXIT_SUCCESS;
}

现在问题:我发布代码的方式,sigprocmask不起作用。但如果我取消注释上面的两行,它就可以了。所以我的两个问题是:

  1. 你能解释一下这种行为吗?
  2. 我能做些什么来解决它? - 我不想以封锁信号开始。

1 个答案:

答案 0 :(得分:1)

因为它是竞争条件。在sig_handler中设置sigintBlocked然后在main函数中进行验证(如果已设置)然后屏蔽信号。

此链接有更多信息 sigprocmask during signal's execution