Установка маски с помощью sigprocmask работает только в зависимости от того, где она вызывается

У меня странное поведение, когда manpage и google не помогли.

В моем коде я хочу заблокировать / разблокировать SIGINT при отправке SIGUSR2. Для этого я устанавливаю обработчик сигнала и подготавливаю набор для маски в функции:

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 ответ

Решение

Потому что это состояние гонки. установите sigintBlocked в sig_handler и затем выполните проверку в основной функции, если она установлена, затем замаскируйте сигнал.

эта ссылка имеет больше информации sigprocmask во время выполнения сигнала

Другие вопросы по тегам