【Linux】進程信號的捕捉處理

在這里插入圖片描述

個人主頁~


進程信號的捕捉處理

  • 一、信號捕捉處理的概述
    • 1、信號捕捉處理全過程
    • 2、用戶態和內核態的區別
      • (一)用戶態
      • (二)內核態
      • (三)用戶態與內核態的切換
      • (四)硬件條件
  • 二、再談進程地址空間
    • 操作系統本質
  • 三、系統調用函數
  • 四、其他補充內容
    • 1、可重入函數
    • 2、volatile關鍵字

一、信號捕捉處理的概述

1、信號捕捉處理全過程

在這里插入圖片描述
如果信號的處理動作是用戶自定義函數,在信號遞達時就調用這個函數,這稱為捕捉信號,這個我們前面說過,但是我們的過程是比較復雜的,首先我們在執行主控制流程的某條指令時因為系統調用等原因會進入內核,然后內核處理完成后發送信號,如果信號的處理動作是自定義的信號處理函數就回到用戶區執行信號處理函數,執行完之后因為信號處理函數的特殊性,它要再次進入內核區,然后回到用戶模式繼續執行

我們在用戶區和內核區來回切換的時候,操作系統負責做我們的身份(用戶身份和內核身份)切換工作,用戶態陷入內核態是通過匯編指令int 80完成的

在進程從內核態返回用戶態時進行信號的檢測和處理

并且main函數和自定義信號捕捉處理函數使用不同的堆棧空間,它們之間不存在調用和被調用的關系

2、用戶態和內核態的區別

用戶態和內核態是操作系統中CPU的兩種運行狀態,它們在訪問權限、資源使用等方面存在顯著差異,以下是對這兩種狀態的標準解釋:

(一)用戶態

用戶態是操作系統為普通用戶程序提供的一種運行模式,在用戶態下運行的程序擁有較低的特權級別,只能訪問受限的系統資源和執行特定的操作,大部分用戶編寫的應用程序(如文本編輯器、瀏覽器等)都是在用戶態下運行的

用戶態程序只能訪問自己的內存空間,不能直接訪問系統的核心資源,如硬件設備、操作系統內核的數據結構等,并且只能執行一部分指令,一些具有高風險或對系統影響較大的指令(如修改系統時鐘、控制硬件中斷等)是被禁止執行的
由于用戶態程序的權限受到限制,即使程序出現錯誤(如內存越界、死循環等),也不會對整個操作系統造成嚴重影響,只會影響到該程序自身

(二)內核態

內核態是操作系統內核運行的模式,具有最高的特權級別,操作系統內核負責管理系統的核心資源,如內存、進程、文件系統、設備驅動等,因此需要在高特權的內核態下運行

內核態程序可以訪問系統的所有資源,包括硬件設備、內核數據結構、所有進程的內存空間等,并且可以執行所有的 CPU 指令,包括那些在用戶態下被禁止執行的特權指令,這些特權指令可以用于實現系統的關鍵功能,如進程調度、內存管理、中斷處理等
由于內核態程序具有最高的權限,一旦內核態程序出現錯誤,可能會導致整個操作系統崩潰或出現嚴重的系統故障

(三)用戶態與內核態的切換

在操作系統的運行過程中,程序需要在用戶態和內核態之間進行切換,以完成不同的任務,常見的切換場景包括:

  • 系統調用:當用戶態程序需要訪問系統資源或執行特權操作時,會通過系統調用(如 openreadwrite 等)請求操作系統內核的服務,在執行系統調用時,程序會從用戶態切換到內核態,由操作系統內核來處理請求,處理完成后再切換回用戶態
  • 中斷處理:當系統發生硬件中斷(如時鐘中斷、鍵盤中斷等)或軟件中斷(如異常、陷阱等)時,CPU 會自動從用戶態切換到內核態,由操作系統內核來處理中斷事件,處理完成后,根據情況決定是否切換回用戶態

但是操作系統是不相信用戶的,所以用戶態和內核態進行切換的時候是操作系統完成的

(四)硬件條件

