一看名字就知道是围绕eventloop转的。
那首先肯定是eventloop是个什么?一般都是IO事件,timer事件的管理器。

那首先看如何new出来一个eventloop:

1、因为libevent是跨平台的,在不同平台上会有不同的配置,首先读配置:

struct event_config {

TAILQ_HEAD(event_configq, event_config_entry) entries;

int n_cpus_hint;

enum event_method_feature require_features;

enum event_base_config_flag flags;

};

有一个event_config 的结构,目的就是为了描述将要创建的event_base的配置。

#define TAILQ_HEAD(name, type) \

struct name { \

struct type *tqh_first; \

struct type **tqh_last; \

}

TAILQ_HEAD是一个结构体的宏。

好吧,为了方便阅读我把宏给干掉了:

struct event_config_entry {

struct {

event_config_entry* tqe_next;

event_config_entry** tqe_prev;

} next;

const char *avoid_method;

};

struct event_config {

struct event_configq{

struct event_config_entry *tqh_first;

struct event_config_entry **tqh_last;

} entries;

int n_cpus_hint;

enum event_method_feature require_features;

enum event_base_config_flag flags;

};

现在清楚了event_config_new就是分配了这个结构体(链表)的内存,并对链表做了初始化。

下一步要开始初始化配置了:

event_base_new_with_config在做这个工作:

哇塞,这个函数里就开始new eventBase了:

而且在对base做初始化

1,、初始化base的时间:

gettime(base, &base->event_tv);

看下实现就清楚了:

static int gettime(struct event_base *base, struct timeval *tp)

{

EVENT_BASE_ASSERT_LOCKED(base);

if (base->tv_cache.tv_sec) {

*tp = base->tv_cache;

return (0);

}

return (evutil_gettimeofday(tp, NULL));

}

如果base的tv_cache保存的有时间就直接用这个时间,以避免调用函数获取时间太频繁;

否则调用evutil_gettimeofday 是会调用各个系统获取当前时间的函数。

之后设置这个几个参数:

base->sig.ev_signal_pair[0] = -1;

base->sig.ev_signal_pair[1] = -1;

base->th_notify_fd[0] = -1;

base->th_notify_fd[1] = -1;

sig是与信号相关的,linux下的,windows下的貌似没什么用;重点看th_notify_fd的作用:

evutil_socket_t th_notify_fd[2];

意思是用来由一些线程通知函数唤醒主线程的。

event_deferred_cb_queue_init(&base->defer_queue);

初始化defer_queue队列,这个队列的作用是:

这个队列内部结构如下:

struct deferred_cb_queue {

void *lock;

int active_count;

void (*notify_fn)(struct deferred_cb_queue *, void *);

void *notify_arg;

TAILQ_HEAD (deferred_cb_list, deferred_cb) deferred_cb_list;

};

里面还有个链表:

struct deferred_cb {

TAILQ_ENTRY (deferred_cb) cb_next;

unsigned queued : 1;

deferred_cb_fn cb;

void *arg;

};

之后设置根据配置backend;

在windows下可以设置为select或者iocp;如果是select代码会在win32select.c里:

struct eventop win32ops = {

"win32",

win32_init,

win32_add,

win32_del,

win32_dispatch,

win32_dealloc,

0,

0,

sizeof(struct idx_info),

}

初始化时:

void *

win32_init(struct event_base *_base)

