RK3568驱动指南|第十三篇 输入子系统-第150章 通用事件处理层event函数分析

本文主要是介绍RK3568驱动指南|第十三篇 输入子系统-第150章 通用事件处理层event函数分析,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

瑞芯微RK3568芯片是一款定位中高端的通用型SOC,采用22nm制程工艺,搭载一颗四核Cortex-A55处理器和Mali G52 2EE 图形处理器。RK3568 支持4K 解码和 1080P 编码,支持SATA/PCIE/USB3.0 外围接口。RK3568内置独立NPU,可用于轻量级人工智能应用。RK3568 支持安卓 11 和 linux 系统,主要面向物联网网关、NVR 存储、工控平板、工业检测、工控盒、卡拉 OK、云终端、车载中控等行业。


【公众号】迅为电子

【粉丝群】824412014(加群获取驱动文档+例程)

【视频观看】嵌入式学习之Linux驱动(第十三篇 输入子系统_全新升级)_基于RK3568

【购买链接】迅为RK3568开发板瑞芯微Linux安卓鸿蒙ARM核心板人工智能AI主板


第150章 通用事件处理层event函数分析

接下来,我们将继续学习event和events函数,如下图所示:

evdev_events函数如下所示:

static void evdev_events(struct input_ handle *handle, const struct input_value *vals, unsigned int count)
{struct evdev *evdev = handle->private;  // 获取输入句柄的私有数据,这里假设是evdev结构体类型struct evdev_client *client;  // 定义evdev客户端指针ktime_t *ev_time = input_get_timestamp(handle->dev);  // 获取输入设备的时间戳rcu_read_lock();  // 开始读取RCU保护区域client = rcu_dereference(evdev->grab);  // RCU安全地获取当前的evdev客户端if (client)evdev_pass_values(client, vals, count, ev_time);  // 如果存在抢占的客户端,则将值传递给抢占的客户端elselist_for_each_entry_rcu(client, &evdev->client_list, node)evdev_pass_values(client, vals, count, ev_time);  // 否则,将值传递给所有注册的客户端rcu_read_unlock();  // 结束读取RCU保护区域
}

该函数用于处理输入设备的事件。它接受一个输入句柄、一个输入值数组以及值的数量作为参数。

首先,它从输入句柄的私有数据中获取一个指向evdev结构体的指针。

然后,它获取输入设备的时间戳。

接下来,它通过使用RCU(Read-Copy-Update)机制来保护数据访问。通过调用rcu_read_lock()函数,它开始读取RCU保护区域。

然后,它通过RCU安全地获取当前的evdev客户端。如果存在抢占的客户端(即非空),它将调用evdev_pass_values函数将值传递给抢占的客户端。否则,它使用list_for_each_entry_rcu宏遍历evdev->client_list链表中的每个客户端,并调用evdev_pass_values函数将值传递给每个注册的客户端。

最后,它通过调用rcu_read_unlock()函数结束对RCU保护区域的读取操作。

我们学习一下evdev_pass_values函数,如下所示:

static void evdev_pass_values(struct evdev_client *client, const struct input_value *vals, unsigned int count, ktime_t *ev_time)
{struct evdev *evdev = client->evdev;  // 获取evdev客户端所属的evdev结构体const struct input_value *v;  // 当前处理的输入值struct input_event event;  // 输入事件结构体struct timespec64 ts;  // 时间戳bool wakeup = false;  // 是否需要唤醒等待线程if (client->revoked)return;  // 如果客户端已被撤销,则直接返回ts = ktime_to_timespec64(ev_time[client->clk_type]);  // 将ev_time转换为struct timespec64类型的时间戳event.input_event_sec = ts.tv_sec;  // 输入事件的秒字段设置为时间戳的秒值event.input_event_usec = ts.tv_nsec / NSEC_PER_USEC;  // 输入事件的微秒字段设置为时间戳的纳秒值除以1000得到的值/* 关中断,只需获取锁即可。 */spin_lock(&client->buffer_lock);  // 获取客户端的缓冲区锁for (v = vals; v != vals + count; v++) {if (__evdev_is_filtered(client, v->type, v->code))continue;  // 如果输入值被过滤,则跳过当前值的处理if (v->type == EV_SYN && v->code == SYN_REPORT) {/* 丢弃空的SYN_REPORT */if (client->packet_head == client->head)continue;  // 如果客户端的数据包头和数据头相同,则跳过当前值的处理wakeup = true;  // 设置唤醒标志为真}event.type = v->type;  // 设置输入事件的类型字段为当前值的类型event.code = v->code;  // 设置输入事件的代码字段为当前值的代码event.value = v->value;  // 设置输入事件的值字段为当前值的值__pass_event(client, &event);  // 将输入事件传递给客户端的事件处理函数}spin_unlock(&client->buffer_lock);  // 释放客户端的缓冲区锁if (wakeup)wake_up_interruptible(&evdev->wait);  // 如果需要唤醒等待线程,则唤醒等待队列中的线程
}