在 CPU 中有一個ecs寄存器,它的后兩個bit位就標記了當前是處于用戶態還是內核態,其中 00 表示處于內核態,11 表示處于用戶態,int 80 指令本質上就是將 11 修改成 00

二、再談進程地址空間

在這里插入圖片描述
我們再來拿出我們的老圖,我們知道,每個進程都有一個PCB,然后PCB中有一個結構體叫做mm_struct,又被稱為進程地址空間,就是我們常說的虛擬地址,虛擬地址通過頁表和MMU映射到物理內存上,可以說頁表就是虛擬內存和物理內存之間的橋梁,每個進程都會有一個頁表,但是,這是對于用戶區空間來說的,我們的內核空間,實際上也是有一個內核頁表,這個內核頁表是唯一的,內核空間中的數據也是唯一的,我們不同的進程,它們用戶區空間的代碼和數據不同,但是內核空間的數據一定相同,所有進程共享一份內核空間以及一份內核頁表

對于進程來說,去調用系統調用接口,就是在我自己的地址空間中執行的,對于操作系統來說,任何一個時刻都有進程執行,我們想執行操作系統的代碼就可以隨時執行

操作系統本質

操作系統的本質就是一個基于時鐘中斷的一個死循環,這里的時鐘中斷與STM32中的《TIM定時器》(學習STM32的可以點開看一下,沒有學習就算了,不影響理解)類似,都是每隔很短的時間向計算機發送時鐘中斷,操作系統在收到時鐘中斷后,就去中斷向量表中執行相應的進程調度之類的方法

三、系統調用函數

sigaction用于設置或檢查信號處理行為的系統調用函數

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

返回值:成功返回0,失敗返回-1
signo:需要處理的信號編號
act:輸入型參數,指向struct sigaction結構體,用于指定新的信號處理動作
oact:輸出型參數,指向struct sigaction結構體,用于保存原來的信號處理動作

其中結構體sigaction是這樣的

struct sigaction {void   (*sa_handler)(int); // 信號處理函數指針,或SIG_IGN(忽略信號)、SIG_DFL(使用默認處理)void   (*sa_sigaction)(int, siginfo_t *, void *); //另一種信號處理函數指針,用于處理帶附加信息的信號sigset_t sa_mask;           //在信號處理函數執行期間需要阻塞的信號集int      sa_flags;          //控制信號處理行為的標志位void   (*sa_restorer)(void); //已棄用,通常設為 NULL
};

其中我們主要研究兩個成員:sa_handlersa_mask

這里我們說一個結論,就是信號處理函數在處理信號時,是不再接受新的信號的,在該信號處理函數被調用時,在剛要調用的某個時間,內核會自動將當前信號加入到進程的信號屏蔽字中,這就可以實現在處理某個信號的過程中,不會被這個信號反復打擾,如果這個信號再次產生,它會被阻塞到當前函數處理結束,如果在當前信號被阻塞以外,還想要屏蔽其他的一些信號,則動用sa_mask說明要額外屏蔽的信號

#include <iostream>
#include <signal.h>
#include <cstring>
#include <unistd.h>using namespace std;void PrintPending()
{sigset_t set;//將當前被阻塞且處于未決狀態的信號集存儲到set信號集中(輸出型參數)sigpending(&set);//打印set位圖(即pending位圖)for (int signo = 31; signo >= 1; signo--){if (sigismember(&set, signo))cout << "1";elsecout << "0";}cout << endl;
}void handler(int signo)
{//收到2號信號打印一句收到了,然后循環調用PrintPendingcout << "catch a signal, signal number : " << signo << endl;while (true){PrintPending();sleep(1);}
}int main()
{//創建并初始化act和oactstruct sigaction act, oact;memset(&act, 0, sizeof(act));memset(&oact, 0, sizeof(oact));//設置sa_mask為全0sigemptyset(&act.sa_mask);//設置信號屏蔽sigaddset(&act.sa_mask, 1);sigaddset(&act.sa_mask, 3);sigaddset(&act.sa_mask, 4);//設置信號處理函數為handleract.sa_handler = handler; // SIG_IGN SIG_DFL//需要處理的就是2號信號sigaction(2, &act, &oact);while (true){cout << "I am a process: " << getpid() << endl;sleep(1);}return 0;
}

