线程间通信之eventfd

线程间通信之eventfd

man手册中的解释:
eventfd()创建了一个“eventfd对象”, 通过它能够实现用户态程序间(我觉得这里主要指线程而非进程)的等待/通知机制,以及内核态向用户态通知的机制(未考证)。
此对象包含了一个被内核所维护的计数(uint64_t), 初始值由initval来决定。


int eventfd(unsigned int initval, int flags);创建一个eventfd文件描述符
int eventfd_read(int fd, eventfd_t *value); 向eventfd中写入一个值
int eventfd_write(int fd, eventfd_t value); 从eventfd中读出一个值

例一、子线程多次写入多个值,主线程一次读出所有值的和

复制代码

 1 #include <sys/eventfd.h>2 #include <unistd.h>3 #include <stdlib.h>4 #include <stdio.h>5 #include <stdint.h>  6 7 int main(int argc, char**argv[])8 {9     int efd, j;
10     uint64_t u;
11     ssize_t s;
12     
13     if (argc < 2)
14     {
15         printf("number of argc is wrong!\n");
16         return 0;
17     }
18     
19     efd = eventfd(0,0);
20     if (-1 == efd)
21     {
22         printf("failed to create eventfd\n");
23     }
24     
25     switch(fork())
26     {
27         case 0:
28         {
29             for(j=1; j<argc;j++)
30             {
31                 printf("child writing %s to efd\n", argv[j]);
32                 u = strtoull(argv[j], NULL, 0);
33                 s = write(efd, &u, sizeof(uint64_t));
34                 if (s!=sizeof(uint64_t))
35                 {
36                     printf("write efd failed\n");
37                 }
38             }
39             printf("Child completed write loop\n");
40             exit(0);
41         }
42         default:
43             sleep(2);
44             printf("Parents about to read\n");
45             s = read(efd, &u, sizeof(uint64_t));
46             if (s != sizeof(uint64_t))
47             {
48                 printf("read efd failed\n");
49             }
50             printf("Parents first read %llu (0x%llx) from efd\n", u, u);
51             exit(0);
52         case -1:
53         {
54             printf("fork error\n");
55         }
56     }
57     
58     return 0;
59 }
60 
61 运行结果
62 kane@kanelinux:/mnt/hgfs/kanelinuxshare/eventfd$ ./a.out 1 2 3 4
63 child writing 1 to efd
64 child writing 2 to efd
65 child writing 3 to efd
66 child writing 4 to efd
67 Child completed write loop
68 Parents about to read
69 Parents first read 10 (0xa) from efd
70 
71 如果有写入操作,但是并没有导致初始值变化,则主线程会一直挂在read操作上
72 kane@kanelinux:/mnt/hgfs/kanelinuxshare/eventfd$ ./a.out 0 0 0 0
73 child writing 0 to efd
74 child writing 0 to efd
75 child writing 0 to efd
76 child writing 0 to efd
77 Child completed write loop
78 Parents about to read
79 ^C

复制代码

 

例二、eventfd可以被epoll监控, 一旦有状态变化,可以触发通知

