徹底學會使用epoll(一)——ET模式實現分析

注:之前寫過兩篇關于epoll實現的文章,但是感覺懂得了實現原理并不一定會使用,所以又決定寫這一系列文章,希望能夠對epoll有比較清楚的認識。是請大家轉載務必注明出處,算是對我勞動成果的一點點尊重吧。另外,文中如果有不全面或者不正確的地方還請大家指出。也可以私信或者發郵件:lvyilong316@163.com

1.?ET模式實現分析

1.1?ET和LT的實現區別

????首先給出下面一張圖,這張圖是從我之前的一篇博文——epoll實現分析中摘取并細化的。這張圖對理解ET模式已經epoll的工作過程只管重要,當然我自己總結出來后也感覺有的小成就,在這里與大家分享。

注:上圖的poll不要理解成和select相似那個poll,這是通過epoll_ctl調用的。

下面簡要分析一下epoll的工作過程:

(1)?epoll_wait調用ep_poll,當rdlist為空(無就緒fd)時掛起當前進程,知道rdlist不空時進程才被喚醒。

(2)?文件fd狀態改變(buffer由不可讀變為可讀或由不可寫變為可寫),導致相應fd上的回調函數ep_poll_callback()被調用。

(3)?ep_poll_callback將相應fd對應epitem加入rdlist,導致rdlist不空,進程被喚醒,epoll_wait得以繼續執行。

(4)?ep_events_transfer函數將rdlist中的epitem拷貝到txlist中,并將rdlist清空。

(5)?ep_send_events函數(很關鍵),它掃描txlist中的每個epitem,調用其關聯fd對用的poll方法(圖中藍線)。此時對poll的調用僅僅是取得fd上較新的events(防止之前events被更新),之后將取得的events和相應的fd發送到用戶空間(封裝在struct?epoll_event,從epoll_wait返回)。之后如果這個epitem對應的fd是LT模式監聽且取得的events是用戶所關心的,則將其重新加入回rdlist(圖中藍線),否則(ET模式)不在加入rdlist。

具體代碼:

/*?掃描整個txlist鏈表...?*/

for?(eventcnt?=?0,?uevent?=?esed->events;

?????!list_empty(head)?&&?eventcnt?<?esed->maxevents;)?{

/*?取出第一個成員?*/

epi?=?list_first_entry(head,?struct?epitem,?rdllink);

/*?然后從鏈表里面移除?*/

list_del_init(&epi->rdllink);

/*?讀取events,?

?*?注意events我們ep_poll_callback()里面已經取過一次了,?為啥還要再取?

?*?1.?我們當然希望能拿到此刻的最新數據,?events是會變的~

?*?2.?不是所有的poll實現,?都通過等待隊列傳遞了events,?有可能某些驅動壓根沒傳

?*?必須主動去讀取.?*/

revents?=?epi->ffd.file->f_op->poll(epi->ffd.file,?NULL)?&

epi->event.events;

?

if?(revents)?{

/*?將當前的事件和用戶傳入的數據都copy給用戶空間,

?*?就是epoll_wait()后應用程序能讀到的那一堆數據.?*/

if?(__put_user(revents,?&uevent->events)?||

????__put_user(epi->event.data,?&uevent->data))?{

/*?如果copy過程中發生錯誤,?會中斷鏈表的掃描,

?*?并把當前發生錯誤的epitem重新插入到ready?list.

?*?剩下的沒處理的epitem也不會丟棄,?在ep_scan_ready_list()

?*?中它們也會被重新插入到ready?list?*/

list_add(&epi->rdllink,?head);

return?eventcnt???eventcnt?:?-EFAULT;

}

eventcnt++;

uevent++;

if?(epi->event.events?&?EPOLLONESHOT)

epi->event.events?&=?EP_PRIVATE_BITS;

else?if?(!(epi->event.events?&?EPOLLET))?{

/*

?*?If?this?file?has?been?added?with?Level

?*?Trigger?mode,?we?need?to?insert?back?inside

?*?the?ready?list,?so?that?the?next?call?to

?*?epoll_wait()?will?check?again?the?events

?*?availability.?At?this?point,?noone?can?insert

?*?into?ep->rdllist?besides?us.?The?epoll_ctl()

?*?callers?are?locked?out?by

?*?ep_scan_ready_list()?holding?"mtx"?and?the

?*?poll?callback?will?queue?them?in?ep->ovflist.

?*/

/*?嘿嘿,?EPOLLET和非ET的區別就在這一步之差呀~

?*?如果是ET,?epitem是不會再進入到readly?list,

?*?除非fd再次發生了狀態改變,?ep_poll_callback被調用.

?*?如果是非ET,?不管你還有沒有有效的事件或者數據,

?*?都會被重新插入到ready?list,?再下一次epoll_wait

?*?時,?會立即返回,?并通知給用戶空間.?當然如果這個

?*?被監聽的fds確實沒事件也沒數據了,?epoll_wait會返回一個0,

?*?空轉一次.

?*/

list_add_tail(&epi->rdllink,?&ep->rdllist);

}

}

}