在這里插入圖片描述
在這里插入圖片描述
正常情況下,我們沒有發送任何信號,1號信號會將進程終止,當我們發送2號信號,sigaction函數將信號捕捉后,我們進入到handler函數,開始打印全0的pending位圖,此時屬于信號處理函數處理信號的過程,然后我們發送1234號信號,因為134號信號我們提前設置進sa_mask當中了,所以我們在發送信號的時候對應的比特位就會由0置1,表示阻塞通過

四、其他補充內容

1、可重入函數

在這里插入圖片描述
這是一個正常的單鏈表,node1node2是兩個待插入節點,其中,我們要頭插node1,其方法就是將next指針指向現在的頭節點,再將自己賦值給頭節點

insert(struct Node* node)
{node->next = head;head = node;
}

但是在node1->next = head;執行完畢后,還沒來得及執行head = p;突然來了一個信號,這個信號剛好被捕捉了,執行自定義動作,剛好自定義動作是將node2頭插

void handler(int signo)
{insert(&node2);
}
insert(struct Node* node)
{node->next = head;head = node;
}

執行完這個信號處理函數后再返回執行main函數中的代碼
這樣一來我們實際上的代碼就變成了

node1->next = head;
node2->next = head;
head = &node2;
head = &node1;

在這里插入圖片描述
這樣上面node1是頭插進去了,但是node2是不可能通過單鏈表的接口訪問到了,這個節點就丟失了,這樣就會導致內存泄漏

如果一個函數像上面一樣被重復進入的情況下,出錯或者可能出錯,就叫做不可重入函數,否則就叫做可重入函數,我們學習到的大部分函數都是不可重入函數,因為只要是涉及指針改指向的問題,基本上都是不可重入的

2、volatile關鍵字

volatile是一個類型修飾符,用于告知編譯器它修飾的內容拒絕優化

該程序在收到2號信號之前一直在while循環中啥也不干,收到2號信號執行handler打印,設置flag為1,然后繼續while判斷,從這里開始就變得不一樣了
在這里插入圖片描述
我們可以看到我們編譯的方法加了-O1選項,這個叫做優化,一共有-O0 -O1 -O2 -O3四種種優化等級,其中O0是無優化
在這里插入圖片描述

這是加了volatile的關鍵詞產生的結果,按下ctrl+c打印執行后,flag置1,while判斷繼續向后執行,打印,程序結束
在這里插入圖片描述
這是不加volatile的關鍵詞產生的結果,按下ctrl+c打印執行后,flag置1,怎么不往下執行了呢,我再次按下ctrl+c打印再次執行
在這里插入圖片描述
這里就是優化的問題,優化其實是CPU管理資源的一種方式,優化后,CPU在第一次讀取flag的時候,將其加載到CPU寄存器中,handler結束后進行while的判斷時,直接判斷CPU上的這個flag,不再去物理內存當中尋找,因為這樣的尋找是耗費資源的,加了volatile關鍵字后,不再產生優化,在判斷時判斷的就是物理內存中的flag

所以,涉及到要隨時了解某些變量的狀態的時候,這些變量在優化的時候最好用volatile修飾一下


今日分享就到這了~

在這里插入圖片描述

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/900008.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/900008.shtml
英文地址,請注明出處:http://en.pswp.cn/news/900008.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

Nyquist內置函數-概述

1 Nyquist內置函數-概述 本章提供奈奎斯特&#xff08;Nyquist&#xff09;語言參考。操作按功能和抽象級別分類。奈奎斯特在兩個重要級別上實現&#xff1a;“高級”級別支持行為抽象&#xff0c;這意味著像 stretch 和 at 這樣的操作可以應用。這些函數是典型用戶期望使用的…