{

struct win32op *winop;

size_t size;

if (!(winop = mm_calloc(1, sizeof(struct win32op))))

return NULL;

winop->num_fds_in_fd_sets = NEVENT;

size = FD_SET_ALLOC_SIZE(NEVENT);

if (!(winop->readset_in = mm_malloc(size)))

goto err;

if (!(winop->writeset_in = mm_malloc(size)))

goto err;

if (!(winop->readset_out = mm_malloc(size)))

goto err;

if (!(winop->writeset_out = mm_malloc(size)))

goto err;

if (!(winop->exset_out = mm_malloc(size)))

goto err;

winop->readset_in->fd_count = winop->writeset_in->fd_count = 0;

winop->readset_out->fd_count = winop->writeset_out->fd_count

= winop->exset_out->fd_count = 0;

if (evsig_init(_base) < 0)

winop->signals_are_broken = 1;

return (winop);

 err:

XFREE(winop->readset_in);

XFREE(winop->writeset_in);

XFREE(winop->readset_out);

XFREE(winop->writeset_out);

XFREE(winop->exset_out);

XFREE(winop);

return (NULL);

}

里面分配了32个fd;

好吧Eventbase的事情终于搞完了。好辛苦!!!!

关键是看event_base_dispatch如何进行mainloop的:

int

event_base_loop(struct event_base *base, int flags)

{

const struct eventop *evsel = base->evsel;

struct timeval tv;

struct timeval *tv_p;

int res, done, retval = 0;

EVBASE_ACQUIRE_LOCK(base, th_base_lock);

if (base->running_loop) {

event_warnx("%s: reentrant invocation.  Only one event_base_loop"

   " can run on each event_base at once.", __func__);

EVBASE_RELEASE_LOCK(base, th_base_lock);

return -1;

}

base->running_loop = 1;

clear_time_cache(base);

if (base->sig.ev_signal_added && base->sig.ev_n_signals_added)

evsig_set_base(base);

done = 0;

#ifndef _EVENT_DISABLE_THREAD_SUPPORT

base->th_owner_id = EVTHREAD_GET_ID();

#endif

base->event_gotterm = base->event_break = 0;

while (!done) {

base->event_continue = 0;

if (base->event_gotterm) {

break;

}

if (base->event_break) {

break;

}

timeout_correct(base, &tv);

tv_p = &tv;

if (!N_ACTIVE_CALLBACKS(base) && !(flags & EVLOOP_NONBLOCK)) {

timeout_next(base, &tv_p);

} else {

evutil_timerclear(&tv);

}

if (!event_haveevents(base) && !N_ACTIVE_CALLBACKS(base)) {

event_debug(("%s: no events registered.", __func__));

retval = 1;

goto done;

}

gettime(base, &base->event_tv);

clear_time_cache(base);

res = evsel->dispatch(base, tv_p);

if (res == -1) {

event_debug(("%s: dispatch returned unsuccessfully.",

__func__));

retval = -1;

goto done;

}

update_time_cache(base);

timeout_process(base);

if (N_ACTIVE_CALLBACKS(base)) {

int n = event_process_active(base);

if ((flags & EVLOOP_ONCE)

   && N_ACTIVE_CALLBACKS(base) == 0

   && n != 0)

done = 1;

} else if (flags & EVLOOP_NONBLOCK)

done = 1;

}

event_debug(("%s: asked to terminate loop.", __func__));

done:

clear_time_cache(base);

base->running_loop = 0;

EVBASE_RELEASE_LOCK(base, th_base_lock);

return (retval);

}

跟之前看过的libev是一个模子出来的,都是先检测有木有可读、可写的马上去处理,然后调用backend->dispatch检测事件,处理timer事件。一直转,除非外部要停止或者木有fd了。

下面看下select事件的检测吧:

int

win32_dispatch(struct event_base *base, struct timeval *tv)