說明:

l?epoll_wait返回的條件是rdlist不空,而使rdlist不空的途徑有兩個,分別對應圖中的紅線和藍線。

l?ET和LT模式下的epitem都可以通過紅線方式加入rdlist從而喚醒epoll_wait,但LT模式下的epitem還可以通過藍線方式重新加入rdlist喚醒epoll_wait。所以ET模式下,fd就緒(通過紅線加入rdlist)只會被通知一次,而LT模式下只要滿足相應讀寫條件就返回就緒(通過藍線加入rdlist)。

l?ET事件發生僅通知一次的原因是只被添加到rdlist中一次,而LT可以有多次添加的機會。

1.2?兩種加入rdlist途徑的不同

下面我們來分析一下圖中兩種將epitem加入rdlist方式(也就是紅線和藍線)的區別。

l?紅線:fd狀態改變是才會觸發。那么什么情況會導致fd狀態的改變呢?

對于讀取操作:

(1)?當buffer由不可讀狀態變為可讀的時候,即由空變為不空的時候。

(2)?當有新數據到達時,即buffer中的待讀內容變多的時候。

對于寫操作:

(1)?當buffer由不可寫變為可寫的時候,即由滿狀態變為不滿狀態的時候。

(2)?當有舊數據被發送走時,即buffer中待寫的內容變少得時候。

?

l?藍線:fd的events中有相應的時間(位置1)即會觸發。那么什么情況下會改變events的相應位呢?

對于讀操作:

(1)?buffer中有數據可讀的時候,即buffer不空的時候fd的events的可讀為就置1。

對于寫操作:

(1)?buffer中有空間可寫的時候,即buffer不滿的時候fd的events的可寫位就置1。

?

說明:紅線是時間驅動被動觸發,藍線是函數查詢主動觸發。

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

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

相關文章

OPENSSL X509證書驗證

openssl實現了標準的x509v3數字證書&#xff0c;其源碼在crypto/x509和crypto/x509v3中。其中x509目錄實現了數字證書以及證書申請相關的各種函數&#xff0c;包括了X509和X509_REQ結構的設置、讀取、打印和比較&#xff1b;數字證書的驗證、摘要&#xff1b;各種公鑰的導入導出…

linux網絡編程九:splice函數,高效的零拷貝

1. splice函數 #include <fcntl.h> ssize_t splice(int fd_in, loff_t *off_in, int fd_out, loff_t *off_out, size_t len, unsigned int flags); splice用于在兩個文件描述符之間移動數據&#xff0c; 也是零拷貝。 fd_in參數是待輸入描述符。如果它是一個管道文件…

sys/queue.h

概述 sys/queue.h是LINUX/UNIX系統下面的一個標準頭文件&#xff0c;用一系列的數據結構定義了一隊列。包括singly-lined list, list, simple queue(Singly-linked Tail queue), tail queue, circle queue五種。 引用此頭文件對這五種數據結構的描述&#xff1a; A singly-lin…

sys/queue.h分析(圖片復制不過來,查看原文)

這兩天有興趣學習使用了下系統頭文件sys/queue.h中的鏈表/隊列的實現&#xff0c;感覺實現的很是優美&#xff0c;關鍵是以后再也不需要自己實現這些基本的數據結構了&#xff0c;哈哈&#xff01; 我的系統環境是 正好需要使用隊列&#xff0c;那么本篇就以其中的尾隊列&…

線程池原理及C語言實現線程池

備注&#xff1a;該線程池源碼參考自傳直播客培訓視頻配套資料&#xff1b; 源碼&#xff1a;https://pan.baidu.com/s/1zWuoE3q0KT5TUjmPKTb1lw 密碼&#xff1a;pp42 引言&#xff1a;線程池是一種多線程處理形式&#xff0c;大多用于高并發服務器上&#xff0c;它能合理有效…

iptables 的mangle表

mangle表的主要功能是根據規則修改數據包的一些標志位&#xff0c;以便其他規則或程序可以利用這種標志對數據包進行過濾或策略路由。 內網的客戶機通過Linux主機連入Internet&#xff0c;而Linux主機與Internet連接時有兩條線路&#xff0c;它們的網關如圖所示。現要求對內網進…

Linux常用命令(一)

history 查看歷史命令 ctrlp 向上翻歷史紀錄 ctrln 向下翻歷史紀錄 ctrlb 光標向左移動 ctrlf 光標向右移動 ctrla 光標移動到行首 ctrle 光標移動到行尾 ctrlh 刪除光標前一個 ctrld 刪除光標后一個 ctrlu 刪除光標前所有 ctrlL clear命令 清屏 tab鍵可以補全命令/填充路徑…

ip route / ip rule /iptables 配置策略路由

Linux 使用 ip route , ip rule , iptables 配置策略路由 要求192.168.0.100以內的使用 10.0.0.1 網關上網&#xff0c;其他IP使用 20.0.0.1 上網。 首先要在網關服務器上添加一個默認路由&#xff0c;當然這個指向是絕大多數的IP的出口網關。 ip route add default gw 20.0.0.…

