转载请注明来源:https://www.cnblogs.com/hookjc/

eventfd 在内核版本,2.6.22以后有效。查看内核版本可以用命令 uname -r 。
  1. #include<sys/eventfd.h>
  2. int eventfd(unsigned int initval,int flags);

这个函数会创建一个 事件对象 (eventfd object), 用来实现,进程(线程)间的等待/通知(wait/notify) 机制. 内核会为这个对象维护一个64位的计数器(uint64_t)。
并且使用第一个参数(initval)初始化这个计数器。调用这个函数就会返回一个新的文件描述符(event object)。2.6.27版本开始可以按位设置第二个参数(flags)。
有如下的一些宏可以使用:

EFD_NONBLOCK , 功能同open(2) 的O_NONBLOCK,设置对象为非阻塞状态,如果没有设置这个状态的话,read(2)读eventfd,并且计数器的值为0 就一直堵塞在read调用当中,要是设置了这个标志, 就会返回一个 EAGAIN 错误(errno = EAGAIN)。效果也如同 额外调用select(2)达到的效果。

EFD_CLOEXEC 我的理解是,这个标识被设置的话,调用exec后会自动关闭文件描述符,防止泄漏。

如果是2.6.26或之前版本的内核,flags 必须设置为0。

创建这个对象后,可以对其做如下操作。

write 将缓冲区写入的8字节整形值加到内核计数器上。

read 读取8字节值, 并把计数器重设为0. 如果调用read的时候计数器为0, 要是eventfd是阻塞的, read就一直阻塞在这里,否则就得到 一个EAGAIN错误。
如果buffer的长度小于8那么read会失败, 错误代码被设置成 EINVAL。

poll select epoll

close 当不需要eventfd的时候可以调用close关闭, 当这个对象的所有句柄都被关闭的时候,内核会释放资源。 为什么不是close就直接释放呢, 如果调用fork 创建

进程的时候会复制这个句柄到新的进程,并继承所有的状态。

(ps:也就是说,在write之后没有read,但是又write新的数据,那么读取的是这两次的8个字节的和,在read之后再write,可以完成read和write之间的交互)