复制代码

  1 #include <sys/eventfd.h>2 #include <unistd.h>3 #include <stdlib.h>4 #include <stdio.h>5 #include <stdint.h>  6 #include <sys/epoll.h>  7 #include <string.h>  8 #include <pthread.h>  9 10 int g_iEvtfd = -1;11 12 void *eventfd_child_Task(void *pArg)13 {14     uint64_t uiWrite = 1;15     16     while(1)17     {18         sleep(2);19         if (0 != eventfd_write(g_iEvtfd, uiWrite))20         {21             printf("child write iEvtfd failed\n");22         }    23     }24 25     return;26 }27 28 int main(int argc, char**argv[])29 {30     int iEvtfd, j;31     uint64_t uiWrite = 1;32     uint64_t uiRead;33     ssize_t s;34     int iEpfd;35     struct epoll_event stEvent;36     int iRet = 0;37     struct epoll_event stEpEvent;38     pthread_t stWthread;39     40     iEpfd = epoll_create(1);41     if (-1 == iEpfd)42     {43         printf("Create epoll failed.\n");44         return 0;45     }46     47     iEvtfd = eventfd(0,0);48     if (-1 == iEvtfd)49     {50         printf("failed to create eventfd\n");51         return 0;52     }53     54     g_iEvtfd = iEvtfd;55     56     memset(&stEvent, 0, sizeof(struct epoll_event));57     stEvent.events = (unsigned long) EPOLLIN;58     stEvent.data.fd = iEvtfd;59     iRet = epoll_ctl(iEpfd, EPOLL_CTL_ADD, g_iEvtfd, &stEvent);60     if (0 != iRet)61     {62         printf("failed to add iEvtfd to epoll\n");63         close(g_iEvtfd);64         close(iEpfd);65         return 0;66     }67     68     iRet = pthread_create(&stWthread, NULL, eventfd_child_Task, NULL);69     if (0 != iRet)70     {71         close(g_iEvtfd);72         close(iEpfd);73         return;74     }75     76     for(;;)77     {78         iRet = epoll_wait(iEpfd, &stEpEvent, 1, -1);79         if (iRet > 0)80         {81             s = eventfd_read(iEvtfd, &uiRead);82             if (s != 0)83             {84                 printf("read iEvtfd failed\n");85                 break;86             }87             printf("Read %llu (0x%llx) from iEvtfd\n", uiRead, uiRead);88         }89     }90     91     close(g_iEvtfd);92     close(iEpfd);93     return 0;94 }95 运行结果96 kane@kanelinux:/mnt/hgfs/kanelinuxshare/eventfd$ ./a.out97 Read 1 (0x1) from iEvtfd98 Read 1 (0x1) from iEvtfd99 Read 1 (0x1) from iEvtfd
100 Read 1 (0x1) from iEvtfd
101 Read 1 (0x1) from iEvtfd
102 Read 1 (0x1) from iEvtfd
103 Read 1 (0x1) from iEvtfd
104 Read 1 (0x1) from iEvtfd
105 ^C

复制代码

例三、被epoll监控的eventfd,如果在子线程中被多次写入,在主线程中是怎么读的?

复制代码

#include <sys/eventfd.h>
#include <unistd.h>
#include <stdlib.h>
#include <stdio.h>
#include <stdint.h>  
#include <sys/epoll.h>  
#include <string.h>  
#include <pthread.h>  int g_iEvtfd = -1;void *eventfd_child_Task(void *pArg)
{uint64_t uiWrite = 1;while(1){sleep(2);eventfd_write(g_iEvtfd, uiWrite);eventfd_write(g_iEvtfd, uiWrite);}return;
}int main(int argc, char**argv[])
{int iEvtfd, j;uint64_t uiWrite = 1;uint64_t uiRead;ssize_t s;int iEpfd;struct epoll_event stEvent;int iRet = 0;struct epoll_event stEpEvent;pthread_t stWthread;iEpfd = epoll_create(1);if (-1 == iEpfd){printf("Create epoll failed.\n");return 0;}iEvtfd = eventfd(0,0);if (-1 == iEvtfd){printf("failed to create eventfd\n");return 0;}g_iEvtfd = iEvtfd;memset(&stEvent, 0, sizeof(struct epoll_event));stEvent.events = (unsigned long) EPOLLIN;stEvent.data.fd = iEvtfd;iRet = epoll_ctl(iEpfd, EPOLL_CTL_ADD, g_iEvtfd, &stEvent);if (0 != iRet){printf("failed to add iEvtfd to epoll\n");close(g_iEvtfd);close(iEpfd);return 0;}iRet = pthread_create(&stWthread, NULL, eventfd_child_Task, NULL);if (0 != iRet){close(g_iEvtfd);close(iEpfd);return;}for(;;){iRet = epoll_wait(iEpfd, &stEpEvent, 1, -1);if (iRet > 0){s = eventfd_read(iEvtfd, &uiRead);if (s != 0){printf("read iEvtfd failed\n");break;}printf("Read %llu (0x%llx) from iEvtfd\n", uiRead, uiRead);}}close(g_iEvtfd);close(iEpfd);return 0;
}运行结果:
kane@kanelinux:/mnt/hgfs/kanelinuxshare/eventfd$ ./a.out
Read 1 (0x1) from iEvtfd
Read 1 (0x1) from iEvtfdRead 1 (0x1) from iEvtfd
Read 1 (0x1) from iEvtfdRead 1 (0x1) from iEvtfd
Read 1 (0x1) from iEvtfdRead 1 (0x1) from iEvtfd
Read 1 (0x1) from iEvtfd^C

复制代码

