Linux设备模型(四) - uevent应用:内核发送uevent,用户空间接收uevent

2024-02-27 04:44

本文主要是介绍Linux设备模型(四) - uevent应用:内核发送uevent,用户空间接收uevent,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

1,内核发送uevent

内核发送uevent的API由lib/kobject_event.c文件实现,include/linux/kobject.h是头文件。

enum kobject_action {KOBJ_ADD,KOBJ_REMOVE,KOBJ_CHANGE,KOBJ_MOVE,KOBJ_ONLINE,KOBJ_OFFLINE,KOBJ_MAX
};/* kobject_uevent不能用在中断上下文 */
int kobject_uevent(struct kobject *kobj, enum kobject_action action);
int kobject_uevent_env(struct kobject *kobj, enum kobject_action action, char *envp[]);

在driver中可以调用kobject_uevent或者kobject_uevent_env来向用户空间发送uevent

kobject_uevent默认会发送”ACTION=xxx”,”DEVPATH=xxx”,”SUBSYSTEM=xxx”这三个uevent环境变量。

kobject_uevent_env可以发送一些如”xxx=xxx”的自定义的uevent环境变量。

static ssize_t esd_info_store(struct device *dev,struct device_attribute *attr,const char *buf, size_t count)
{if (!buf || count <= 0)return -EINVAL;kobject_uevent(&core_data->pdev->dev.kobj, KOBJ_CHANGE);return count;
}static int test_uevent(struct device *dev,struct kobj_uevent_env *env)
{int ret = 0;ret = add_uevent_var(env,"COMMENT=%s", "test uevent message");if(ret)return ret;return 0;
}const struct device_type my_dev_type ={.name ="william_ts",.uevent= test_uevent,
};static int xxx_probe(struct platform_device *pdev)
{pdev->dev.type = &my_dev_type;
}

2,用户空间解析uevent

#define UEVENT_MSG_LEN 2048static void uevent_event(uint32_t /*epevents*/, int event_fd)
{char msg[UEVENT_MSG_LEN + 2];int n;char *cp;n = uevent_kernel_multicast_recv(event_fd, msg, UEVENT_MSG_LEN);if (n <= 0)return;if (n >= UEVENT_MSG_LEN) /* overflow -- discard */return;msg[n] = '\0';msg[n + 1] = '\0';cp = msg;ALOGE("william get the uevent size n = %d, msg = %s", n, msg);while(*cp) {ALOGE("william receive the msg = %s", cp);/* advance to after the next \0 */while(*cp++);}ALOGE("william uevent received %s", msg);
}void *MotTouch::uevent_thread_loop(void)
{int epoll_fd, uevent_fd;struct epoll_event ev;int nevents = 0;ALOGE("creating uevent thread");uevent_fd = uevent_open_socket(64 * 1024, true);if (uevent_fd < 0) {ALOGE("uevent_init: uevent_open_socket failed\n");return NULL;}fcntl(uevent_fd, F_SETFL, O_NONBLOCK);ev.events = EPOLLIN;ev.data.ptr = (void *)uevent_event;epoll_fd = epoll_create(64);if (epoll_fd == -1) {ALOGE("epoll_create failed; errno=%d", errno);goto error;}if (epoll_ctl(epoll_fd, EPOLL_CTL_ADD, uevent_fd, &ev) == -1) {ALOGE("epoll_ctl failed; errno=%d", errno);goto error;}while (!destroyThread) {struct epoll_event events[64];nevents = epoll_wait(epoll_fd, events, 64, -1);if (nevents == -1) {if (errno == EINTR) continue;ALOGE("usb epoll_wait failed; errno=%d", errno);break;}for (int n = 0; n < nevents; ++n) {if (events[n].data.ptr)(*(void (*)(uint32_t, int event_fd))events[n].data.ptr)(events[n].events, uevent_fd);}}ALOGI("exiting worker thread");
error:close(uevent_fd);if (epoll_fd >= 0)close(epoll_fd);return NULL;
}void sighandler(int sig) {if (sig == SIGUSR1) {destroyThread = true;ALOGI("destroy set");return;}signal(SIGUSR1, sighandler);
}

3,测试结果

上报的event会通知到所有的用户进程,这里能够收到系统中所有的uevent消息,这里只打印了我们自己添加测试的event

william get the uevent size n = 234, msg = change@/devices/platform/william_ts.0