一个例子:

  1. #include <stdio.h>
  2. #include <unistd.h>
  3. #include <sys/time.h>
  4. #include <stdint.h>
  5. #include <pthread.h>
  6. #include <sys/eventfd.h>
  7. #include <sys/epoll.h>
  8. int efd = -1;
  9. void *read_thread(void *dummy)
  10. {
  11. int ret = 0;
  12. uint64_t count = 0;
  13. int ep_fd = -1;
  14. struct epoll_event events[10];
  15. if (efd < 0)
  16. {
  17. printf("efd not inited.\n");
  18. goto fail;
  19. }
  20. ep_fd = epoll_create(1024);
  21. if (ep_fd < 0)
  22. {
  23. perror("epoll_create fail: ");
  24. goto fail;
  25. }
  26. {
  27. struct epoll_event read_event;
  28. read_event.events = EPOLLHUP | EPOLLERR | EPOLLIN;
  29. read_event.data.fd = efd;
  30. ret = epoll_ctl(ep_fd, EPOLL_CTL_ADD, efd, &read_event);
  31. if (ret < 0)
  32. {
  33. perror("epoll ctl failed:");
  34. goto fail;
  35. }
  36. }
  37. while (1)
  38. {
  39. ret = epoll_wait(ep_fd, &events[0], 10, 5000);
  40. if (ret > 0)
  41. {
  42. int i = 0;
  43. for (; i < ret; i++)
  44. {
  45. if (events[i].events & EPOLLHUP)
  46. {
  47. printf("epoll eventfd has epoll hup.\n");
  48. goto fail;
  49. }
  50. else if (events[i].events & EPOLLERR)
  51. {
  52. printf("epoll eventfd has epoll error.\n");
  53. goto fail;
  54. }
  55. else if (events[i].events & EPOLLIN)
  56. {
  57. int event_fd = events[i].data.fd;
  58. ret = read(event_fd, &count, sizeof(count));
  59. if (ret < 0)
  60. {
  61. perror("read fail:");
  62. goto fail;
  63. }
  64. else
  65. {
  66. struct timeval tv;
  67. gettimeofday(&tv, NULL);
  68. printf("success read from efd, read %d bytes(%llu) at %lds %ldus\n",
  69. ret, count, tv.tv_sec, tv.tv_usec);
  70. }
  71. }
  72. }
  73. }
  74. else if (ret == 0)
  75. {
  76. /* time out */
  77. printf("epoll wait timed out.\n");
  78. break;
  79. }
  80. else
  81. {
  82. perror("epoll wait error:");
  83. goto fail;
  84. }
  85. }
  86. fail:
  87. if (ep_fd >= 0)
  88. {
  89. close(ep_fd);
  90. ep_fd = -1;
  91. }
  92. return NULL;
  93. }
  94. int main(int argc, char *argv[])
  95. {
  96. pthread_t pid = 0;
  97. uint64_t count = 0;
  98. int ret = 0;
  99. int i = 0;
  100. efd = eventfd(0, 0);
  101. if (efd < 0)
  102. {
  103. perror("eventfd failed.");
  104. goto fail;
  105. }
  106. ret = pthread_create(&pid, NULL, read_thread, NULL);
  107. if (ret < 0)
  108. {
  109. perror("pthread create:");
  110. goto fail;
  111. }
  112. for (i = 0; i < 5; i++)
  113. {
  114. count = 4;
  115. ret = write(efd, &count, sizeof(count));
  116. if (ret < 0)
  117. {
  118. perror("write event fd fail:");
  119. goto fail;
  120. }
  121. else
  122. {
  123. struct timeval tv;
  124. gettimeofday(&tv, NULL);
  125. printf("success write to efd, write %d bytes(%llu) at %lds %ldus\n",
  126. ret, count, tv.tv_sec, tv.tv_usec);
  127. }
  128. sleep(1);
  129. }
  130. fail:
  131. if (0 != pid)
  132. {
  133. pthread_join(pid, NULL);
  134. pid = 0;
  135. }
  136. if (efd >= 0)
  137. {
  138. close(efd);
  139. efd = -1;
  140. }
  141. return ret;
  142. }

  1. success write to efd, write 8 bytes(4) at 1328805612s 21939us
  2. success read from efd, read 8 bytes(4) at 1328805612s 21997us
  3. success write to efd, write 8 bytes(4) at 1328805613s 22247us
  4. success read from efd, read 8 bytes(4) at 1328805613s 22287us
  5. success write to efd, write 8 bytes(4) at 1328805614s 22462us
  6. success read from efd, read 8 bytes(4) at 1328805614s 22503us
  7. success write to efd, write 8 bytes(4) at 1328805615s 22688us
  8. success read from efd, read 8 bytes(4) at 1328805615s 22726us
  9. success write to efd, write 8 bytes(4) at 1328805616s 22973us
  10. success read from efd, read 8 bytes(4) at 1328805616s 23007us
  11. epoll wait timed out.

同时他也是支持进程间通信的,过程和这个差不多。

来源:python脚本自动迁移

