当前位置: 首页 > news >正文

epoll_wait未触发的小Bug

上次看了一下epoll监听的原理,在Android Jni里使用epoll,来监听Gpio口的变化事件,具体代码如下:

  1. 打开 GPIO 文件描述符,因为该文件是内核虚拟出来的,不是实际文件,所以无法使用FileObserver来监听,应该是不会触发Inotify
int *fds = malloc(num_gpios * sizeof(int));LOGD("Allocating memory for %d GPIOs", num_gpios);for (int i = 0; i < num_gpios; i++) {jstring gpioPin = (jstring) (*env)->GetObjectArrayElement(env, gpioPins, i);const char *pin = (*env)->GetStringUTFChars(env, gpioPin, NULL);char gpio_value_path[50];snprintf(gpio_value_path, sizeof(gpio_value_path), "/sys/class/gpio/gpio%s/value", pin);LOGD("Opening GPIO path: %s", gpio_value_path);fds[i] = open(gpio_value_path, O_RDONLY);if (fds[i] < 0) {LOGD("Failed to open GPIO %s", pin);(*env)->ReleaseStringUTFChars(env, gpioPin, pin);free(fds);return;}
  1. 创建 epoll 实例,epoll_create1这个函数看着强迫症都犯了。。。
    int epfd = epoll_create1(0);if (epfd < 0) {LOGD("Failed to create epoll instance");free(fds);return;}struct epoll_event ev;for (int i = 0; i < num_gpios; i++) {ev.events = EPOLLPRI | EPOLLERR;  // 等待优先事件和错误事件ev.data.fd = fds[i];if (epoll_ctl(epfd, EPOLL_CTL_ADD, fds[i], &ev) < 0) {LOGD("Failed to add GPIO %d to epoll", i);free(fds);return;}}
  1. 阻塞监听 GPIO 变化,默认会触发一次,可先执行一次 lseek + read 吸掉脏数据。
    这是 epoll 的通用惯例:在 add 到 epoll 前就要清空一次触发状态,否则可能误触发一次后就永远不再触发了。
    while (1) {struct epoll_event events[num_gpios];LOGD("epoll_wai...");int n = epoll_wait(epfd, events, num_gpios, -1);  // 阻塞直到事件发生if (n > 0) {for (int i = 0; i < n; i++) {if (events[i].events & EPOLLPRI) {char gpio_value;lseek(events[i].data.fd, 0, SEEK_SET);read(events[i].data.fd, &gpio_value, 1);LOGD("GPIO fd %d changed! New value: %c", events[i].data.fd, gpio_value);}}

4.监听是否成功触发,和这些几个点相关:

  • /sys/class/gpio/gpioX/edge 的设置(none、rising、falling、both)
  • 内核中对应 gpio_irq_handler
  • 中断触发机制(边沿触发的行为)
  • 用户态 epoll 等待的数据准备逻辑(f_op->poll())

在这里插入图片描述5. 写入 /sys/class/gpio/gpioX/edge,本质上调用了如下路径

gpio_sysfs_set_edge()-> gpio_setup_irq()-> irq_set_irq_type(irq, IRQ_TYPE_EDGE_RISING / IRQ_TYPE_EDGE_FALLING / BOTH)

实际代码如下:
在这里插入图片描述

  1. 中断触发 → 唤醒等待的进程
static irqreturn_t gpio_irq_handler(int irq, void *dev_id) {struct gpio_desc *desc = dev_id;...// 设置状态标志desc->irq_data.triggered = true;// 唤醒 epoll/poll 的 waitqueuewake_up_interruptible(&desc->wait);return IRQ_HANDLED;
}
http://www.xdnf.cn/news/570511.html

相关文章:

  • adb抓包
  • 元宇宙数字人设计大赛:往届获奖作品赏析
  • 公司OA系统中金格iWebOffice2015智能文档中间件不能用了怎么办?
  • 深入解析C++静态成员变量与函数
  • ABC 354
  • Linux上运行程序加载动态库失败
  • Redis语法大全
  • 【Flutter】创建BMI计算器应用并添加依赖和打包
  • 【HTML-5】HTML 实体:完整指南与最佳实践
  • DSP定时器的计算
  • Spring Boot集成Spring AI与Milvus实现智能问答系统
  • dali本地安装和使用
  • WSD3043 MOSFET 在吸黑头仪中的应用
  • 小数第n位--快速幂+数学
  • 软件设计师“数据流图”真题考点分析——求三连
  • System.in 的本质:输入流的方向​
  • 高效能、高可靠性——SILM94112/08-AQ可编程半桥电机驱动器
  • WordPress_Madara 本地文件包含漏洞复现(CVE-2025-4524)
  • 深入浅出:线程安全问题的原因与解决方案
  • 5月21日直播安排
  • Taro 安全区域
  • React-改变当前页class默认的样式
  • PHP 扇形的面积(Area of a Circular Sector)
  • Redis集群在NoSQL中的应用与优化策略
  • 提升加密交易效率:PumpSwap批量交易功能深度解析
  • JAVA批量发送邮件(含excel内容)
  • Proteus 51单片机仿真模拟步骤详解【附有51单片机的仿真图,仿真软件】【调试专用】
  • 【VSCode】在远程服务器Linux 系统 实现 Anaconda 安装与下载
  • 职坐标编程开发进阶路径
  • 详解Redis缓存穿透、缓存雪崩、缓存击穿:原理、场景与解决方案