国产精品电影_久久视频免费_欧美日韩国产激情_成年人视频免费在线播放_日本久久亚洲电影_久久都是精品_66av99_九色精品美女在线_蜜臀a∨国产成人精品_冲田杏梨av在线_欧美精品在线一区二区三区_麻豆mv在线看

OS內核的信號機制:所有的異步都可以是同步的

系統 其他OS
每個進程,在OS內核里都被一個task結構體表示,這個結構體的其中一個成員變量就是記錄信號的:我們給他起名叫sigmap,Linux的不一定要叫這個名字,但肯定有這一項。

今天條友@xiamenuser給我提了一個關于操作系統的問題:怎么把定時器線程里的回調函數,(在定時器觸發之后)挪到工作線程里運行?

這個需求要做的事,跟Linux內核的信號機制是一樣的。

OS內核的信號機制,在1970年的Unix時代就有了,是一個上古話題。

在unix里,可以使用kill -9 pid命令殺掉進程(pid為進程號),在Linux里也可以。

1.OS內核的信號

有個專有的宏定義#define SIGKILL 9,然后信號9就成了一個特別牛的信號,大概除了0號idle進程和1號init進程之外,其他進程都可以殺死。

0號進程和1號進程是不能殺死的,否則系統就崩潰了!

int sys_kill(int sig, int pid)
{
if (sig < 0 || pid < 0)
return -EINVAL;
if (0 == pid || 1 == pid) {
if (SIGKILL == sig) return -1;
}
tasks[pid]->sigmap |= 1 << sig;
return 0;
}

OS內核里對應著kill命令的sys_kill()系統調用,大概是上面這樣:

在進程的task結構體的sigmap成員變量上,設置1個標志位,進程就可以收到信號了。

每個進程,在OS內核里都被一個task結構體表示,這個結構體的其中一個成員變量就是記錄信號的:我們給他起名叫sigmap,Linux的不一定要叫這個名字,但肯定有這一項。

這個信號在什么時候處理呢?

等到收信號的進程下一次被調度運行的時候。

當前運行的進程,肯定是發信號的進程,否則它沒法主動發起kill()系統調用。

發信號的進程做的事,只是把信號設置到接收進程的信號圖上,這時信號實際上已經發到了:但是接收進程并不會馬上因為SIGKILL信號而被殺死。

SIGKILL信號的殺進程,實際上進程是自殺的!

當收到信號的進程再次被調度運行的時候,操作系統會讓它先執行信號的處理函數,而SIGKILL的處理函數,就是exit()系統調用:進程退出。

這個過程可以是異步的,等到接收進程下一次被調度時再處理,至于什么時候輪到它:等吧。

也可以讓它馬上同步處理,只需要在sys_kill()函數的末尾加一行代碼就行:

shedule_task( tasks[pid] );

直接選擇接收進程是下一個要調度的進程,并且馬上調度它運行:接下來它就完事了。

不需要等OS內核統計時間片,確定調度的優先級了,既然用戶想讓它掛掉,OS當然要馬上讓它掛掉。

畢竟Linux系統也惹不起用戶啊,用戶是可以重裝windows的?

接下來,說說shedule_task()之后的細節。

2.信號是怎么處理的

每個信號都有一個處理函數,叫信號處理函數。

信號處理函數,是在用戶態的代碼里運行的。

所以,程序員可以自己給部分信號編寫處理函數,用signal()系統調用注冊到OS內核,就可以(在收到信號時)運行這個自己編寫的函數了。

如果信號處理函數是在內核狀態運行的,那顯然用戶編寫的函數是沒法運行的,因為用戶函數的內存地址在用戶空間(它在進程的代碼段里)。

OS內核在信號處理時要做的是,把進程從內核返回后要運行的代碼地址,改成信號處理函數的地址。

修改過程如下:

系統內核的信號處理過程

1)進程從內核返回時的狀態,如上圖。

內核棧上的寄存器排布順序不一定是對的,這要查intel的手冊,但是這些項肯定都有。

在進程使用iret指令(中斷返回)從內核返回的那一刻,內核棧上的這些數據都要彈出到對應的寄存器。

然后,進程就會運行EIP指向的用戶代碼,同時用戶態的棧頂就是ESP。

EIP和ESP指向的內容到底是什么,內核不需要管:這是由程序員寫代碼時確定的。

進程從內核返回之后的錯誤,錯的是程序員,不是系統內核。

但要是返不回來,或者不能處理信號,錯的就是系統內核了。

2)OS內核要做的是,修改內核棧上、保存的、用戶態的、EIP和ESP(注意這3個定語):

A,讓EIP指向信號處理函數,

B,讓ESP指向信號處理函數的參數,

C,在信號處理函數的下方,放上“真正的”返回地址,

D,在信號處理函數運行完之后,丟掉(信號處理函數的)參數,彈出真正的返回地址:讓程序恢復正常的狀態,繼續運行。

如上圖中的綠字部分。

如果一次要處理多個信號的話,就順著用戶棧繼續疊加就行。

