在 Linux 系统,libco 调用 epoll_wait 有点用力过猛,虽然 libco 针对高并发,但是小问题的处理,略显粗糙。
1. 问题
co_epoll_wait 的 timeout 这里默认为 1,在循环里,每毫秒执行事件处理。
1
2
3
4
5
6
7
8
void co_eventloop(stCoEpoll_t *ctx, pfn_co_eventloop_t pfn, void *arg) {
...
for (;;) {
int ret = co_epoll_wait(ctx->iEpollFd, result, stCoEpoll_t::_EPOLL_SIZE, 1);
...
}
...
}
2. 建议
应该先捞一个快到期的事件,当前时间与到期的时间差进行等待。
详细可以参考 redis 的事件处理: numevents = aeApiPoll(eventLoop, tvp);
3. 测试
3.1. 测试代码
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
int co_sleep(int ms) {
struct pollfd pf = {0};
pf.fd = -1;
pf.events = -1;
return poll(&pf, 1, ms);
}
void* co_handle_timer(void* arg) {
co_enable_hook_sys();
for (;;) {
co_sleep(1000);
}
return 0;
}
int main() {
stCoRoutine_t* co;
co_create(&co, NULL, co_handle_timer, nullptr);
co_resume(co);
co_eventloop(co_get_epoll_ct(), 0, 0);
return 0;
}
3.2. cpu
3.3. 性能火焰图
火焰图参考:如何生成火焰图🔥