博客專欄

EEPW首頁 > 博客 > 嵌入式Linux:發(fā)送實(shí)時(shí)信號

嵌入式Linux:發(fā)送實(shí)時(shí)信號

發(fā)布人:美男子玩編程 時(shí)間:2024-10-12 來源:工程師 發(fā)布文章

非實(shí)時(shí)信號有一個(gè)明顯的局限性:當(dāng)同一個(gè)信號多次發(fā)生時(shí),它只會被記錄為一次,且不會記錄發(fā)生的次數(shù)。因此,當(dāng)該信號被解除阻塞后,它僅會被處理一次。這種行為使得標(biāo)準(zhǔn)信號在某些應(yīng)用場景下不夠靈活。

相比之下,實(shí)時(shí)信號 提供了幾個(gè)關(guān)鍵的優(yōu)勢:

  • 信號范圍擴(kuò)大:標(biāo)準(zhǔn)信號僅提供 SIGUSR1 和 SIGUSR2 供用戶自定義使用,而實(shí)時(shí)信號的編號范圍更大(SIGRTMIN 到 SIGRTMAX,對應(yīng)編號范圍 34~64),可以應(yīng)用于更多自定義目的。

  • 隊(duì)列化管理:實(shí)時(shí)信號采取隊(duì)列化管理,這意味著如果同一個(gè)實(shí)時(shí)信號多次發(fā)生,內(nèi)核將記錄每次事件并按順序傳遞,而標(biāo)準(zhǔn)信號只會傳遞一次。

  • 附帶數(shù)據(jù):實(shí)時(shí)信號允許攜帶附帶數(shù)據(jù)(可以是整型數(shù)據(jù)或指針),供接收方在信號處理函數(shù)中使用,這為信號傳遞帶來了更大的靈活性。

  • 傳遞順序保證:當(dāng)多個(gè)不同的實(shí)時(shí)信號處于等待狀態(tài)時(shí),信號編號越小的信號會優(yōu)先傳遞。如果同一個(gè)信號多次發(fā)生,傳遞順序會與發(fā)送順序保持一致。


為了使用實(shí)時(shí)信號,通常需要滿足以下要求:

  • 發(fā)送實(shí)時(shí)信號:發(fā)送進(jìn)程需要使用 sigqueue() 系統(tǒng)調(diào)用發(fā)送實(shí)時(shí)信號及其伴隨數(shù)據(jù)。

  • 接收實(shí)時(shí)信號:接收進(jìn)程需要為該信號設(shè)置信號處理函數(shù),并在 sigaction 函數(shù)中啟用 SA_SIGINFO 標(biāo)志,以確??梢越邮瞻殡S數(shù)據(jù)。


sigqueue() 的函數(shù)原型如下:


#include <signal.h>int sigqueue(pid_t pid, int sig, const union sigval value);


  • 參數(shù):

    • pid:接收信號的進(jìn)程 ID。

    • sig:需要發(fā)送的信號編號。與 kill() 類似,也可以將 sig 設(shè)置為 0 來檢查進(jìn)程是否存在。

    • value:一個(gè) union sigval 類型的值,表示伴隨信號傳遞的數(shù)據(jù),可以是整型或指針。

  • 返回值:

    • 成功返回 0;

    • 失敗返回 -1,并設(shè)置 errno。


union sigval 是一個(gè)共用體,定義如下:


typedef union sigval {    int sival_int;    // 整型數(shù)據(jù)    void *sival_ptr;  // 指針數(shù)據(jù)} sigval_t;

1


發(fā)送進(jìn)程

使用 sigqueue() 向另一個(gè)進(jìn)程發(fā)送實(shí)時(shí)信號及其伴隨數(shù)據(jù)。

#include <stdio.h>#include <stdlib.h>#include <signal.h> int main(int argc, char *argv[]) {    union sigval sig_val;    int pid, sig;        // 檢查參數(shù)個(gè)數(shù)    if (argc < 3) {        exit(-1);    }        // 獲取命令行傳遞的參數(shù)    pid = atoi(argv[1]);    sig = atoi(argv[2]);        printf("Sending signal %d to process %d\n", sig, pid);        // 發(fā)送信號,附帶整型數(shù)據(jù)    sig_val.sival_int = 10;    if (sigqueue(pid, sig, sig_val) == -1) {        perror("sigqueue error");        exit(-1);    }        puts("Signal sent successfully!");    return 0;}

2


接收進(jìn)程

使用 sigaction() 綁定實(shí)時(shí)信號處理函數(shù),并接收伴隨數(shù)據(jù)。


#include <stdio.h>#include <stdlib.h>#include <signal.h>#include <unistd.h> static void sig_handler(int sig, siginfo_t *info, void *context) {    sigval_t sig_val = info->si_value;        printf("Received real-time signal: %d\n", sig);    printf("Attached data: %d\n", sig_val.sival_int);} int main(int argc, char *argv[]) {    struct sigaction sa = {0};    int sig;        // 檢查參數(shù)個(gè)數(shù)    if (argc < 2) {        exit(-1);    }        // 獲取命令行傳遞的信號編號    sig = atoi(argv[1]);        // 綁定信號處理函數(shù)    sa.sa_sigaction = sig_handler;    sa.sa_flags = SA_SIGINFO;  // 啟用 SA_SIGINFO 標(biāo)志,以接收附帶數(shù)據(jù)        if (sigaction(sig, &sa, NULL) == -1) {        perror("sigaction error");        exit(-1);    }        // 無限循環(huán),等待信號    while (1) {        sleep(1);    }        return 0;}


標(biāo)準(zhǔn)信號和實(shí)時(shí)信號在 Linux 信號處理機(jī)制中各有優(yōu)劣。標(biāo)準(zhǔn)信號適用于大多數(shù)常見場景,但其無法記錄信號的多次發(fā)生,且缺少附帶數(shù)據(jù)傳遞的能力。而實(shí)時(shí)信號則提供了更靈活的功能,包括多次傳遞信號、附帶數(shù)據(jù)和保證傳遞順序。這些特性使得實(shí)時(shí)信號在高性能和復(fù)雜信號處理需求下尤為有用。

*博客內(nèi)容為網(wǎng)友個(gè)人發(fā)布,僅代表博主個(gè)人觀點(diǎn),如有侵權(quán)請聯(lián)系工作人員刪除。



關(guān)鍵詞: 嵌入式 Linux

相關(guān)推薦

技術(shù)專區(qū)

關(guān)閉