Libev源码分析07:Linux下的eventfd简介
#include <sys/eventfd.h> int eventfd(unsigned int initval, int flags);
eventfd创建一个eventfd对象,该对象可用于用户空间的程序实现事件等待、通知机制,也可用于由内核向用户空间的应用进行事件的通知。eventfd对象在内核中包含了一个计数器,该计数器是64位的无符号整数(uint64_t),该计数器由eventfd函数的initval参数进行初始化。
在Linux2.6.26之前的版本,flags参数是无用的,必须置为0。从Linux2.6.27开始,可以在flags中指定下面的值以改变eventfd的行为:
EFD_NONBLOCK:在新建的文件描述符上设置O_NONBLOCK文件状态标志;
EFD_CLOEXEC:在新建的文件描述符上设置FD_CLOEXEC标志。
eventfd返回的文件描述符可以执行下面的操作:
read:如果eventfd的计数器的值非0,则read操作在一个8字节缓冲区中返回该值,并且将计数器重置为0,返回的值是主机字节序。如果计数器的值为0,若文件描述符设置为非阻塞的话,则read返回EAGAIN错误,否则,read一直阻塞,直到计数器非0。
如果提供给read的缓冲区长度小于8字节的话,read返回EINVAL错误。
write:write操作会将buffer中的8字节整数加到计数器上。计数器的最大值是64位无符号整数的最大值减1,也就是0xfffffffffffffffe。如果增加计数器的值会超过该最大值,若文件描述符设置成非阻塞的话,则write返回EAGAIN错误,否则,write操作会一直阻塞,直到在该文件描述符上执行了read操作。
如果write缓冲区的大小小于8字节,或者直接写0xffffffffffffffff的话,write将返回EINVAL错误。也就是说,下面的代码,不管计数器的当前值是多少,write操作都会返回EINVAL错误:
unsigned char buf[8] = {0};
memset(buf, 0xff, 8);
write(efd, buf, 8);
poll、select、epoll等:eventfd创建的文件描述符支持poll、epoll和select等操作:
如果计数器的值大于0,则该文件描述符可读;
如果在该描述符上至少能非阻塞的写入1,则该描述符就是可写的;
如果检测到计数器发生了溢出,则select会指示该文件描述符可读且可写;poll会返回POLLERR事件。上面提到的write操作不可能会使得计数器溢出。然而,KAIO子系统执行的2^64 eventfd “signal posts”操作会导致计数器溢出。这种情况,尽管理论上是可能的,但实际上却不太可能。如果确实发生了溢出,则read操作将会返回0xffffffffffffffff。
close:当不再需要该文件描述符时,应该使用close关闭。与其他文件描述符一样,如果底层eventfd对象关联的所有文件描述符都关闭之后,则该对象的资源就会被内核释放掉。
与其他文件描述符类似,fork之后,子进程拥有父进程的eventfd文件描述符的副本,它们指向相同的底层eventfd对象。执行execve之后,除非设置了close-on-exec标志,否则eventfd创建的描述符在子进程中依然保留。
eventfd调用成功,返回新的eventfd文件描述符,失败时返回-1,并设置相应的errno。
eventfd是linux特有的,自linux2.6.22才开始引入,在glibc的2.8版本中开始被支持。
在应用程序使用管道处理信号事件的所有场景中,可以用eventfd替换pipe。内核使用eventfd文件描述符的负载要比pipe低很多,并且仅仅使用一个描述符即可,而不是像pipe那样使用两个。
在内核中使用eventfd时,eventfd文件描述符起到了桥接内核和用户空间的作用,允许像KAIO(内核异步IO)这样的功能给一个文件描述符发送某些操作已经完成的信号。
eventfd文件描述符的关键优势在于它能够用于select、poll或者epoll。这样,应用程序可以同时监听常规文件和支持eventfd接口的其他内核机制的可读性。
例子:
#define handle_error(msg) \
do { perror(msg); exit(EXIT_FAILURE); } while (0) int main(int argc, char *argv[])
{
int efd, j;
uint64_t u;
ssize_t s; if (argc < 2)
{
fprintf(stderr, "Usage: %s <num1> <num2>...\n", argv[0]);
exit(EXIT_FAILURE);
} efd = eventfd(0, 0);
if (efd == -1)
handle_error("eventfd"); switch (fork())
{
case 0:
for (j = 1; j < argc; j++)
{
printf("Child writing %s to efd\n", argv[j]);
u = strtoull(argv[j], NULL, 0);
s = write(efd, &u, sizeof(uint64_t));
if (s != sizeof(uint64_t))
handle_error("write");
}
printf("Child completed write loop\n"); exit(EXIT_SUCCESS); default:
wait(); printf("Parent about to read\n");
s = read(efd, &u, sizeof(uint64_t));
if (s != sizeof(uint64_t))
handle_error("read"); printf("Parent read %llu (0x%llx) from efd\n",
(unsigned long long) u, (unsigned long long) u);
exit(EXIT_SUCCESS); case -1:
handle_error("fork");
}
}
结果:
# ./a.out 11 23 1
Child writing 11 to efd
Child writing 23 to efd
Child writing 1 to efd
Child completed write loop
Parent about to read
Parent read 35 (0x23) from efd
Libev源码分析07:Linux下的eventfd简介的更多相关文章
- [转]Libev源码分析 -- 整体设计
Libev源码分析 -- 整体设计 libev是Marc Lehmann用C写的高性能事件循环库.通过libev,可以灵活地把各种事件组织管理起来,如:时钟.io.信号等.libev在业界内也是广受好 ...
- QTimer源码分析(以Windows下实现为例)
QTimer源码分析(以Windows下实现为例) 分类: Qt2011-04-13 21:32 5026人阅读 评论(0) 收藏 举报 windowstimerqtoptimizationcallb ...
- Libev源码分析09:select突破处理描述符个数的限制
众所周知,Linux下的多路复用函数select采用描述符集表示处理的描述符.描述符集的大小就是它所能处理的最大描述符限制.通常情况下该值为1024,等同于每个进程所能打开的描述符个数. 增大描述符集 ...
- SpringAOP使用及源码分析(SpringBoot下)
一.SpringAOP应用 先搭建一个SpringBoot项目 <?xml version="1.0" encoding="UTF-8"?> < ...
- Spring AMQP 源码分析 07 - MessageListenerAdapter
### 准备 ## 目标 了解 Spring AMQP 如何用 POJO 处理消息 ## 前置知识 <Spring AMQP 源码分析 04 - MessageListener> ## 相 ...
- 内核源码分析之linux内核栈(基于3.16-rc4)
在3.16-rc4内核源码中,内核给每个进程分配的内核栈大小为8KB.这个内核栈被称为异常栈,在进程的内核空间运行时或者执行异常处理程序时,使用的都是异常栈,看下异常栈的代码(include/linu ...
- JDK源码分析(三)——HashMap 下(基于JDK8)
目录 概述 内部字段及构造方法 哈希值与索引计算 存储元素 扩容 删除元素 查找元素 总结 概述 在上文我们基于JDK7分析了HashMap的实现源码,介绍了HashMap的加载因子loadFac ...
- Libev源码分析01:Libev中的监视器结构(C结构体实现继承)
在Libev的源码中,用到了一种用C实现类似C++中继承的技巧,主要是用宏和结构体实现. 在Libev中,最关键的数据结构就是各种监视器,比如IO监视器,信号监视器等等.这些监视器的多数成员都是一样的 ...
- Libev源码分析06:异步信号同步化--sigwait、sigwaitinfo、sigtimedwait和signalfd
一:信号简述 信号是典型的异步事件.内核在某个信号出现时有三种处理方式: a:忽略信号,除了SIGKILL和SIGSTOP信号不能忽略外,其他大部分信号都可以被忽略: b:捕捉信号,也就是在信号发生时 ...
随机推荐
- P3303 [SDOI2013]淘金
题目描述 小Z在玩一个叫做<淘金者>的游戏.游戏的世界是一个二维坐标.X轴.Y轴坐标范围均为1..N.初始的时候,所有的整数坐标点上均有一块金子,共N*N块. 一阵风吹过,金子的位置发生了 ...
- 举例分析private的作用【c/c++学习】
抛砖引玉: c++中private的用处 我知道我们可以用 public 中的值,把private中的数据给提出来,但是还是搞不懂private该怎么用,或者说在一个具体程序中,private有什么用 ...
- php7不再支持HTTP_RAW_POST_DATA,微信支付$GLOBALS[‘HTTP_RAW_POST_DATA’]获取不到数据,
升级到php7后, 发现旧的web系统有些问题, 查看后才发现原来是php7不再支持HTTP_RAW_POST_DATA 原来系统一些地方, 使用$GLOBALS[‘HTTP_RAW_POST_DAT ...
- 使用fast-json-stringify代替JSON.stringify
使用JSON.stringify的思考 使用过JSON对象的程序员最常做的一项工作便是,将JSON对象转化为字符串.该字符串的用途很多,例如可以使用在WEB的URL中,在多个页面间进行传递. cons ...
- Leetcode695.Max Area of Island岛屿的最大面积
给定一个包含了一些 0 和 1的非空二维数组 grid , 一个 岛屿 是由四个方向 (水平或垂直) 的 1 (代表土地) 构成的组合.你可以假设二维矩阵的四个边缘都被水包围着. 找到给定的二维数组中 ...
- php array_key_exists() 与 isset() 的区别
一个基本的区别是isset()可用于数组和变量,而array_key_exits()只能用于数组. 但是最主要的区别在于在设定的条件下的返回值. 现在我们来验证一下这个最主要的区别. array_ke ...
- 基于spring-boot的测试桩设计--几种常见的controller
第一种:通过@RequestBody,直接将请求体映射到对象 //@RequestBody @RequestMapping(value = "addUser", method = ...
- 一.JDBC学习入门
一.JDBC相关概念介绍 1.1.数据库驱动 这里的驱动的概念和平时听到的那种驱动的概念是一样的,比如平时购买的声卡,网卡直接插到计算机上面是不能用的,必须要安装相应的驱动程序之后才能够使用声卡和网卡 ...
- 基于OPNET的路由协议仿真教程(AODV、OLSR 、DSR等)
前言: 目前由于项目需要,学习了基于opnet的网络仿真方法,发现该软件的学习资料少之又少,所以将自己搜集到的学习资料进行整理,希望能帮助后来的人. 主要参考资料:OPNET网络仿真(清华陈敏版) 仿 ...
- 简单利用XSS获取Cookie信息实例演示
简单利用XSS获取Cookie信息实例演示 首先要找到一个有XXS的站,这里就不整什么大站了,谷歌一下inurl:'Product.asp?BigClassName',搜出来的命中率也比较高.随便 ...