该函数用于将输入值传递给evdev客户端进行处理。它接受一个evdev_client结构体指针,一个输入值数组以及值的数量作为参数。

首先,它从客户端结构体中获取所属的evdev结构体。

然后,它检查客户端是否已被撤销,如果是,则直接返回。接下来,它将ev_time转换为struct timespec64类型的时间戳,并将其赋值给ts变量。然后,它将时间戳的秒值赋给输入事件结构体的input_event_sec字段,将时间戳的纳秒值除以1000得到的值赋给输入事件结构体的input_event_usec字段。然后,它获取客户端的缓冲区锁,以确保在处理输入值时不会有竞争条件。

接下来,它遍历输入值数组中的每个值。对于每个值,它首先检查是否被客户端过滤。如果被过滤,则跳过当前值的处理。

然后,它检查当前值是否为EV_SYN类型且代码为SYN_REPORT。如果是,它进一步检查客户端的数据包头和数据头是否相同。如果相同,则说明是一个空的SYN_REPORT,可以丢弃。否则,将唤醒标志设置为真。

接下来,它设置输入事件的类型字段为当前值的类型,代码字段为当前值的代码,值字段为当前值的值,并调用__pass_event函数将输入事件传递给客户端的事件处理函数。

完成对所有输入值的处理后,它释放客户端的缓冲区锁,解除对缓冲区的访问限制。

最后,如果需要唤醒等待线程(即唤醒标志为真),则调用wake_up_interruptible函数唤醒等待队列中的线程,以通知它们有新的事件可用。

   __pass_event函数将输入事件传递给客户端的事件处理函数,我们来学习一下。

static void __pass_event(struct evdev_client *client, const struct input_event *event)
{client->buffer[client->head++] = *event;  // 将事件复制到客户端的缓冲区中,然后将缓冲区头指针递增client->head &= client->bufsize - 1;  // 将缓冲区头指针掩码处理,确保其在缓冲区范围内if (unlikely(client->head == client->tail)) {/** 这实际上"丢弃"了所有未消耗的事件,只保留了EV_SYN/SYN_DROPPED加上最新的事件。*/client->tail = (client->head - 2) & (client->bufsize - 1);  // 更新缓冲区尾指针,使其指向倒数第二个事件client->buffer[client->tail] = (struct input_event) {.input_event_sec = event->input_event_sec,.input_event_usec = event->input_event_usec,.type = EV_SYN,.code = SYN_DROPPED,.value = 0,};  // 在缓冲区尾指针位置插入一个EV_SYN/SYN_DROPPED事件,表示丢弃了事件client->packet_head = client->tail;  // 更新数据包头指针为缓冲区尾指针}if (event->type == EV_SYN && event->code == SYN_REPORT) {client->packet_head = client->head;  // 更新数据包头指针为缓冲区头指针kill_fasync(&client->fasync, SIGIO, POLL_IN);  // 向注册的异步通知处理函数发送SIGIO信号,通知有新的事件可读取}
}

该函数用于将输入事件传递给evdev客户端的缓冲区进行存储。它接受一个evdev_client结构体指针和一个输入事件指针作为参数。

首先,它将输入事件复制到客户端的缓冲区中,并递增缓冲区头指针。

然后,它对缓冲区头指针进行掩码处理,以确保其在缓冲区范围内。

接下来,它检查缓冲区头指针是否等于缓冲区尾指针。如果相等,表示缓冲区已满,需要丢弃一些事件。

在这种情况下,它将缓冲区尾指针更新为缓冲区头指针减去2,并进行掩码处理,以确保其在缓冲区范围内。这样做是为了给丢弃的事件留出空间。

然后,它在缓冲区尾指针的位置插入一个EV_SYN/SYN_DROPPED事件,表示丢弃了事件。该事件具有与最新事件相同的时间戳。