{

struct win32op *win32op = base->evbase;

int res = 0;

unsigned j, i;

int fd_count;

SOCKET s;

if (win32op->resize_out_sets) {

size_t size = FD_SET_ALLOC_SIZE(win32op->num_fds_in_fd_sets);

if (!(win32op->readset_out = mm_realloc(win32op->readset_out, size)))

return (-1);

if (!(win32op->exset_out = mm_realloc(win32op->exset_out, size)))

return (-1);

if (!(win32op->writeset_out = mm_realloc(win32op->writeset_out, size)))

return (-1);

win32op->resize_out_sets = 0;

}

fd_set_copy(win32op->readset_out, win32op->readset_in);

fd_set_copy(win32op->exset_out, win32op->writeset_in);

fd_set_copy(win32op->writeset_out, win32op->writeset_in);

fd_count =

   (win32op->readset_out->fd_count > win32op->writeset_out->fd_count) ?

   win32op->readset_out->fd_count : win32op->writeset_out->fd_count;

if (!fd_count) {

long msec = tv ? evutil_tv_to_msec(tv) : LONG_MAX;

if (msec < 0)

msec = LONG_MAX;

Sleep(msec);

return (0);

}

EVBASE_RELEASE_LOCK(base, th_base_lock);

res = select(fd_count,

    (struct fd_set*)win32op->readset_out,

    (struct fd_set*)win32op->writeset_out,

    (struct fd_set*)win32op->exset_out, tv);

EVBASE_ACQUIRE_LOCK(base, th_base_lock);

event_debug(("%s: select returned %d", __func__, res));

if (res <= 0) {

return res;

}

if (win32op->readset_out->fd_count) {

i = rand() % win32op->readset_out->fd_count;

for (j=0; jreadset_out->fd_count; ++j) {

if (++i >= win32op->readset_out->fd_count)

i = 0;

s = win32op->readset_out->fd_array[i];

evmap_io_active(base, s, EV_READ);

}

}

if (win32op->exset_out->fd_count) {

i = rand() % win32op->exset_out->fd_count;

for (j=0; jexset_out->fd_count; ++j) {

if (++i >= win32op->exset_out->fd_count)

i = 0;

s = win32op->exset_out->fd_array[i];

evmap_io_active(base, s, EV_WRITE);

}

}

if (win32op->writeset_out->fd_count) {

SOCKET s;

i = rand() % win32op->writeset_out->fd_count;

for (j=0; jwriteset_out->fd_count; ++j) {

if (++i >= win32op->writeset_out->fd_count)

i = 0;

s = win32op->writeset_out->fd_array[i];

evmap_io_active(base, s, EV_WRITE);

}

}

return (0);

}

考虑的很周到,select之前会释放lock,因为select这里是阻塞的,但不希望阻塞主线程,select会继续加锁。把可读,可写fd进行copy,里面还用了一个简单的随机 保证公平。

之后就更新timecache为当前时间,然后检测timer时间,到点就执行timer事件:

static void

timeout_process(struct event_base *base)

{

struct timeval now;

struct event *ev;

if (min_heap_empty(&base->timeheap)) {

return;

}

gettime(base, &now);

while ((ev = min_heap_top(&base->timeheap))) {

if (evutil_timercmp(&ev->ev_timeout, &now, >))

break;

event_del_internal(ev);

event_debug(("timeout_process: call %p",

ev->ev_callback));

event_active_nolock(ev, EV_TIMEOUT, 1);

}

}

一看代码就明白了使用最小堆管理时间的,当然时间间隔最小的要先执行,处理流程是这样的:

检测时间堆里的事件,如果时间已经超过当前时间,那么先把这个事件从堆里删除,然后再active这个event,就是调用注册进来的timer函数。

好了,暂时写到这里吧。