linux内核中的eventfd的更多相关文章

  1. Linux 内核中的 Device Mapper 机制

    本文结合具体代码对 Linux 内核中的 device mapper 映射机制进行了介绍.Device mapper 是 Linux 2.6 内核中提供的一种从逻辑设备到物理设备的映射框架机制,在该机 ...

  2. 向linux内核中添加外部中断驱动模块

    本文主要介绍外部中断驱动模块的编写,包括:1.linux模块的框架及混杂设备的注册.卸载.操作函数集.2.中断的申请及释放.3.等待队列的使用.4.工作队列的使用.5.定时器的使用.6.向linux内 ...

  3. Linux内核中双向链表的经典实现

    概要 前面一章"介绍双向链表并给出了C/C++/Java三种实现",本章继续对双向链表进行探讨,介绍的内容是Linux内核中双向链表的经典实现和用法.其中,也会涉及到Linux内核 ...

  4. Linux内核中的fastcall和asmlinkage宏

    代码中看见:#define _fastcall 所以了解下fastcall -------------------------------------------------------------- ...

  5. Linux内核中的GPIO系统之(3):pin controller driver代码分析

    一.前言 对于一个嵌入式软件工程师,我们的软件模块经常和硬件打交道,pin control subsystem也不例外,被它驱动的硬件叫做pin controller(一般ARM soc的datash ...

  6. (十)Linux内核中的常用宏container_of

    Container_of在Linux内核中是一个常用的宏,用于从包含在某个结构中的指针获得结构本身的指针,通俗地讲就是通过结构体变量中某个成员的首地址进而获得整个结构体变量的首地址. Containe ...

  7. Apparmor——Linux内核中的强制访问控制系统

      AppArmor 因为最近在研究OJ(oline judge)后台的安全模块的实现,所以一直在研究Linux下沙箱的东西,同时发现了Apparmor可以提供访问控制. AppArmor(Appli ...

  8. KSM剖析——Linux 内核中的内存去耦合

    简介: 作为一个系统管理程序(hypervisor),Linux® 有几个创新,2.6.32 内核中一个有趣的变化是 KSM(Kernel Samepage Merging)  允许这个系统管理程序通 ...

  9. linux内核中的get_user和put_user

    linux内核中的get_user和put_user 在 内核空间和用户空间交换数据时,get_user和put_user是两个两用的函数.相对于copy_to_user和 copy_from_use ...

随机推荐

  1. CS5268替代AG9321MCQ 替代AG9321方案 TYPEC转HDMI多功能拓展坞

    台湾安格AG9321MCQ是一款TYPEC拓展坞产品方案,他集中了TYPEC 转HDMI  VGA  PD3.0快充  QC3.0数据传输 I2S接口的音频DAC输出以及可以各种读卡器功能. Caps ...

  2. 台湾旺玖MA8601|USB HUB方案|MA8601测试版

    MA8601是USB 2.0高速4端口集线器控制器的高性能解决方案,完全符合通用串行总线规范2.0.MA8601继承了先进的串行接口技术,当4个DS(下游)端口同时工作时,功耗最低. MA8601采用 ...

  3. Reflection 基础知识(一)

    反射机制的定义 反射通常指在程序在运行时能够获取自身的信息. 静态语言反射 在java中使用反射的一个例子 Class<?> clazz = Class.forName("com ...

  4. Java Web程序设计笔记 • 【第10章 JSTL标签库】

    全部章节   >>>> 本章目录 10.1 JSTL 概述 10.1.1 JSTL 简介 10.1.1 JSTL 使用 10.1.2 实践练习 10.2 核心标签库 10.2. ...

  5. 在 jQuery 中使用滑入滑出动画效果,实现二级下拉导航菜单的显示与隐藏效果

    查看本章节 查看作业目录 需求说明: 在 jQuery 中使用滑入滑出动画效果,实现二级下拉导航菜单的显示与隐藏效果 用户将光标移动到"最新动态页"或"帮助查询" ...

  6. JZOJ5405 & AtCoder Grand Contest 001 F. Permutation

    题目大意 给出一个长度为\(n\)的排列\(P\)与一个正整数\(k\). 你需要进行如下操作任意次, 使得排列\(P\)的字典序尽量小. 对于两个满足\(|i-j|>=k\) 且\(|P_i- ...

  7. Jenkins_忘记管理员密码的处理方法

    1.查看jenkins配置存放目录 2.修改config.xml的useSecurity的true为flase 3.重启jenkins服务 4.进入jenkins,不输入密码直接就进入了jenkins ...

  8. 关于 this.$route.meta.operations.includes('delete') 取不到值的问题

    原因是:src/mock/api/sys.login.js中定义的路径 要与src/router/modules/下定义的路由要一致 作用this.$route.matched可以查看匹配信息 来自为 ...

  9. mysql数据库忘记root密码怎么办?

    mysql数据库忘记root密码怎么破解和修改 1.停止数据库的运行 [root@localhost ~]# /etc/init.d/mysqld stop 或者[root@localhost ~]# ...

  10. Texture+PBR两种工作流程

    一.导入Texture 1.Inpspector TextureSize 2的n次幂,底层图形学需要,计算更快:不使用2的倍数,系统也会添加像素补全2n: 有最大尺寸限制8k,cubemap最高4k: ...