接下来,它将数据包头指针更新为缓冲区尾指针,以确保数据包头始终指向最新的事件。

最后,如果输入事件的类型为EV_SYN且代码为SYN_REPORT,表示一个数据包的结束,它将数据包头指针更新为缓冲区头指针,并向注册的异步通知处理函数发送SIGIO信号,通知有新的事件可读取。

至此,通用事件处理层event函数分析完毕。

这篇关于RK3568驱动指南|第十三篇 输入子系统-第150章 通用事件处理层event函数分析的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

JDK21对虚拟线程的几种用法实践指南

《JDK21对虚拟线程的几种用法实践指南》虚拟线程是Java中的一种轻量级线程,由JVM管理,特别适合于I/O密集型任务,:本文主要介绍JDK21对虚拟线程的几种用法,文中通过代码介绍的非常详细,... 目录一、参考官方文档二、什么是虚拟线程三、几种用法1、Thread.ofVirtual().start(

从基础到高级详解Go语言中错误处理的实践指南

《从基础到高级详解Go语言中错误处理的实践指南》Go语言采用了一种独特而明确的错误处理哲学,与其他主流编程语言形成鲜明对比,本文将为大家详细介绍Go语言中错误处理详细方法,希望对大家有所帮助... 目录1 Go 错误处理哲学与核心机制1.1 错误接口设计1.2 错误与异常的区别2 错误创建与检查2.1 基础

Nginx分布式部署流程分析

《Nginx分布式部署流程分析》文章介绍Nginx在分布式部署中的反向代理和负载均衡作用,用于分发请求、减轻服务器压力及解决session共享问题,涵盖配置方法、策略及Java项目应用,并提及分布式事... 目录分布式部署NginxJava中的代理代理分为正向代理和反向代理正向代理反向代理Nginx应用场景

Python函数作用域与闭包举例深度解析

《Python函数作用域与闭包举例深度解析》Python函数的作用域规则和闭包是编程中的关键概念,它们决定了变量的访问和生命周期,:本文主要介绍Python函数作用域与闭包的相关资料,文中通过代码... 目录1. 基础作用域访问示例1:访问全局变量示例2:访问外层函数变量2. 闭包基础示例3:简单闭包示例4

Redis中的有序集合zset从使用到原理分析

《Redis中的有序集合zset从使用到原理分析》Redis有序集合(zset)是字符串与分值的有序映射,通过跳跃表和哈希表结合实现高效有序性管理,适用于排行榜、延迟队列等场景,其时间复杂度低,内存占... 目录开篇:排行榜背后的秘密一、zset的基本使用1.1 常用命令1.2 Java客户端示例二、zse

Redis中的AOF原理及分析

《Redis中的AOF原理及分析》Redis的AOF通过记录所有写操作命令实现持久化,支持always/everysec/no三种同步策略,重写机制优化文件体积,与RDB结合可平衡数据安全与恢复效率... 目录开篇:从日记本到AOF一、AOF的基本执行流程1. 命令执行与记录2. AOF重写机制二、AOF的

使用Java填充Word模板的操作指南

《使用Java填充Word模板的操作指南》本文介绍了Java填充Word模板的实现方法,包括文本、列表和复选框的填充,首先通过Word域功能设置模板变量,然后使用poi-tl、aspose-words... 目录前言一、设置word模板普通字段列表字段复选框二、代码1. 引入POM2. 模板放入项目3.代码

Python中isinstance()函数原理解释及详细用法示例

《Python中isinstance()函数原理解释及详细用法示例》isinstance()是Python内置的一个非常有用的函数,用于检查一个对象是否属于指定的类型或类型元组中的某一个类型,它是Py... 目录python中isinstance()函数原理解释及详细用法指南一、isinstance()函数

python中的高阶函数示例详解

《python中的高阶函数示例详解》在Python中,高阶函数是指接受函数作为参数或返回函数作为结果的函数,下面:本文主要介绍python中高阶函数的相关资料,文中通过代码介绍的非常详细,需要的朋... 目录1.定义2.map函数3.filter函数4.reduce函数5.sorted函数6.自定义高阶函数

Python中的sort方法、sorted函数与lambda表达式及用法详解

《Python中的sort方法、sorted函数与lambda表达式及用法详解》文章对比了Python中list.sort()与sorted()函数的区别,指出sort()原地排序返回None,sor... 目录1. sort()方法1.1 sort()方法1.2 基本语法和参数A. reverse参数B.