Stories

Detail Return Return

Android C++系列:Linux信號(二) - Stories Detail

PCB的信號集

如果在進程解除對某信號的阻塞之前這種信號產生過多次,將如何處理?POSIX.1允 許系統遞送該信號一次或多次。Linux是這樣實現的:常規信號在遞達之前產生多次只 計一次,而實時信號在遞達之前產生多次可以依次放在一個隊列裏。本文不討論實時信 號。每個信號只有一個bit的未決標誌,非0即1,不記錄該信號產生了多少 次,阻塞標誌也是這樣表示的。因此,未決和阻塞標誌可以用相同的數據類型sigset_t 來存儲,sigset_t稱為信號集,這個類型可以表示每個信號的“有效”或“無效”狀態, 在阻塞信號集中“有效”和“無效”的含義是該信號是否被阻塞,而在未決信號集中“有 效”和“無效”的含義是該信號是否處於未決狀態。

阻塞信號集也叫做當前進程的信號屏蔽字(Signal Mask),這裏的“屏蔽”應該理解 為阻塞而不是忽略。

sigprocmask

調用函數sigprocmask可以讀取或更改進程的信號屏蔽字。

#include <signal.h>
int sigprocmask(int how, const sigset_t *set, sigset_t *oset); 
返回值:若成功則為0,若出錯則為-1

如果oset是非空指針,則讀取進程的當前信號屏蔽字通過oset參數傳出。如果set是非 空指針,則更改進程的信號屏蔽字,參數how指示如何更改。如果oset和set都是非空指針, 則先將原來的信號屏蔽字備份到oset裏,然後根據set和how參數更改信號屏蔽字。假設當前 的信號屏蔽字為mask,下表説明了how參數的可選值。

how參數的含義

  • SIG_BLOCK set包含了我們希望添加到當前信號屏蔽字的信號,相當於mask=mask|set
  • SIG_UNBLOCK set包含了我們希望從當前信號屏蔽字中解除阻塞的信號,相當於mask=mask&~set
  • SIG_SETMASK設置當前信號屏蔽字為set所指向的值,相當於mask=set

如果調用sigprocmask解除了對當前若干個未決信號的阻塞,則在sigprocmask返回前, 至少將其中一個信號遞達。

sigpending

#include <signal.h>
int sigpending(sigset_t *set);

sigpending讀取當前進程的未決信號集,通過set參數傳出。調用成功則返回0,出錯則 返回-1。

下面用剛學的幾個函數做個實驗。程序如下:

#include <signal.h> 
#include <stdio.h>
void printsigset(const sigset_t *set) {
    int i;
    for (i = 1; i < 32; i++)
        if (sigismember(set, i) == 1) 
            putchar('1');
        else 
        putchar('0');
    puts("");
}
int main(void) {
    sigset_t s, p;
    sigemptyset(&s);
    sigaddset(&s, SIGINT); 
    sigprocmask(SIG_BLOCK, &s, NULL); 
    while (1) {
        sigpending(&p); 
        printsigset(&p); 
        sleep(1);
        }
    return 0; 
}

程序運行時,每秒鐘把各信號的未決狀態打印一遍,由於我們阻塞了SIGINT信號,按Ctrl-C將會使SIGINT信號處於 未決狀態,按Ctrl-\仍然可以終止程序,因為SIGQUIT信號沒有阻塞。

qingkouwei@ubuntu:~$ ./a.out 
0000000000000000000000000000000 
0000000000000000000000000000000(這時按Ctrl-C) 
0100000000000000000000000000000 
0100000000000000000000000000000(這時按Ctrl-\) Quit (core dumped)

信號捕捉設定

image.png

#include <signal.h>
int sigaction(int signum, const struct sigaction *act, struct sigaction *oldact);

struct sigaction 定義: 
struct sigaction {
    void (*sa_handler)(int);
    void (*sa_sigaction)(int, siginfo_t *, void *);
    sigset_t sa_mask;
    int sa_flags; 
    void (*sa_restorer)(void);
};
  • sa_handler : 早期的捕捉函數
  • sa_sigaction : 新添加的捕捉函數,可以傳參 , 和sa_handler互斥,兩者通過sa_flags選擇採用哪種捕捉函數
  • sa_mask : 在執行捕捉函數時,設置阻塞其它信號,sa_mask | 進程阻塞信號集,退出捕捉函數後,還原回原有的 阻塞信號集
  • sa_flags : SA_SIGINFO 或者 0
  • sa_restorer : 保留,已過時

舉例SIGINT被捕捉: 當前進程從內核返回用户空間代碼前檢查是否有信號遞達,有則去響應

利用SIGUSR1和SIGUSR2實現父子進程同步輸出

注意:子進程繼承了父進程的信號屏蔽字和信號處理動作

總結

如果在進程解除對某信號的阻塞之前這種信號產生過多次,將如何處理?POSIX.1允 許系統遞送該信號一次或多次。Linux是這樣實現的:常規信號在遞達之前產生多次只 計一次,而實時信號在遞達之前產生多次可以依次放在一個隊列裏。

user avatar u_15878077 Avatar himeka Avatar chen_christins Avatar veronicaaa Avatar jkkang Avatar djz1234 Avatar artificer Avatar pipigao Avatar huaweichenai Avatar
Favorites 9 users favorite the story!
Favorites

Add a new Comments

Some HTML is okay.