william receive the msg = change@/devices/platform/william_ts.0

william receive the msg = ACTION=change

william receive the msg = DEVPATH=/devices/platform/william_ts.0

william receive the msg = SUBSYSTEM=platform

william receive the msg = DEVTYPE=william_ts

william receive the msg = DRIVER=william_ts

william receive the msg = MODALIAS=platform:william_ts

william receive the msg = MAC=05:04:03:02:01:00

william receive the msg = COMMENT=test uevent message

william receive the msg = SEQNUM=13492

william uevent received change@/devices/platform/william_ts.0

这篇关于Linux设备模型(四) - uevent应用:内核发送uevent,用户空间接收uevent的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/751255

相关文章

Linux进程CPU绑定优化与实践过程

《Linux进程CPU绑定优化与实践过程》Linux支持进程绑定至特定CPU核心,通过sched_setaffinity系统调用和taskset工具实现,优化缓存效率与上下文切换,提升多核计算性能,适... 目录1. 多核处理器及并行计算概念1.1 多核处理器架构概述1.2 并行计算的含义及重要性1.3 并

JSONArray在Java中的应用操作实例

《JSONArray在Java中的应用操作实例》JSONArray是org.json库用于处理JSON数组的类,可将Java对象(Map/List)转换为JSON格式,提供增删改查等操作,适用于前后端... 目录1. jsONArray定义与功能1.1 JSONArray概念阐释1.1.1 什么是JSONA

nginx -t、nginx -s stop 和 nginx -s reload 命令的详细解析(结合应用场景)

《nginx-t、nginx-sstop和nginx-sreload命令的详细解析(结合应用场景)》本文解析Nginx的-t、-sstop、-sreload命令,分别用于配置语法检... 以下是关于 nginx -t、nginx -s stop 和 nginx -s reload 命令的详细解析,结合实际应

Linux线程之线程的创建、属性、回收、退出、取消方式

《Linux线程之线程的创建、属性、回收、退出、取消方式》文章总结了线程管理核心知识:线程号唯一、创建方式、属性设置(如分离状态与栈大小)、回收机制(join/detach)、退出方法(返回/pthr... 目录1. 线程号2. 线程的创建3. 线程属性4. 线程的回收5. 线程的退出6. 线程的取消7.

Linux下进程的CPU配置与线程绑定过程

《Linux下进程的CPU配置与线程绑定过程》本文介绍Linux系统中基于进程和线程的CPU配置方法,通过taskset命令和pthread库调整亲和力,将进程/线程绑定到特定CPU核心以优化资源分配... 目录1 基于进程的CPU配置1.1 对CPU亲和力的配置1.2 绑定进程到指定CPU核上运行2 基于

golang程序打包成脚本部署到Linux系统方式

《golang程序打包成脚本部署到Linux系统方式》Golang程序通过本地编译(设置GOOS为linux生成无后缀二进制文件),上传至Linux服务器后赋权执行,使用nohup命令实现后台运行,完... 目录本地编译golang程序上传Golang二进制文件到linux服务器总结本地编译Golang程序

Linux下删除乱码文件和目录的实现方式

《Linux下删除乱码文件和目录的实现方式》:本文主要介绍Linux下删除乱码文件和目录的实现方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录linux下删除乱码文件和目录方法1方法2总结Linux下删除乱码文件和目录方法1使用ls -i命令找到文件或目录

Linux在线解压jar包的实现方式

《Linux在线解压jar包的实现方式》:本文主要介绍Linux在线解压jar包的实现方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录linux在线解压jar包解压 jar包的步骤总结Linux在线解压jar包在 Centos 中解压 jar 包可以使用 u

linux解压缩 xxx.jar文件进行内部操作过程

《linux解压缩xxx.jar文件进行内部操作过程》:本文主要介绍linux解压缩xxx.jar文件进行内部操作,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、解压文件二、压缩文件总结一、解压文件1、把 xxx.jar 文件放在服务器上,并进入当前目录#

Linux系统性能检测命令详解

《Linux系统性能检测命令详解》本文介绍了Linux系统常用的监控命令(如top、vmstat、iostat、htop等)及其参数功能,涵盖进程状态、内存使用、磁盘I/O、系统负载等多维度资源监控,... 目录toppsuptimevmstatIOStatiotopslabtophtopdstatnmon