數據驅動防災:AI 大模型在地質災害應急決策中的關鍵作用。基于DeepSeek/ChatGPT的AI智能體開發

全球氣候變化加劇了滑坡、泥石流等地質災害的發生頻率與不確定性&#xff0c;傳統基于統計與物理模型的預測方法常受限于?數據稀疏性?與?動態耦合復雜性?。近年來&#xff0c;AI智能體&#xff08;AI Agents&#xff09;與大型語言模型&#xff08;LLMs&#xff09;的突破為…

光譜相機在工業中的應用

光譜相機&#xff08;多光譜、高光譜、超光譜成像技術&#xff09;在工業領域通過捕捉物質的光譜特征&#xff08;反射、透射、輻射等&#xff09;&#xff0c;結合化學計量學與人工智能算法&#xff0c;為工業檢測、質量控制和工藝優化提供高精度、非接觸式的解決方案。以下是…

Dify工作流中如何去除deepseek-r1思考內容

在工作流中deepseek-r1的think標簽內部的內容&#xff0c;很容易讓工作流其他的llm產生幻覺&#xff0c;導致不能良好的生成目標效果。 我們通過代碼的方式讓deepseek-r1既有think思考鏈的效果&#xff0c;又不傳遞思考鏈。 工作流的邏輯為上圖 去除think中的代碼為 import re…

容器的CPU

1、限制進程的CPU 通過Cgroup來限制進程資源的使用&#xff0c;CPU Cgroup 是 Cgroups 其中的一個 Cgroups 子系統&#xff0c;它是用來限制進程的 CPU 使用的。 cpu.cfs_period_us&#xff0c;它是 CFS 算法的一個調度周期&#xff0c;一般它的值是 100000&#xff0c;以 mic…

【系統分析師-第二篇】

學習目標 通過參加考試&#xff0c;訓練學習能力&#xff0c;而非單純以拿證為目的。 1.在復習過程中&#xff0c;訓練快速閱讀能力、掌握三遍讀書法、運用番茄工作法。 2.從底層邏輯角度理解知識點&#xff0c;避免死記硬背。 3.通過考試驗證學習效果。 學習方法 第二遍快速…

【再探圖論】深入理解圖論經典算法

一、bellman_ford 1. 是什么松弛 在《算法四》中&#xff0c;對松弛的解釋是&#xff1a;relax the edge&#xff0c;看起來比較抽象&#xff0c;不過如果我們從生活中的實例去理解&#xff0c;就簡單多了&#xff1a; 試想一根繩索&#xff0c;當你握著繩索的兩頭使勁用力拉…

基于pycharm的YOLOv11模型訓練方法

提示&#xff1a;文章寫完后&#xff0c;目錄可以自動生成&#xff0c;如何生成可參考右邊的幫助文檔 文章目錄 前言一、前期準備1.1 軟件環境配置1.2 訓練集參考 二、訓練步驟2.1 打開文件夾2.2 打開文件2.3 data.yaml最終代碼 三、train.py四、最終結果五、detect.py六、 拓展…

用nodejs連接mongodb數據庫對標題和內容的全文本搜索,mogogdb對文檔的全文本索引的設置以及用node-rs/jieba對標題和內容的分詞

//首先我們要在Nodejs中安裝 我們的分詞庫node-rs/jieba,這個分詞不像jieba安裝時會踩非常多的雷&#xff0c;而且一半的機率都是安裝失敗&#xff0c;node-rs/jieba比jieba庫要快20-30%&#xff1b;安裝分詞庫是為了更好達到搜索的效果 這個庫直接npm install node-rs/jieba即…

水下聲吶探測儀,應急救援中的高效水下定位技術|深圳鼎躍

近年來&#xff0c;隨著水域活動增多及自然災害頻發&#xff0c;水下救援需求日益增長。傳統人工打撈方法在復雜水域中效率低、風險高&#xff0c;尤其在能見度差、水流湍急或深水區域中&#xff0c;救援難度倍增。 在此背景下&#xff0c;水下聲吶探測儀憑借其聲波定位與視頻…

AI 網關代理 LLMs 最佳實踐

