微信公众号搜"智元新知"关注
微信扫一扫可直接关注哦!

监听 sysfs 电池事件

如何解决监听 sysfs 电池事件

我正在编写一个检查电池容量的守护进程。这是用于运行 Linux 的太阳能嵌入式设备。我读过在守护进程中使用 sleep() 是个坏主意,因此我正在尝试使用事件。所以我写了一些 PoC,但我没有收到任何事件!正如他们向我推荐的那样,我的第一个实现使用了 libudevpoll()

#include <fcntl.h>
#include <libudev.h>
#include <poll.h>
#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <unistd.h>

int main(void)
{
    struct udev *udev;
    struct udev_monitor *mon;
    struct pollfd fds[1];
    int fd;
    udev = udev_new();
    if (udev == NULL)
        return 1;
    mon = udev_monitor_new_from_netlink(udev,"udev");
    udev_monitor_filter_add_match_subsystem_devtype(mon,"power_supply",NULL);
    udev_monitor_enable_receiving(mon);
    fd = udev_monitor_get_fd(mon);
    fds[0].fd = fd;
    fds[0].events = POLLIN;
    fds[0].revents = 0;
    if (poll(fds,1,-1) > 0) {
        /* Never gets here! */
        struct udev_device *const dev = udev_monitor_receive_device(mon);
        if (dev != NULL) {
            puts(udev_device_get_sysname(dev));
            udev_device_unref(dev);
        }
        else
            fputs("udev_monitor_receive_device() Failed\n",stderr);
    }
    udev_unref(udev);
    return 0;
}

我得到的唯一事件是当我插入/拔出充电器时!然后我认为我在我的笔记本电脑安装中使用的状态栏确实显示电池容量。我查看了 source,他们使用 inotify 来监控电池的 uevent。但是我到处都读到我不应该将 inotify 用于 sysfs!尽管如此,我还是尝试了:

#include <stdio.h>
#include <sys/inotify.h>
#include <unistd.h>

#define BAT_PATH "/sys/class/power_supply/BAT0"

int main(void)
{
    struct inotify_event ev = {0};
    int wd,ret = 1;
    ssize_t len;
    const int fd = inotify_init1(IN_CLOEXEC);
    if (fd < 0) {
        perror("inotify_init() Failed");
        return ret;
    }
    /* else */
    wd = inotify_add_watch(fd,BAT_PATH "/uevent",IN_ACCESS);
    if (wd < 0)
        goto end;
    /* else */
    len = read(fd,&ev,sizeof(ev));
    /* Again... never gets here. */
    if (len > 0 && (ev.mask & IN_ACCESS))
        puts("It worked!");
    inotify_rm_watch(fd,wd);
    ret = 0;
end:
    close(fd);
    return ret;
}

事实证明这也不起作用!它如何适用于我的状态栏但在我尝试时不起作用?我做错了什么吗?谢谢。

解决方法

关于你的第一个实现(会评论但没有足够的代表。因为我对 libudev 一无所知):我遵循的指南成功使用 sysfs 来轮询()一个用于中断的 GPIO 建议寻找一个 POLLPRI 事件,而不是您在第一个实现中显示的 POLLIN(有关事件类型,请参阅 man poll)。

更重要的是,你说当你连接/断开充电器时你会得到一个事件,你的意思是每个软件执行一个事件吗?如果是这种情况,可能是因为您没有清除中断标志:在 poll() 命中后,在 sysfs 中需要 int len = read(fds[0].fd,*buf,SIZE); 将中断标记为已服务,并且lseek(fds[0].fd,0); 以便下一次 read() 成功(有关代码示例,请参阅我的其他答案 here)。

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


Selenium Web驱动程序和Java。元素在(x,y)点处不可单击。其他元素将获得点击?
Python-如何使用点“。” 访问字典成员?
Java 字符串是不可变的。到底是什么意思?
Java中的“ final”关键字如何工作?(我仍然可以修改对象。)
“loop:”在Java代码中。这是什么,为什么要编译?
java.lang.ClassNotFoundException:sun.jdbc.odbc.JdbcOdbcDriver发生异常。为什么?
这是用Java进行XML解析的最佳库。
Java的PriorityQueue的内置迭代器不会以任何特定顺序遍历数据结构。为什么?
如何在Java中聆听按键时移动图像。
Java“Program to an interface”。这是什么意思?