例一中并没有epoll做监控,
因此在read前,如果eventfd被写多次,在read的时候也是一次全部读出。

 

注:eventfd中的SEMAPHORE标志用法

* If EFD_SEMAPHORE was not specified and the eventfd counter
has a nonzero value, then a read(2) returns 8 bytes contain‐
ing that value, and the counter's value is reset to zero.

* If EFD_SEMAPHORE was specified and the eventfd counter has a
nonzero value, then a read(2) returns 8 bytes containing the
value 1, and the counter's value is decremented by 1.

通过测试发现。如果eventfd在创建的时候传入EFD_SEMAPHORE 标志,则会按上面man手册中提到的那样,每次在eventfd_read的时候只减一,并不是把值一次性全部读出。见下例 :

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/384983.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

定时器timerfd

1.为什么要加入此定时器接口 linux2.6.25版本新增了timerfd这个供用户程序使用的定时接口&#xff0c;这个接口基于文件描述符&#xff0c;当超时事件发生时&#xff0c;该文件描述符就变为可读。我首次接触这个新特性是在muduo网络库的定时器里看到的&#xff0c;那么新增一个…

timerfd与epoll

linux timerfd系列函数总结 网上关于timerfd的文章很多&#xff0c;在这儿归纳总结一下方便以后使用&#xff0c;顺便贴出一个timerfd配合epoll使用的简单例子 一、timerfd系列函数 timerfd是Linux为用户程序提供的一个定时器接口。这个接口基于文件描述符&#xff0c;通过文…

linux僵尸进程产生的原因以及如何避免产生僵尸进程defunct

给进程设置僵尸状态的目的是维护子进程的信息&#xff0c;以便父进程在以后某个时间获取。这些信息包括子进程的进程ID、终止状态以及资源利用信息(CPU时间&#xff0c;内存使用量等等)。如果一个进程终止&#xff0c;而该进程有子进程处于僵尸状态&#xff0c;那么它的所有僵尸…

linux下僵尸进程(Defunct进程)的产生与避免

在测试基于 DirectFBGstreamer 的视频联播系统的一个 Demo 的时候&#xff0c;其中大量使用 system 调用的语句&#xff0c;例如在 menu 代码中的 system("./play") &#xff0c;而且多次执行&#xff0c;这种情况下&#xff0c;在 ps -ef 列表中出现了大量的 defunc…

读过的最好的epoll讲解

首先我们来定义流的概念&#xff0c;一个流可以是文件&#xff0c;socket&#xff0c;pipe等等可以进行I/O操作的内核对象。 不管是文件&#xff0c;还是套接字&#xff0c;还是管道&#xff0c;我们都可以把他们看作流。 之后我们来讨论I/O的操作&#xff0c;通过read&#xf…

C语言指针转换为intptr_t类型

C语言指针转换为intptr_t类型 1、前言 今天在看代码时&#xff0c;发现将之一个指针赋值给一个intptr_t类型的变量。由于之前没有见过intptr_t这样数据类型&#xff0c;凭感觉认为intptr_t是int类型的指针。感觉很奇怪&#xff0c;为何要将一个指针这样做呢&#xff1f;如是果…

北京加密机现场select问题

问题描述 北京项目通过调用我们提供的库libsigxt.a与加密机通信&#xff0c;c/s架构&#xff0c;客户端启用多个线程&#xff0c;每个线程流程有以下三步&#xff0c;连接加密机&#xff0c;签名&#xff0c;关闭链接。在正常运行一段时间后会出现不能连接加密机服务问题。 连…

处理SIGCHLD信号

在上一讲中&#xff0c;我们使用fork函数得到了一个简单的并发服务器。然而&#xff0c;这样的程序有一个问题&#xff0c;就是当子进程终止时&#xff0c;会向父进程发送一个SIGCHLD信号&#xff0c;父进程默认忽略&#xff0c;导致子进程变成一个僵尸进程。僵尸进程一定要处理…

nginx源码阅读(一).综述

前言 nginx作为一款开源的轻量级高性能web服务器,是非常值得立志从事服务端开发方向的人学习的。现今nginx的最新版本是nginx-1.13.6,代码量也日渐庞大,但是由于其核心思想并没改变,为了降低阅读难度,我选择的是nginx-1.0.15版本,并且由于时间和水平有限,重点关注的是nginx的启…

系统级性能分析工具perf的介绍与使用

