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

相关文章

python panda库从基础到高级操作分析

《pythonpanda库从基础到高级操作分析》本文介绍了Pandas库的核心功能,包括处理结构化数据的Series和DataFrame数据结构,数据读取、清洗、分组聚合、合并、时间序列分析及大数据... 目录1. Pandas 概述2. 基本操作:数据读取与查看3. 索引操作:精准定位数据4. Group

破茧 JDBC:MyBatis 在 Spring Boot 中的轻量实践指南

《破茧JDBC:MyBatis在SpringBoot中的轻量实践指南》MyBatis是持久层框架,简化JDBC开发,通过接口+XML/注解实现数据访问,动态代理生成实现类,支持增删改查及参数... 目录一、什么是 MyBATis二、 MyBatis 入门2.1、创建项目2.2、配置数据库连接字符串2.3、入

MySQL中EXISTS与IN用法使用与对比分析

《MySQL中EXISTS与IN用法使用与对比分析》在MySQL中,EXISTS和IN都用于子查询中根据另一个查询的结果来过滤主查询的记录,本文将基于工作原理、效率和应用场景进行全面对比... 目录一、基本用法详解1. IN 运算符2. EXISTS 运算符二、EXISTS 与 IN 的选择策略三、性能对比

MySQL常用字符串函数示例和场景介绍

《MySQL常用字符串函数示例和场景介绍》MySQL提供了丰富的字符串函数帮助我们高效地对字符串进行处理、转换和分析,本文我将全面且深入地介绍MySQL常用的字符串函数,并结合具体示例和场景,帮你熟练... 目录一、字符串函数概述1.1 字符串函数的作用1.2 字符串函数分类二、字符串长度与统计函数2.1

Apache Ignite 与 Spring Boot 集成详细指南

《ApacheIgnite与SpringBoot集成详细指南》ApacheIgnite官方指南详解如何通过SpringBootStarter扩展实现自动配置,支持厚/轻客户端模式,简化Ign... 目录 一、背景:为什么需要这个集成? 二、两种集成方式(对应两种客户端模型) 三、方式一:自动配置 Thick

SQL Server跟踪自动统计信息更新实战指南

《SQLServer跟踪自动统计信息更新实战指南》本文详解SQLServer自动统计信息更新的跟踪方法,推荐使用扩展事件实时捕获更新操作及详细信息,同时结合系统视图快速检查统计信息状态,重点强调修... 目录SQL Server 如何跟踪自动统计信息更新:深入解析与实战指南 核心跟踪方法1️⃣ 利用系统目录

使用IDEA部署Docker应用指南分享

《使用IDEA部署Docker应用指南分享》本文介绍了使用IDEA部署Docker应用的四步流程:创建Dockerfile、配置IDEADocker连接、设置运行调试环境、构建运行镜像,并强调需准备本... 目录一、创建 dockerfile 配置文件二、配置 IDEA 的 Docker 连接三、配置 Do

MySQL 内存使用率常用分析语句

《MySQL内存使用率常用分析语句》用户整理了MySQL内存占用过高的分析方法,涵盖操作系统层确认及数据库层bufferpool、内存模块差值、线程状态、performance_schema性能数据... 目录一、 OS层二、 DB层1. 全局情况2. 内存占js用详情最近连续遇到mysql内存占用过高导致

Python进行JSON和Excel文件转换处理指南

《Python进行JSON和Excel文件转换处理指南》在数据交换与系统集成中,JSON与Excel是两种极为常见的数据格式,本文将介绍如何使用Python实现将JSON转换为格式化的Excel文件,... 目录将 jsON 导入为格式化 Excel将 Excel 导出为结构化 JSON处理嵌套 JSON:

深入浅出SpringBoot WebSocket构建实时应用全面指南

《深入浅出SpringBootWebSocket构建实时应用全面指南》WebSocket是一种在单个TCP连接上进行全双工通信的协议,这篇文章主要为大家详细介绍了SpringBoot如何集成WebS... 目录前言为什么需要 WebSocketWebSocket 是什么Spring Boot 如何简化 We