siska內核demo里的信號處理代碼,如下的3張圖:

因為信號處理函數有參數,而參數要壓在用戶態的棧上,所以信號處理函數運行完之后還要清理它。

所以,與一般的C函數不同,信號處理函數是被調函數清理堆棧的:即它是pascal調用,而不是C調用!

C調用,都是主調函數清理堆棧的。

所以,信號處理函數的總入口是一段匯編代碼,用來在C語言里完成這個pascal調用。

這么看來,pascal這種老語言,也不是想象的那么差?

這個信號處理方式,是我給出來的解決方案?

至于Linux是不是也這么做的,我就不知道了。

但是,這么做是可行的。

siska信號處理,pascal調用的匯編

上圖95行的call *(%eax),就是調用信號處理的函數指針。

它前后的匯編代碼,都是準備參數和清理堆棧。

3.回到開頭的問題

怎么讓定時器線程在觸發之后,讓回調函數在工作線程里運行?

回調函數一般有一個參數,表示回調上下文,但沒有返回值。

因為定時器的添加和處理在2個線程里,回調函數的返回值沒有意義。

如果回調函數的處理出錯了,就在上下文里設置錯誤碼作為提示。

所以,它的函數聲明是這樣的:void callback(void* ctx);

要讓它正常運行,必須把回調上下文的指針添加到工作線程的用戶棧上,同時讓工作線程的內核棧上保存的EIP指向回調函數。

這個處理方式,與OS內核的信號處理方式是一樣的。

信號處理函數的聲明:void sighandler(int sig); 也是一個參數、無返回值。

在定時器觸發之后,定時器線程可以發起一個系統調用,把這些信息給到內核,然后內核修改工作線程的數據,讓定時器的回調處理“像個信號”一樣就可以了?

這個系統調用如果Linux沒有提供的話,就只能自己修改Linux內核代碼,或者給Linus大牛提個需求了(他有可能看不過來你的郵件)。

PS:

工作線程和定時器線程在同一個進程里,所以它們的用戶態內存的代碼段、數據段、堆都是共享的,只是內核棧和用戶棧不一樣。

內核棧:在內核看來,每個線程也是一個可調度的進程,它必須有自己的內核棧和頁表。

同一個進程的不同線程之間共享內存,靠的是頁表的映射:把它們映射到同一個物理內存頁上。

用戶棧:不同的線程可以并發運行,它們的用戶棧肯定是不同的,否則局部變量就互相覆蓋了:這肯定是不可能的。

siska里信號處理的代碼,如下:

siska信號處理,1

siska信號處理,2

責任編輯:武曉燕 來源: 今日頭條
相關推薦

2023-04-03 09:44:18

2025-02-03 07:00:00

2011-07-05 18:40:19

QT 信號 機制

2011-07-05 18:32:52

QT 信號 機制

2024-07-16 10:52:09

2009-07-21 09:31:00

Scala操作符

2025-04-29 06:53:36

2021-12-10 00:01:53

Vsync信號機制

2017-05-15 11:39:36

2018-02-25 09:00:00

LinuxQ4OS開源

2011-10-11 16:57:13

windows8TechEd 2011

2022-03-31 08:15:59

遞歸代碼非遞歸

2009-11-12 09:51:11

Visual C++項

2014-09-24 13:04:13

微信企業號

2022-03-11 11:40:26

AI數據技術

2013-04-08 12:41:35

JavaScriptJS

2011-06-17 14:36:50

Linux

2023-03-20 18:34:02

營銷大促質量保障穩定性

2022-03-07 10:27:03

Linux開源社區

2017-09-14 09:40:32

PythonUbuntu信號機制
點贊
收藏

51CTO技術棧公眾號

