一 libev简介
libev是一个轻量级的事件通知库,具备支持多种事件通知能力,通过对libev的源码的阅读,可以清楚了解事件通知实现内部机制。
二 核心数据结构
在libev中关键的数据结构是,loop结构体,该结构体定义的字段较多,但是主要核心的可以分为两大类
1.各类事件的watcher集合
loop中有支持很多类型的事件,如下
ev_io // IO可读可写
ev_stat // 文件属性变化
ev_signal // 信号处理
ev_timer // 相对定时器
ev_periodic // 绝对定时器
ev_child // 子进程状态变化
ev_fork // fork事件
ev_cleanup // event loop退出触发事件
ev_idle // event loop空闲触发事件
ev_embed // 嵌入另一个后台循环
ev_prepare // event loop之前事件
ev_check // event loop之后事件
ev_async // 线程间异步事件
这些事件的监控管理都对应一种类型的watcher数组,比如
(loop)->anfds :维护所有fd事件
(loop)->timers :维护所有的定时器
(loop)->periodics:周期性事件
(loop)->prepares:该事件是loop启动之间就会执行的事件
等,每类事件都能在loop结构中找到对应的数组来维护对应的watchers。
2.2 watcher结构
这个是公共watcher得到结构,是会被所有的子watcher所共享。
active:表示在loop中对应的watcher数组中的下标
pending & EV_DECL_PRIORITY:分别用来记录在全局loop->pendings中列 & 行下标(loop->pendings下文会介绍)
比如io事件的watcher如上定义,类似的各种watcher都采取此类方式来生成(这样的好处是可以减少代码的重复度,降低了代码维护的成本,但是可读性方面也相应降低)
2.全局触发事件集合loop->pendings
loop->pendings记录管理当前已经发生等待调用回调函数的watcher集合,libev检查到事件发生后将对应的watcher加入到loop->pendings
2.1 数据结构:
*Watcher[N][M]:二位数组用来维护一轮循环下来,需要触发回调函数的watcher
其中N:是每一类watcher的优先级
另外还有loop->pendingcnt结构,也是一个二维数组,用来维护pending中每一行最大watcher下标。
以下是事件函数回调过程的代码,
本质上就是个遍历loop->pendings挨个调用watcher的注册的回调函数,可以看到按照优先级从高(小)到低(大),对于同一优先级watcher按照下标从大到小的方式来调用callBack函数。
值得注意的,就是pendingcnt结构,该结构维护当前每一行watcher数组当前的最大下标。
三 事件触发之io事件
该节将会以IO事件在EPOLL平台的触发整个流程来阐述libev是如何来实现事件触发回调的整个过程。在介绍IO事件触发之前需要介绍一下相关的数据结构
anfds:是ANFD*类型的数组,用来管理每一个fd的监听的事件
changfds:是int*类型数组,每个元素记录当前发生更改的fd,比如加入新的监听fd,或者fd的监听事件发送修改。
在每次循环之前,都会对changefds和anfds的结构中对应的fd事件列表的所有event进行 | 操作,得到当前整个fd当前的监听事件和ANFD.events进行对比,如果没有修改将不会该表epoll的fd的监听事件,这样做的好处,避免了无效的修改,保证了所有对epoll的修改都是必须的,毕竟频繁对epoll进行修改代价还是挺大的。
好了,下面正式分析IO事件是如何触发的,当一个fd监听事件加入时
step1:调用ev_io_start将watcher加入到anfds中,并将修改记录在changfds中
steps2: 调用fd_reify,通过比对changfds & anfds确定是否需要加入epoll事件,此时显然是需要的
核心代码如下,第一处:暂时保存fd的events,第二处:通过遍历watcher链表计算新的events,比较前后是否发送变化,第三处:如果发生变化将修改epoll的监听事件
自此完成监听事件的添加。
step3:调用backend_poll函数得到当前监听已经发生的事件,
#得到事件的fd & 已经发生的events,从anfds中获取对应的ANFD
#遍历ANFD中的watcher链表,比对监听事件和已经发生的监听事件,如果符合将该watcher加入到loop->pendings,修改watcher中的pending变量标记在loop->pendings中的数组下标
step4:遍历循环loop->pendings结构,挨个调用回调函数,从而完成事件触发的一个完整过程
四 事件触发之定时器事件
定时器采用小根堆的方式来维护所有的timer,libev整个过程是采用loop循环的方式,周期性的检测是否有事件发生
在loop中会获取当前堆顶的timer(最近要发生的)以及其他信息来计算当前可以sleep多长时间,从而可以保证进程在休眠的这段时间也不会有事件发生而没有及时通知。
五 ev_run函数解析
ev_run将整个libev的各类事件通知流程串起。整个过程就是个大循环。
过程大致分为
1.检测是否有fork事件,如果有进行fork事件的回调函数
2.在loop之前调用prepare事件的回调函数。
3.检测fd的监听事件是否发生变化,是否需要修改epoll的监听事件
4.计算需要休眠的事件
#根据定时器 & 周期任务 & timeout_blocktime(超时事件收集间隔事件) & io_blocktime(io事件收集间隔事件) 等信息计算此次循环需要sleep的时间
5.如果需要休眠则进行休眠
6.进程从休眠态唤起后,从epoll(pool,kqueue)中获取发生的事件,将对应的watcher加入到 loop->pendings中
7.将定时时间到了的定时器,加入loop->pendings中
8.收集周期任务,加入loop->pendings中
9.收集空闲事件加入loop->pendings中
10.依此对loop->pendings中的watcher中注册的回调函数
自此整个loop完成,总体来说就是在整个loop中检测所有的监听的事件是否发生,然后依次对发生的事件,调用注册的回调函数。
六 源码文件结构
#个平台网络编程接口,不同平台使用不同的文件,从而支持多平台
ev_pool.c
ev_port.c
ev_kqueue.c
ev_select.c
ev_vars.h:定义ev_loop数据结构,使用宏定义的方式进行定义
ev_warp.c:,使用宏定义的方式封装全局变量loop中字段的访问
ev.c:整个libev的核心部分,实现了整个libev的事件通知的大部分业务逻辑
七 总结
libev从整个设计来看还是比较精巧的,大体上将整个事件通知机制划分为两个阶段
#事件发生检测:
各个事件检测过程实现不太一样(IO事件,定时器,周期性任务等各不一样),将检测到发生的事件加入loop->pendings中
#事件回调触发
对loop->pendings的事件,遍历依次触发回调函数
整个libev可能作者出于对代码复用减少重复代码的原因,大量使用宏定义,甚至用宏定义实现了简单的继承关系,这也使得整个项目代码看起来比较晦涩难懂。