Linux eventfd 原理与实践

本文摘自写给应用开发的 Android Framework 教程 ,完整教程请查阅 yuandaimaahao.github.io/AndroidFram... 更为详细的视频教程与答疑服务,请联系微信 zzh0838

1. 基本原理

eventfd 是 Linux 2.6.27 添加了一个新的特性,用来实现多进程或多线程的之间的事件通知的,也可以由内核通知用户空间应用程序。

eventfd 通过一个进程间共享的 64 位计数器完成进程间通信,这个计数器在 linux 内核空间维护,用户可以通过调用 write 方法向内核空间写入一个 64 位的值,也可以调用 read 方法读取这个值。

eventfd 有三个常用的操作:

  • 初始化:给内核计数器赋初始值,一般为 0
  • 写操作:递增内核计数器
  • 读操作:当内核计数器为 0 时,读操作阻塞,当内核计数器大于 0 时,读操作不会阻塞

2. eventfd 的使用

2.1 初始化

使用 eventfd 函数来进行初始化:

cpp 复制代码
#include <sys/eventfd.h>
int eventfd(unsigned int initval, int flags);
  • initval:创建eventfd时它所对应的 64 位计数器的初始值;
  • flags: eventfd文件描述符的标志,用以改变 eventfd 的行为,大多情况设置为 0 即可。
    • 如果是2.6.26或之前版本的内核,flags 必须设置为 0
    • EFD_CLOEXEC (since Linux 2.6.27):文件被设置成 O_CLOEXEC,创建子进程 (fork) 时不继承父进程的文件描述符
    • EFD_NONBLOCK(since Linux 2.6.27):设置文件描述符为非阻塞的,设置了这个标志后,如果没有数据可读,就返回一个 EAGAIN 错误,不会一直阻塞。
    • EFD_SEMAPHORE (since Linux 2.6.30):提供类似信号量语义的 read 操作,简单说就是计数值 count 递减 1。可以多次 read

举个例子:

cpp 复制代码
int efd = eventfd(0, 0);
if (efd == -1)
{
    handle_error("eventfd");
} 

2.2 读写操作

初始化后的 eventfd,在内核中有一个 64 位的整数与之对应,我们可以通过 read/write 系统调用来修改这个数字。

write 的时候,累加计数:

c 复制代码
uint64_t u = 1;
ssize_t n;

// 写 eventfd,必须是 64 位整数
// 内部计数器变为 1
n = write(efd, &u, sizeof(uint64_t));

// 内部计数器变为 3
u = 2;
n = write(efd, &u, sizeof(uint64_t));

// 内部计数器变为 6
u = 3;
n = write(efd, &u, sizeof(uint64_t));

read 的时候,读出计数器的值,并将内核中的计数器赋值为 0:

c 复制代码
n = read(efd, &u, sizeof(uint64_t));

参考资料

相关推荐
逐光老顽童5 小时前
Java 与 Kotlin 混合开发避坑指南:30 个真实案例实录
android·kotlin
爱勇宝17 小时前
鸿蒙生态的下半场:开发者不只要能开发,还要能赚钱
android·前端·程序员
Yeyu20 小时前
刷新一帧的艺术:invalidate / postInvalidate / postInvalidateOnAnimation全解析
android
潘潘潘1 天前
Android OTA 升级原理和流程介绍
android
plainGeekDev1 天前
null 判断 → Kotlin 可空类型
android·java·kotlin
plainGeekDev1 天前
getter/setter → Kotlin 属性
android·java·kotlin
YXL1111YXL1 天前
Handler 消息回收与协程异步执行的时序陷阱
android
恋猫de小郭1 天前
KMP / CMP 鸿蒙版本 Beta 发布,他有什么特别之处?
android·前端·flutter
三少爷的鞋1 天前
Android 协程并发控制:别动线程池,控制好并发语义就够了
android
weiggle2 天前
第七篇:状态提升与单向数据流——架构设计的核心
android