日本一区不卡| 久久久性生活视频| 色94色欧美sute亚洲线路一ni| 91中文字精品一区二区| 精品成人av| 亚洲欧洲综合另类| 裸体裸乳免费看| 亚洲一区二区三区| 最近2019中文字幕一页二页| av在线资源网| 999久久久亚洲| 亚洲午夜精品久久久久久久久久久久| 日本不卡二区| 久久99高清| 日韩视频亚洲视频| 伊人精品影院| 日本vs亚洲vs韩国一区三区 | 成人夜色视频网站在线观看| 亚洲最大福利网站| 视频精品在线观看| 黄色精品在线观看| 成人高清视频在线| 亚洲国产欧洲综合997久久 | 中文产幕区在线观看| 91麻豆成人久久精品二区三区| 日韩成人av电影在线| 福利在线午夜| 国产欧美日韩一区二区三区在线| 欧美精品一二三区| 亚洲精品一区二| 亚洲国产国产亚洲一二三| 国产精品都在这里| 欧美美乳视频| 欧洲日本亚洲国产区| 91精品一区| 久久精品最新地址| 亚洲欧洲日韩精品在线| 色偷偷av一区二区三区乱| 欧美色999| 中文字幕亚洲一区二区三区五十路 | 久久久久无码国产精品一区| 国产精品99在线观看| 国产精品亚洲一区二区三区| 成人激情诱惑| 成人福利在线视频| 国内在线视频| 91久色porny| 久色视频在线播放| 91玉足脚交白嫩脚丫在线播放| 日本视频久久久| swag国产精品一区二区| 欧美激情第一页xxx| 在线中文字幕视频| 香蕉成人伊视频在线观看| 黄页网站视频在线观看| 精品国产91久久久| 黄色大片在线免费观看| 91麻豆精品91久久久久同性| 欧美精品久久久久久久自慰| 成人亚洲一区二区一| 成人久久久久久久久| 中文字幕免费在线观看视频一区| 天堂在线亚洲| 极品少妇一区二区三区精品视频| 国外成人性视频| 亚洲免费成人av在线| 精品国产乱码久久久久久夜甘婷婷| 欧美xxxxhdvideosex| 亚洲日韩欧美一区二区在线| a视频免费看| 欧美视频在线免费看| 国产鲁鲁视频在线观看特色| 亚洲另类xxxx| 都市激情亚洲欧美| 91香蕉视频在线下载| 久久午夜精品| 欧洲亚洲免费在线| 欧美激情视频一区二区三区在线播放 | 日精品一区二区| 五月天色婷婷综合| 久久久精品免费网站| julia中文字幕久久亚洲蜜臀| 欧美日韩中字一区| 无码粉嫩虎白一线天在线观看| 91视频91自| 最新亚洲伊人网| 欧美一卡二卡在线观看| 日韩福利在线观看| 日韩一区二区在线观看视频播放| 一个人www视频在线免费观看| www.日韩免费| 91精品动漫在线观看| 欧美老女人xx| 888久久久| 蜜桃视频成人在线观看| 亚洲人成人一区二区在线观看 | 九九久久综合网站| 欧美一区精品| 阿v天堂2018| 麻豆精品精品国产自在97香蕉| 成人亲热视频网站| 久久精品二区亚洲w码| 91午夜国产| 精品一区二区三区四区在线| 中文精品一区二区| 91制片厂免费观看| 亚洲成人综合视频| 素人一区二区三区| 高清av免费一区中文字幕| 成人一区不卡| 大陆极品少妇内射aaaaaa| 亚洲大尺度视频在线观看| 欧美色999| 久久综合九九| 午夜亚洲视频| 乱小说综合网站| 久久精品男人天堂av| www黄色日本| 欧美片网站yy| 美女久久99| 日本精品一区在线观看| 日韩欧美一区二区在线视频| 欧美另类69xxxxx| 国产午夜伦鲁鲁| 日韩久久免费av| 亚洲国产精品综合久久久| aaa毛片在线观看| 亚洲国产成人精品久久| 国模一区二区三区| 色婷婷综合缴情免费观看| 久久综合五月天| 国产综合色产在线精品| 国产成人艳妇aa视频在线| 欧美午夜免费电影| 精品国产一区二区三区四区| 女人和拘做爰正片视频| 亚洲精品久久久久久下一站 | 亚洲国产一区二区三区在线观看| 在线一区免费| 久久久久久77777| 久久69精品久久久久久国产越南| 激情六月婷婷久久| 欧美xxxx免费虐| 精品欧美国产| 在线免费观看成人短视频| 清纯唯美日韩| 在线观看成年人视频| 777777777亚洲妇女| 欧美激情在线一区二区三区| 91麻豆精品| 成人一对一视频| 综合久久久久久久| 四虎国产精品成人免费影视| 日韩人妻精品一区二区三区| 亚洲国产黄色片| 久久国产精品99精品国产 | 国产视频久久网| 免费成人美女在线观看| 拔插拔插海外华人免费| 日韩av在线网址| 日本成人超碰在线观看| 国产经典三级在线| 国产成人综合自拍| 蜜臀a∨国产成人精品| 在线91av| 亚洲一区二区三区视频| 日韩高清在线不卡| 影音先锋男人在线资源| 欧美日韩国产综合在线| 7777精品伊人久久久大香线蕉完整版 | 日韩黄色影视| 亚洲精品xxxx| 成人国产亚洲欧美成人综合网| 影视一区二区三区| 青青青国产在线视频| 国产99视频精品免视看7| 偷拍一区二区三区| 亚洲小说区图片区| heyzo高清国产精品| 青草成人免费视频| 一二三区精品福利视频| 无需播放器亚洲| 欧美日韩在线视频免费观看| 久久久久99精品成人片| 久久久欧美一区二区| 婷婷综合另类小说色区| 午夜在线视频观看日韩17c| 惠美惠精品网| jizzjizzjizz亚洲女| 亚洲最大成人免费视频| 日韩欧美亚洲国产另类| 成人久久18免费网站麻豆| 欧美成人一区在线观看| 国产高清视频在线| 国产欧美自拍视频| 久久久久久久电影一区| 在线观看91视频| 国产凹凸在线观看一区二区| 美女精品一区最新中文字幕一区二区三区| 北条麻妃在线|