libevent源码分析:event_add、event_del
event_add、event_del两个函数分别是使event生效和失效的,下面就来看一下两个函数的实现。
event_add
- int
- event_add(struct event *ev, const struct timeval *tv)
- {
- int res;
- if (EVUTIL_FAILURE_CHECK(!ev->ev_base)) {
- event_warnx("%s: event has no event_base set.", __func__);
- return -;
- }
- EVBASE_ACQUIRE_LOCK(ev->ev_base, th_base_lock);
- res = event_add_nolock_(ev, tv, );
- EVBASE_RELEASE_LOCK(ev->ev_base, th_base_lock);
- return (res);
- }
- /* Implementation function to add an event. Works just like event_add,
- * except: 1) it requires that we have the lock. 2) if tv_is_absolute is set,
- * we treat tv as an absolute time, not as an interval to add to the current
- * time */
- int
- event_add_nolock_(struct event *ev, const struct timeval *tv,
- int tv_is_absolute)
- {
- struct event_base *base = ev->ev_base;
- int res = ;
- int notify = ;
- EVENT_BASE_ASSERT_LOCKED(base);
- event_debug_assert_is_setup_(ev);
- event_debug((
- "event_add: event: %p (fd "EV_SOCK_FMT"), %s%s%s%scall %p",
- ev,
- EV_SOCK_ARG(ev->ev_fd),
- ev->ev_events & EV_READ ? "EV_READ " : " ",
- ev->ev_events & EV_WRITE ? "EV_WRITE " : " ",
- ev->ev_events & EV_CLOSED ? "EV_CLOSED " : " ",
- tv ? "EV_TIMEOUT " : " ",
- ev->ev_callback));
- EVUTIL_ASSERT(!(ev->ev_flags & ~EVLIST_ALL));
- if (ev->ev_flags & EVLIST_FINALIZING) {
- /* XXXX debug */
- return (-);
- }
- /*
- * prepare for timeout insertion further below, if we get a
- * failure on any step, we should not change any state.
- */
- if (tv != NULL && !(ev->ev_flags & EVLIST_TIMEOUT)) {
- if (min_heap_reserve_(&base->timeheap,
- + min_heap_size_(&base->timeheap)) == -)
- return (-); /* ENOMEM == errno */
- }
- /* If the main thread is currently executing a signal event's
- * callback, and we are not the main thread, then we want to wait
- * until the callback is done before we mess with the event, or else
- * we can race on ev_ncalls and ev_pncalls below. */
- #ifndef EVENT__DISABLE_THREAD_SUPPORT
- if (base->current_event == event_to_event_callback(ev) &&
- (ev->ev_events & EV_SIGNAL)
- && !EVBASE_IN_THREAD(base)) {
- ++base->current_event_waiters;
- EVTHREAD_COND_WAIT(base->current_event_cond, base->th_base_lock);
- }
- #endif
- if ((ev->ev_events & (EV_READ|EV_WRITE|EV_CLOSED|EV_SIGNAL)) &&
- !(ev->ev_flags & (EVLIST_INSERTED|EVLIST_ACTIVE|EVLIST_ACTIVE_LATER))) {
- if (ev->ev_events & (EV_READ|EV_WRITE|EV_CLOSED))
- res = evmap_io_add_(base, ev->ev_fd, ev);
- else if (ev->ev_events & EV_SIGNAL)
- res = evmap_signal_add_(base, (int)ev->ev_fd, ev);
- if (res != -)
- event_queue_insert_inserted(base, ev);
- if (res == ) {
- /* evmap says we need to notify the main thread. */
- notify = ;
- res = ;
- }
- }
- /*
- * we should change the timeout state only if the previous event
- * addition succeeded.
- */
- if (res != - && tv != NULL) {
- struct timeval now;
- int common_timeout;
- #ifdef USE_REINSERT_TIMEOUT
- int was_common;
- int old_timeout_idx;
- #endif
- /*
- * for persistent timeout events, we remember the
- * timeout value and re-add the event.
- *
- * If tv_is_absolute, this was already set.
- */
- if (ev->ev_closure == EV_CLOSURE_EVENT_PERSIST && !tv_is_absolute)
- ev->ev_io_timeout = *tv;
- #ifndef USE_REINSERT_TIMEOUT
- if (ev->ev_flags & EVLIST_TIMEOUT) {
- event_queue_remove_timeout(base, ev);
- }
- #endif
- /* Check if it is active due to a timeout. Rescheduling
- * this timeout before the callback can be executed
- * removes it from the active list. */
- if ((ev->ev_flags & EVLIST_ACTIVE) &&
- (ev->ev_res & EV_TIMEOUT)) {
- if (ev->ev_events & EV_SIGNAL) {
- /* See if we are just active executing
- * this event in a loop
- */
- if (ev->ev_ncalls && ev->ev_pncalls) {
- /* Abort loop */
- *ev->ev_pncalls = ;
- }
- }
- event_queue_remove_active(base, event_to_event_callback(ev));
- }
- gettime(base, &now);
- common_timeout = is_common_timeout(tv, base);
- #ifdef USE_REINSERT_TIMEOUT
- was_common = is_common_timeout(&ev->ev_timeout, base);
- old_timeout_idx = COMMON_TIMEOUT_IDX(&ev->ev_timeout);
- #endif
- if (tv_is_absolute) {
- ev->ev_timeout = *tv;
- } else if (common_timeout) {
- struct timeval tmp = *tv;
- tmp.tv_usec &= MICROSECONDS_MASK;
- evutil_timeradd(&now, &tmp, &ev->ev_timeout);
- ev->ev_timeout.tv_usec |=
- (tv->tv_usec & ~MICROSECONDS_MASK);
- } else {
- evutil_timeradd(&now, tv, &ev->ev_timeout);
- }
- event_debug((
- "event_add: event %p, timeout in %d seconds %d useconds, call %p",
- ev, (int)tv->tv_sec, (int)tv->tv_usec, ev->ev_callback));
- #ifdef USE_REINSERT_TIMEOUT
- event_queue_reinsert_timeout(base, ev, was_common, common_timeout, old_timeout_idx);
- #else
- event_queue_insert_timeout(base, ev);
- #endif
- if (common_timeout) {
- struct common_timeout_list *ctl =
- get_common_timeout_list(base, &ev->ev_timeout);
- if (ev == TAILQ_FIRST(&ctl->events)) {
- common_timeout_schedule(ctl, &now, ev);
- }
- } else {
- struct event* top = NULL;
- /* See if the earliest timeout is now earlier than it
- * was before: if so, we will need to tell the main
- * thread to wake up earlier than it would otherwise.
- * We double check the timeout of the top element to
- * handle time distortions due to system suspension.
- */
- if (min_heap_elt_is_top_(ev))
- notify = ;
- else if ((top = min_heap_top_(&base->timeheap)) != NULL &&
- evutil_timercmp(&top->ev_timeout, &now, <))
- notify = ;
- }
- }
- /* if we are not in the right thread, we need to wake up the loop */
- if (res != - && notify && EVBASE_NEED_NOTIFY(base))
- evthread_notify_base(base);
- event_debug_note_add_(ev);
- return (res);
- }
这里以epoll作为后端来举例分析event_add函数的调用流程:
event_del
- int
- event_del(struct event *ev)
- {
- return event_del_(ev, EVENT_DEL_AUTOBLOCK);
- }
- static int
- event_del_(struct event *ev, int blocking)
- {
- int res;
- if (EVUTIL_FAILURE_CHECK(!ev->ev_base)) {
- event_warnx("%s: event has no event_base set.", __func__);
- return -;
- }
- EVBASE_ACQUIRE_LOCK(ev->ev_base, th_base_lock);
- res = event_del_nolock_(ev, blocking);
- EVBASE_RELEASE_LOCK(ev->ev_base, th_base_lock);
- return (res);
- }
- /** Helper for event_del: always called with th_base_lock held.
- *
- * "blocking" must be one of the EVENT_DEL_{BLOCK, NOBLOCK, AUTOBLOCK,
- * EVEN_IF_FINALIZING} values. See those for more information.
- */
- int
- event_del_nolock_(struct event *ev, int blocking)
- {
- struct event_base *base;
- int res = , notify = ;
- event_debug(("event_del: %p (fd "EV_SOCK_FMT"), callback %p",
- ev, EV_SOCK_ARG(ev->ev_fd), ev->ev_callback));
- /* An event without a base has not been added */
- if (ev->ev_base == NULL)
- return (-);
- EVENT_BASE_ASSERT_LOCKED(ev->ev_base);
- if (blocking != EVENT_DEL_EVEN_IF_FINALIZING) {
- if (ev->ev_flags & EVLIST_FINALIZING) {
- /* XXXX Debug */
- return ;
- }
- }
- /* If the main thread is currently executing this event's callback,
- * and we are not the main thread, then we want to wait until the
- * callback is done before we start removing the event. That way,
- * when this function returns, it will be safe to free the
- * user-supplied argument. */
- base = ev->ev_base;
- #ifndef EVENT__DISABLE_THREAD_SUPPORT
- if (blocking != EVENT_DEL_NOBLOCK &&
- base->current_event == event_to_event_callback(ev) &&
- !EVBASE_IN_THREAD(base) &&
- (blocking == EVENT_DEL_BLOCK || !(ev->ev_events & EV_FINALIZE))) {
- ++base->current_event_waiters;
- EVTHREAD_COND_WAIT(base->current_event_cond, base->th_base_lock);
- }
- #endif
- EVUTIL_ASSERT(!(ev->ev_flags & ~EVLIST_ALL));
- /* See if we are just active executing this event in a loop */
- if (ev->ev_events & EV_SIGNAL) {
- if (ev->ev_ncalls && ev->ev_pncalls) {
- /* Abort loop */
- *ev->ev_pncalls = ;
- }
- }
- if (ev->ev_flags & EVLIST_TIMEOUT) {
- /* NOTE: We never need to notify the main thread because of a
- * deleted timeout event: all that could happen if we don't is
- * that the dispatch loop might wake up too early. But the
- * point of notifying the main thread _is_ to wake up the
- * dispatch loop early anyway, so we wouldn't gain anything by
- * doing it.
- */
- event_queue_remove_timeout(base, ev);
- }
- if (ev->ev_flags & EVLIST_ACTIVE)
- event_queue_remove_active(base, event_to_event_callback(ev));
- else if (ev->ev_flags & EVLIST_ACTIVE_LATER)
- event_queue_remove_active_later(base, event_to_event_callback(ev));
- if (ev->ev_flags & EVLIST_INSERTED) {
- event_queue_remove_inserted(base, ev);
- if (ev->ev_events & (EV_READ|EV_WRITE|EV_CLOSED))
- res = evmap_io_del_(base, ev->ev_fd, ev);
- else
- res = evmap_signal_del_(base, (int)ev->ev_fd, ev);
- if (res == ) {
- /* evmap says we need to notify the main thread. */
- notify = ;
- res = ;
- }
- }
- /* if we are not in the right thread, we need to wake up the loop */
- if (res != - && notify && EVBASE_NEED_NOTIFY(base))
- evthread_notify_base(base);
- event_debug_note_del_(ev);
- return (res);
- }
这里以epoll作为后端来分析event_del的调用流程:
结论:
到这里event_add、event_del函数就分析完了,这两个函数的功能就是使事件生效和失效,以epoll作为后端举例,最后都会调用epoll_ctl来修改事件,libevent实现的很复杂,是因为它考虑到效率的问题,关于libevent如何保证了libevent的高效,这个待之后彻底理解了libevent的设计之后再来分析。
libevent源码分析:event_add、event_del的更多相关文章
- Libevent源码分析—event_add()
接下来就是将已经初始化的event注册到libevent的事件链表上,通过event_add()来实现,源码位于event.c中. event_add() 这个函数主要完成了下面几件事: 1.将eve ...
- 【转】libevent源码分析
libevent源码分析 转自:http://www.cnblogs.com/hustcat/archive/2010/08/31/1814022.html 这两天没事,看了一下Memcached和l ...
- Libevent源码分析 (1) hello-world
Libevent源码分析 (1) hello-world ⑨月份接触了久闻大名的libevent,当时想读读源码,可是由于事情比较多一直没有时间,现在手头的东西基本告一段落了,我准备读读libeven ...
- libevent源码分析
这两天没事,看了一下Memcached和libevent的源码,做个小总结. 1.入门 1.1.概述Libevent是一个用于开发可扩展性网络服务器的基于事件驱动(event-driven)模型的网络 ...
- Libevent源码分析系列【转】
转自:https://www.cnblogs.com/zxiner/p/6919021.html 1.使用libevent库 源码那么多,该怎么分析从哪分析呢?一个好的方法就是先用起来,会用了 ...
- Libevent源码分析系列
1.使用libevent库 源码那么多,该怎么分析从哪分析呢?一个好的方法就是先用起来,会用了,然后去看底层相应的源码,这样比较有条理,自上向下掌握.下面用libevent库写个程序,每隔1秒 ...
- Libevent源码分析—event_base_dispatch()
我们知道libevent是一个Reactor模式的事件驱动的网络库. 到目前为止,我们已经看了核心的event和event_base结构体的源码,看了初始化这两个结构体的源码,看了注册event的 ...
- libevent源码分析一--io事件响应
这篇文章将分析libevent如何组织io事件,如何捕捉事件的发生并进行相应的响应.这里不会详细分析event与event_base的细节,仅描述io事件如何存储与如何响应. 1. select l ...
- libevent源码分析之信号处理
新看看官方demo的libevent如何使用信号 int called = 0; static void signal_cb(int fd, short event, void *arg) { str ...
随机推荐
- SIGTERM等信号含义【转】
主要是做sigterm 和sigkill区别的比较,之前的好多操作,在结束正在运行的一个程序的时候,常用kill,这个以后的注意,在sigterm不起作用的时候,再使用kill; 原文地址:http: ...
- C++11开发中的Atomic原子操作
C++11开发中的Atomic原子操作 Nicol的博客铭 原文 https://taozj.org/2016/09/C-11%E5%BC%80%E5%8F%91%E4%B8%AD%E7%9A%84 ...
- fedora国内源常见配置
yum install yum-fastestmirror3.rpmfusion源 rpm -ivh http://download1.rpmfusion.org/free/fedora/rpmfus ...
- 【Oracle】Oracle约束的总结
你对ORACLE约束的了解如何?比较模糊还是相当透彻?如果你对下面几个问题了如指掌的话,恭喜你,你已经对约束掌握得比较好了,不用看这篇文章了.ORACLE的约束有啥功能作用? 有哪些类型约束(不同版本 ...
- MongoDB Linux环境安装及配置[转]
CentOS 6.5系统中使用yum安装MongoDB 2.6 教程 CentOS 6.5系统中使用yum安装MongoDB 2.6 教程,本文共分5个步骤完成MongoDB的安装.下面我们在Cent ...
- ASP.NET实现二维码 ASP.Net上传文件 SQL基础语法 C# 动态创建数据库三(MySQL) Net Core 实现谷歌翻译ApI 免费版 C#发布和调试WebService ajax调用WebService实现数据库操作 C# 实体类转json数据过滤掉字段为null的字段
ASP.NET实现二维码 using System;using System.Collections.Generic;using System.Drawing;using System.Linq;us ...
- oracle三大范式(转载)
标准化表示从你的数据存储中移去数据冗余 (redundancy)的过程.如果数据库设计达到了完全的标准化,则把所有的表通过关键字连接在一起时,不会出现任何数据的复本 (repetition).标准化的 ...
- golang 学习笔记 ---数组/字符串/切片
数组 数组是一个由固定长度的特定类型元素组成的序列,一个数组可以由零个或多个元素组成.数组的长度是数组类型的组成部分.因为数组的长度是数组类型的一个部分,不同长度或不同类型的数据组成的数组都是不同的类 ...
- shell脚本监控cpu/内存使用率 转
该脚本检测cpu和内存的使用情况,只需要调整memorySetting.cpuSetting.userEmail要发邮件报警的email地址即可 如果没有配置发邮件参数的哥们,已配置了的,直接飞到代码 ...
- iOS自己主动化測试的那些干货
前言 假设有測试大佬发现内容不正确.欢迎指正,我会及时改动. 大多数的iOS App(没有持续集成)迭代流程是这种 也就是说.測试是公布之前的最后一道关卡.假设bug不能在測试中发现,那么bug 就会 ...