一看名字就知道是围绕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. linux c coding style

    Linux kernel coding style This is a short document describing the preferred coding style for the lin ...

  2. linux mount挂载设备(u盘,光盘,iso等 )使用说明

    对于新手学习,mount 命令,一定会有很多疑问.其实我想疑问来源更多的是对linux系统本身特殊性了解问题. linux是基于文件系统,所有的设备都会对应于:/dev/下面的设备.如: [cheng ...

  3. 正则表达式验证数字、汉字、电话号码,email,整数,浮点数

    验证数字的正则表达式集 验证数字:^[0-9]*$验证n位的数字:^\d{n}$验证至少n位数字:^\d{n,}$验证m-n位的数字:^\d{m,n}$验证零和非零开头的数字:^(0|[1-9][0- ...

  4. SQL 局域网远程备份

     局域网远程备份,还真有点问题 我用的是2003,发现sql和windows兼容不太好 . 如果我在windows下面映射远程目录,sql发现不了,找不到路径,备份不了,在sql下映射远程目录,win ...

  5. BZOJ 2212: [Poi2011]Tree Rotations( 线段树 )

    线段树的合并..对于一个点x, 我们只需考虑是否需要交换左右儿子, 递归处理左右儿子. #include<bits/stdc++.h> using namespace std; #defi ...

  6. Java中StringBuilder的清空方法比較

    StringBuilder 没有提供clear或empty方法. 清空有3种方法: 1)新生成一个,旧的由系统自己主动回收 2)使用delete 3)使用setLength 将三种方法循环1000万次 ...

  7. Oracle cloud control 12c 怎样改动sysmanpassword

        前阵子在虚拟机部署了Oracle Cloud Control 12c.事别几日,居然忘记了登录password. 主要是由于如今的Oracle有关的Software比之前提供更强的安全机制.什 ...

  8. Integer ==与Equals【原创】

    package Equals; public class IntegerEquals { public static void main(String[] args) { printLine(128) ...

  9. [置顶] oracle存储过程中单引号及字符串拼接处理

    在ORACLE中,单引号有两个作用,一是字符串是由单引号引用,二是转义.单引号的使用是就近配对,即就近原则.而在单引号充当转义角色时相对不好理解 1.从第二个单引号开始被视为转义符,如果第二个单引号后 ...

  10. display:inline-block的运用

    在实习中做专题时,遇到的一个问题:建立一个宽度很长的一个页面,里面包含许多列.或许许多人认为直接设置float:left:不就行了 但是这个有一个问题,你必须把外面的div的宽度设置的很长已满足大于所 ...