系统级性能优化通常包括两个阶段&#xff1a;性能剖析&#xff08;performance profiling&#xff09;和代码优化。 性能剖析的目标是寻找性能瓶颈&#xff0c;查找引发性能问题的原因及热点代码。 代码优化的目标是针对具体性能问题而优化代码或编译选项&#xff0c;以改善软…

C/C++内存问题检查利器——Purify

C&#xff0f;C内存问题检查利器——Purify 一、 引言 我们都知道软件的测试&#xff08;在以产品为主的软件公司中叫做QA—Quality Assessment&#xff09;占了整个软件工程的30% -50%&#xff0c;但有这么一种说法&#xff0c;即使是最优秀测试专家设计出来的测试…

浅析三种特殊进程:孤儿进程,僵尸进程和守护进程

其实有时想想linux内核的设计也蕴含着很多人生哲学,在linux中有这么几个特殊进程中,我们一开始见到它们的名字可能还会觉得很诧异,但在了解完了原理后,我们仔细想想,这样的命名也不无道理!下面我就给大家分别介绍一下这三种特殊的进程! 1.孤儿进程 如果父进程先退出,子进程还没…

差生文具多之(二): perf

栈回溯和符号解析是使用 perf 的两大阻力&#xff0c;本文以应用程序 fio 的观测为例子&#xff0c;提供一些处理它们的经验法则&#xff0c;希望帮助大家无痛使用 perf。 前言 系统级性能优化通常包括两个阶段&#xff1a;性能剖析和代码优化&#xff1a; 性能剖析的目标是寻…

Linux下的I/O复用与epoll详解(ET与LT)

前言 I/O多路复用有很多种实现。在linux上&#xff0c;2.4内核前主要是select和poll&#xff0c;自Linux 2.6内核正式引入epoll以来&#xff0c;epoll已经成为了目前实现高性能网络服务器的必备技术。尽管他们的使用方法不尽相同&#xff0c;但是本质上却没有什么区别。本文将重…

彻底学会使用epoll(一)——ET模式实现分析

注&#xff1a;之前写过两篇关于epoll实现的文章&#xff0c;但是感觉懂得了实现原理并不一定会使用&#xff0c;所以又决定写这一系列文章&#xff0c;希望能够对epoll有比较清楚的认识。是请大家转载务必注明出处&#xff0c;算是对我劳动成果的一点点尊重吧。另外&#xff0…

sys/queue.h分析(图片复制不过来,查看原文)

这两天有兴趣学习使用了下系统头文件sys/queue.h中的链表/队列的实现&#xff0c;感觉实现的很是优美&#xff0c;关键是以后再也不需要自己实现这些基本的数据结构了&#xff0c;哈哈&#xff01; 我的系统环境是 正好需要使用队列&#xff0c;那么本篇就以其中的尾队列&…

线程池原理及C语言实现线程池

备注&#xff1a;该线程池源码参考自传直播客培训视频配套资料&#xff1b; 源码&#xff1a;https://pan.baidu.com/s/1zWuoE3q0KT5TUjmPKTb1lw 密码&#xff1a;pp42 引言&#xff1a;线程池是一种多线程处理形式&#xff0c;大多用于高并发服务器上&#xff0c;它能合理有效…

iptables 的mangle表

mangle表的主要功能是根据规则修改数据包的一些标志位&#xff0c;以便其他规则或程序可以利用这种标志对数据包进行过滤或策略路由。 内网的客户机通过Linux主机连入Internet&#xff0c;而Linux主机与Internet连接时有两条线路&#xff0c;它们的网关如图所示。现要求对内网进…

Linux之静态库

命名规则&#xff1a; lib 库的名字 .a 制作步骤 生成对应.o文件 .c .o 将生成的.o文件打包 ar rcs 静态库的名字&#xff08;libMytest.a&#xff09; 生成的所有的.o 发布和使用静态库&#xff1a; 1&#xff09; 发布静态 2&#xff09; 头文件 文件如下图所示&…

Linux之动态库

命令规则 lib 名字 .so 制作步骤 1&#xff09;生成与位置无关的代码&#xff08;生成与位置无关的代码&#xff09; 2&#xff09;将.o打包成共享库&#xff08;动态库&#xff09; 发布和使用共享库 动态库运行原理&#xff1a; 生成动态库&#xff1a; gcc -fPIC -c *.c -…