libevent简单分析的更多相关文章

  1. 简单分析JavaScript中的面向对象

    初学JavaScript的时候有人会认为JavaScript不是一门面向对象的语言,因为JS是没有类的概念的,但是这并不代表JavaScript没有对象的存在,而且JavaScript也提供了其它的方 ...

  2. CSipSimple 简单分析

    简介 CSipSimple是一款可以在android手机上使用的支持sip的网络电话软件,可以在上面设置使用callda网络电话.连接使用方式最好是使用wifi,或者3g这样上网速度快,打起电话来效果 ...

  3. C#中异常:“The type initializer to throw an exception(类型初始值设定项引发异常)”的简单分析与解决方法

    对于C#中异常:“The type initializer to throw an exception(类型初始值设定项引发异常)”的简单分析,目前本人分析两种情况,如下: 情况一: 借鉴麒麟.NET ...

  4. 透过byte数组简单分析Java序列化、Kryo、ProtoBuf序列化

    序列化在高性能网络编程.分布式系统开发中是举足轻重的之前有用过Java序列化.ProtocolBuffer等,在这篇文章这里中简单分析序列化后的byte数组观察各种序列化的差异与性能,这里主要分析Ja ...

  5. 简单分析Java的HashMap.entrySet()的实现

    关于Java的HashMap.entrySet(),文档是这样描述的:这个方法返回一个Set,这个Set是HashMap的视图,对Map的操作会在Set上反映出来,反过来也是.原文是 Returns ...

  6. Ffmpeg解析media容器过程/ ffmpeg 源代码简单分析 : av_read_frame()

    ffmpeg 源代码简单分析 : av_read_frame() http://blog.csdn.net/leixiaohua1020/article/details/12678577 ffmpeg ...

  7. FFmpeg的HEVC解码器源码简单分析:解析器(Parser)部分

    ===================================================== HEVC源码分析文章列表: [解码 -libavcodec HEVC 解码器] FFmpeg ...

  8. FFmpeg资料来源简单分析:libswscale的sws_getContext()

    ===================================================== FFmpeg库函数的源代码的分析文章: [骨架] FFmpeg源码结构图 - 解码 FFmp ...

  9. wp7之换肤原理简单分析

    wp7之换肤原理简单分析 纠结很久...感觉勉强过得去啦.还望各位大牛指点江山 百度找到这篇参考文章http://www.cnblogs.com/sonyye/archive/2012/03/12/2 ...

随机推荐

  1. Python文本处理(1)

    每次处理一个字符 解决方法: 创建列表 thestring='abcdefg' thelist=list(thestring) print thelist 结果 ['a', 'b', 'c', 'd' ...

  2. 【LeetCode】Repeated DNA Sequences 解题报告

    [题目] All DNA is composed of a series of nucleotides abbreviated as A, C, G, and T, for example: &quo ...

  3. SICP练习1.6-1.8

    1.6 死循环 1.7 #lang racket (define (square x) (* x x)) (define (sqrt-iter guess x) (if (good-enough? g ...

  4. 解决win7 中source insight没有courier new字节的问题

    解决win7 中source insight没有courier new字节的问题  http://blog.csdn.net/season_hangzhou/article/details/18665 ...

  5. 3522: [Poi2014]Hotel( 树形dp )

    枚举中点x( 即选出的三个点 a , b , c 满足 dist( x , a ) = dist( x , b ) = dist( x , c ) ) , 然后以 x 为 root 做 dfs , 显 ...

  6. HDOJ 1005

    Input The input consists of multiple test cases. Each test case contains 3 integers A, B and n on a ...

  7. 【C# -- OpenCV】Emgu CV 第一个实例

    原文 [C# -- OpenCV]Emgu CV 第一个实例 Emgu CV下载地址 http://sourceforge.net/projects/emgucv/files/ 找最新的下就行了,傻瓜 ...

  8. Swift - .plist文件数据的读取和存储

    每次在Xcode中新建一个iOS项目后,都会自己产生一个.plist文件,里面记录项目的一些配置信息.我们也可以自己创建.plist文件来进行数据的存储和读取. .plist文件其实就是一个XML格式 ...

  9. Java基础11 对象引用

    链接地址:http://www.cnblogs.com/vamei/archive/2013/04/01/2992484.html 作者:Vamei 出处:http://www.cnblogs.com ...

  10. Debian安装Nexus

    前置条件 安装jdk (如果使用 nexus-2.6 以上版本需要jdk7) 1 apt-get install openjdk-6-jre / apt-get install openjdk-6-j ...