1.イベントループco_eventloop
ファイルイベントとタイムイベントをタイムホイールに登録した後、メインコルーチンはイベントループ関数co_eventloopを呼び出してタイムホイールをポーリングします。タイムアウトイベントが発生すると、対応するイベントのコールバック関数が呼び出されて実行されます。
/*
* @param ctx :epoll管理器
* @param pfn :每轮事件循环的最后会调用该函数
* @param arg :pfn的参数
*/
void co_eventloop(stCoEpoll_t *ctx, pfn_co_eventloop_t pfn, void *arg)
{
if( !ctx->result )
{
ctx->result = co_epoll_res_alloc( stCoEpoll_t::_EPOLL_SIZE );
}
co_epoll_res *result = ctx->result;
for(;;)
{
int ret = co_epoll_wait(ctx->iEpollFd,result,stCoEpoll_t::_EPOLL_SIZE, 1 ); // 获取epoll中活跃的文件事件
stTimeoutItemLink_t *active = (ctx->pstActiveList);
stTimeoutItemLink_t *timeout = (ctx->pstTimeoutList);
memset(timeout,0,sizeof(stTimeoutItemLink_t));
/* 将活跃文件事件加入到活跃链表中 */
for (int i = 0; i < ret; i++)
{
/* 获取在co_poll_inner放入epoll_event中的stTimeoutItem_t结构体 */
stTimeoutItem_t *item = (stTimeoutItem_t*)result->events[i].data.ptr;
if( item->pfnPrepare )
{
item->pfnPrepare(item,result->events[i], active); // 执行预处理回调函数
}
else
{
AddTail(active, item); // 加入到活跃链表
}
}
/* 获取时间轮中的超时事件,并放入timeout中 */
unsigned long long now = GetTickMS();
TakeAllTimeout(ctx->pTimeout, now, timeout);
/* 将timeout中的超时事件加入到活跃链表中active */
stTimeoutItem_t *lp = timeout->head;
while (lp)
{
//printf("raise timeout %p\n",lp);
lp->bTimeout = true;
lp = lp->pNext;
}
Join<stTimeoutItem_t, stTimeoutItemLink_t>(active, timeout);
/* 遍历所有活跃链表active,然后执行对应的回调函数 */
lp = active->head;
while (lp)
{
PopHead<stTimeoutItem_t, stTimeoutItemLink_t>(active); // 把链表的头节点弹出
if (lp->pfnProcess)
{
lp->pfnProcess(lp); // 执行事件的回调函数
}
lp = active->head;
}
/* 每次事件循环结束以后执行该函数, 用于终止协程 */
if (pfn)
{
if (-1 == pfn(arg))
{
break;
}
}
}
}
2.TakeAllTimeout関数
この関数の機能は、現在の時刻に従ってタイムアウトしたイベントをタイムホイールから取り出し、タイムアウトリンクリストに挿入することです。
inline void TakeAllTimeout(stTimeout_t *apTimeout, unsigned long long allNow, stTimeoutItemLink_t *apResult)
{
if(apTimeout->ullStart == 0) // 第一次调用是设置初始时间
{
apTimeout->ullStart = allNow;
apTimeout->llStartIdx = 0;
}
/* 当前时间小于初始时间时返回 */
if(allNow < apTimeout->ullStart)
{
return ;
}
/* 求一个取出事件的有效区间 */
int cnt = allNow - apTimeout->ullStart + 1;
if( cnt > apTimeout->iItemSize )
{
cnt = apTimeout->iItemSize;
}
if( cnt < 0 )
{
return;
}
/* 把上面求的有效区间过一遍,某一项存在数据的话插入到超时链表中 */
for (int i = 0; i < cnt; i++)
{
int idx = (apTimeout->llStartIdx + i) % apTimeout->iItemSize;
Join<stTimeoutItem_t, stTimeoutItemLink_t>(apResult, apTimeout->pItems + idx);
}
/* 更新时间轮 */
apTimeout->ullStart = allNow;
apTimeout->llStartIdx += cnt - 1;
}