iptables:tproxy做透明代理

什么是透明代理 客戶端向真實服務器發起連接&#xff0c;代理機冒充服務器與客戶端建立連接&#xff0c;并以客戶端ip與真實服務器建立連接進行代理轉發。因此對于客戶端與服務器來說&#xff0c;代理機都是透明的。 如何建立透明代理 本地socket捕獲數據包 nat方式 iptables…

編譯參數(-D)

程序中可以使用#ifdef來控制輸出信息 #include<stdio.h> #define DEBUGint main() {int a 10;int b 20;int sum a b; #ifdef DEBUGprintf("%d %d %d\n",a,b,sum); #endifreturn 0; } 這樣在有宏定義DEBGU的時候就會有信息輸出 如果注銷掉宏定義就不會有輸…

libpcap講解與API接口函數講解

ibpcap&#xff08;Packet Capture Library&#xff09;&#xff0c;即數據包捕獲函數庫&#xff0c;是Unix/Linux平臺下的網絡數據包捕獲函數庫。它是一個獨立于系統的用戶層包捕獲的API接口&#xff0c;為底層網絡監測提供了一個可移植的框架。 一、libpcap工作原理 libpcap…

Linux常用命令(三)

man 查看幫助文檔 alias ls : 查看命令是否被封裝 echo &#xff1a; 顯示字符串到屏幕終端 echo $PATH : 將環境變量打印出來 poweroff&#xff1a;關機 rebot&#xff1a;重啟 需要管理員權限 vim是從vi發展過來的文本編輯器 命令模式&#xff1a;打開文件之后默認進入命令模…

淺談iptables防SYN Flood攻擊和CC攻擊

何為syn flood攻擊&#xff1a; SYN Flood是一種廣為人知的DoS&#xff08;拒絕服務攻擊&#xff09;是DDoS&#xff08;分布式拒絕服務攻擊&#xff09;的方式之一&#xff0c;這是一種利用TCP協議缺陷&#xff0c;發送大量偽造的TCP連接請求&#xff0c;從而使得被攻擊方資源…

Linux之靜態庫

命名規則&#xff1a; lib 庫的名字 .a 制作步驟 生成對應.o文件 .c .o 將生成的.o文件打包 ar rcs 靜態庫的名字&#xff08;libMytest.a&#xff09; 生成的所有的.o 發布和使用靜態庫&#xff1a; 1&#xff09; 發布靜態 2&#xff09; 頭文件 文件如下圖所示&…

iptables詳解和練習

防火墻&#xff0c;其實說白了講&#xff0c;就是用于實現Linux下訪問控制的功能的&#xff0c;它分為硬件的或者軟件的防火墻兩種。無論是在哪個網絡中&#xff0c;防火墻工作的地方一定是在網絡的邊緣。而我們的任務就是需要去定義到底防火墻如何工作&#xff0c;這就是防火墻…

Linux之動態庫

命令規則 lib 名字 .so 制作步驟 1&#xff09;生成與位置無關的代碼&#xff08;生成與位置無關的代碼&#xff09; 2&#xff09;將.o打包成共享庫&#xff08;動態庫&#xff09; 發布和使用共享庫 動態庫運行原理&#xff1a; 生成動態庫&#xff1a; gcc -fPIC -c *.c -…

linux下源碼安裝vsftpd-3.0.2

1&#xff09;在http://vsftpd.beasts.org/網站中查找并下載 vsftpd-3.0.2.tar.gz源碼包 2)如果自己的機器上安裝有yum可以用yum grouplist | less指令查看以下開發環境&#xff0c;當然這一步不做也行 3&#xff09;拆解源碼包 4&#xff09;查看源碼包 5&#xff09;編輯…

Linux之GDB調試命令

gdb啟動 gdb 程序名 l 查看源代碼&#xff08;默認顯示十行&#xff09; l 文件名&#xff1a;行數 l 文件名&#xff1a;函數名 添加斷點 break 行數 &#xff08;b 也行&#xff09; b 15 if i 15 條件斷點 i b 查看斷點信息 start 程序執行一步 n 單步調試 s 單步&#xf…

Gdb 調試core文件詳解

一&#xff0c;什么是coredump 我們經常聽到大家說到程序core掉了&#xff0c;需要定位解決&#xff0c;這里說的大部分是指對應程序由于各種異常或者bug導致在運行過程中異常退出或者中止&#xff0c;并且在滿足一定條件下&#xff08;這里為什么說需要滿足一定的條件呢&#…

Linux之GDB命令(二)

gdb命令&#xff1a; 前提條件&#xff1a;可執行文件必須包含調試信息 gcc -ggdb 文件名 –啟動gdb調試查看代碼命令 當前文件&#xff1a; list 行號&#xff08;函數名&#xff09; 指定文件&#xff1a; list 文件名&#xff1a;行號&#xff08;函數名&#x…