LT 模式下:
描述符上事件就绪后,如果没有把数据处理完成,或者没有处理,下一次 epoll 会继续提醒应用程序,直到把数据读完。
ET 模式下:
描述符上事件就绪后,如果没有把数据处理完成,或者没有处理,下一次 epoll 不会提醒应用层序,所以要求应用程序在收到一次提醒时,必须当下将所有数据处理完成。
内核实现:
- epoll_wait 每次将收集到的就绪事件和描述符返回给应用程序,过程是这样,当检测到 rdlist 不为空时,就说明有事件就绪,使用 ep_collect_ready_items 方法将就绪队列 rdlist 中的数据挪到 txlist 中,此时,rdlist 为空。
- 再通过 ep_send_events 方法将就绪事件返回给应用程序,同时又会掉用ep_reinject_items方法将一些有设置EPOLLET 的事件的描述符又放回到 rdlist,这样下一轮,epoll_wait会发现有就绪事件,但会再次检查是否有数据,有就返回,没有继续阻塞。
- 但设置了 EPOLLET 事件,相应的描述符在 ep_reinject_items 方法中不会被放回
rdlist,只有等设备驱动程序检查到有事件产生才会再次将事件放到 rdlist。所以,如果 ET 模式数据没有读完不会放回rdlist,那么也就不会再去检查是否有数据,提醒应用程序了,直到下次设备驱动程序检查到了事件才会放入 rdlist。