作者&#xff1a;付宇軒&#xff08;計緣&#xff09; DeepSeek/QWen 普惠 AI 趨勢 隨著 DeepSeek-R1 的橫空出世&#xff0c;又一次點燃了原本已經有點冷淡的大語言模型市場和話題&#xff0c;并且快速成為了現象級&#xff0c;小到中小學生&#xff0c;大到父母輩都知道了中…

策略模式實際用處,改吧改吧直接用,兩種方式

controller RestController RequestMapping("admin/test") RequiredArgsConstructor(onConstructor __(Autowired)) public class TestController {Autowiredprivate VideoFactory VideoFactory;GetMapping("getList")public R getList(){// 第一種方式T…

chromium魔改——修改 navigator.webdriver 檢測

chromium源碼官網 https://source.chromium.org/chromium/chromium/src 說下修改的chromium源碼思路&#xff1a; 首先在修改源碼過檢測之前&#xff0c;我們要知道它是怎么檢測的&#xff0c;找到他通過哪個JS的API來做的檢測&#xff0c;只有知道了如何檢測&#xff0c;我們…

Muduo網絡庫實現 [九] - EventLoopThread模塊

目錄 設計思路 類的設計 模塊的實現 私有接口 公有接口 設計思路 我們說過一個EventLoop要綁定一個線程&#xff0c;未來該EventLoop所管理的所有的連接的操作都需要在這個EventLoop綁定的線程中進行&#xff0c;所以我們該如何實現將EventLoop和線程綁定呢&#xff1f;…

UE5學習筆記 FPS游戲制作38 繼承標準UI

文章目錄 UE的UIUMG的繼承繼承標準控件創建標準控件繼承標準控件的用處 UE的UI 和Untiy有onGui和UGui類似&#xff0c;UE有slateUI和UMG,slateUI是早期只能用C編寫的UI&#xff0c;UMG是現在使用的&#xff0c;可以拖拽編輯的UI slateUI是UMG的父類 UMG的繼承 我們編寫一個控…

C#核心學習(七)面向對象--封裝(6)C#中的拓展方法與運算符重載: 讓代碼更“聰明”的魔法

目錄 一、什么是拓展方法&#xff1f; 二、拓展方法有啥用&#xff1f;怎么寫拓展方法&#xff1f; 1. ?核心用途 2. ?編寫步驟 實現步驟 關鍵點說明 關鍵規則 3. ?注意事項 三、什么是運算符重載&#xff1f; 四、運算符重載有啥用&#xff1f;怎么寫&#xff1f;…

銀行卡歸屬地查詢API接口如何對接?

銀行卡歸屬地查詢 API 接口是一種能讓開發者通過編程方式獲取銀行卡歸屬地等相關信息的工具。借助此接口&#xff0c;開發者可將銀行卡歸屬地查詢功能集成到自己的應用程序或系統里&#xff0c;像電商平臺、第三方支付公司等都能運用它來提升業務的準確性與安全性。 銀行卡歸屬…

ORM mybits mybits-plus

ORM ORM 即對象關系映射&#xff08;Object Relational Mapping&#xff09;&#xff0c;是一種程序設計技術&#xff0c;用于實現面向對象編程語言里不同類型系統的數據之間的轉換。下面從基本概念、工作原理、優勢與劣勢、常見的 ORM 框架等方面詳細介紹 ORM。 常見的orm框架…

網絡編程—網絡概念

目錄 1 網絡分類 1.1 局域網 1.2 廣域網 2 常見網絡概念 2.1 交換機 2.2 路由器 2.3 集線器 2.4 IP地址 2.5 端口號 2.6 協議 3 網絡協議模型 3.1 OSI七層模型 3.2 TCP/IP五層模型 3.3 每層中常見的協議和作用 3.3.1 應用層 3.3.2 傳輸層 3.3.3 網絡層 3.3.4…

4月3日工作日志

一個樸實無華的目錄 今日學習內容&#xff1a;1.關系數據庫 今日學習內容&#xff1a; 1.關系數據庫