epoll能監(jiān)聽(tīng)普通文件嗎?了解epoll背后的魔法!
epoll 是 Linux 系統(tǒng)中常用的多路復(fù)用 I/O 組件,一般用于監(jiān)聽(tīng) socket 是否能夠進(jìn)行 I/O 操作。那么,epoll 能監(jiān)聽(tīng)普通文件嗎?
我們先通過(guò)下面的例子來(lái)驗(yàn)證一下,epoll 能不能監(jiān)聽(tīng)普通文件:
編譯并且運(yùn)行,結(jié)果如下:
可以看到上面的運(yùn)行結(jié)果報(bào) Operation not allow 的錯(cuò)誤,這說(shuō)明 epoll 是不能監(jiān)聽(tīng)普通文件的,為什么呢?
【文章福利】小編推薦自己的Linux內(nèi)核技術(shù)交流群:【891587639】整理了一些個(gè)人覺(jué)得比較好的學(xué)習(xí)書(shū)籍、視頻資料共享在群文件里面,有需要的可以自行添加哦?。?!前100名進(jìn)群領(lǐng)取,額外贈(zèng)送一份價(jià)值699的內(nèi)核資料包(含視頻教程、電子書(shū)、實(shí)戰(zhàn)項(xiàng)目及代碼)?


尋根究底
我們應(yīng)該對(duì)追尋真相抱著熱衷的態(tài)度,所以必須找出 epoll 不能監(jiān)聽(tīng)普通文件的原因。
因?yàn)樵谏厦娴睦又?,?epoll_ctl 函數(shù)報(bào)的錯(cuò),所以我們首先應(yīng)該從 epoll_ctl 的源碼入手,如下:
從上面代碼可以看出,當(dāng)被監(jiān)聽(tīng)的文件沒(méi)有提供 poll 接口時(shí),就會(huì)返回 EPERM 的錯(cuò)誤,這個(gè)錯(cuò)誤就是 Operation not allow 的錯(cuò)誤號(hào)。
所以,出現(xiàn) Operation not allow 的原因就是:被監(jiān)聽(tīng)的文件沒(méi)有提供 poll 接口。
由于我們的文件系統(tǒng)是 ext4,所以我們來(lái)看看 ext4 文件系統(tǒng)中的文件是否提供了 poll 接口(位于文件 /fs/ext4/file.c 中):
ext4 文件的文件操作函數(shù)集被設(shè)置為 ext4_file_operations(也說(shuō)就是:file->f_op = ext4_file_operations),從上面代碼可以看出,ext4_file_operations 并沒(méi)有提供 poll 接口。所以,當(dāng)調(diào)用 epoll_ctl 把文件添加到 epoll 中進(jìn)行監(jiān)聽(tīng)時(shí),就會(huì)返回 Operation not allow 的錯(cuò)誤。
從上面的分析可知,當(dāng)文件系統(tǒng)提供 poll 接口時(shí),就可以把文件添加到 epoll 中進(jìn)行監(jiān)聽(tīng)。
