LinuxAndroid: 旋转编码器input输入事件适配(旋转输入)

rk3588s: 旋转编码器input输入事件适配

基于Android 12 + kernel-5.10版本

参考文档:
https://blog.csdn.net/szembed/article/details/131551950
Linux 输入设备调试详解(零基础开发)Rotary_Encoder旋转编码器驱动 通用GPIO为例 挂载input输入子系统

https://source.android.google.cn/docs/core/interaction/input?hl=zh-cn
https://developer.android.google.cn/reference/android/support/wearable/input/RotaryEncoder
https://developer.android.google.cn/training/wearables/user-input/rotary-input?hl=zh-cn
旋转输入
某些 Wear OS 设备包含实体侧面旋钮。当用户旋转此类旋钮时,应用的当前视图会向上或向下滚动。此类输入称为“旋转输入”。

1,驱动层配置
配置设备树,使用已有的rotary_encoder.c驱动代码。

linux驱动设备树配置参考:
https://elixir.bootlin.com/linux/latest/source/drivers/input/misc/rotary_encoder.c
https://elixir.bootlin.com/linux/latest/source/arch/arm64/boot/dts/freescale/imx8mn-dimonoff-gateway-evk.dtsrotary: rotary-encoder {compatible = "rotary-encoder";pinctrl-names = "default";pinctrl-0 = <&pinctrl_rotary>;gpios = <&gpio5 12 GPIO_ACTIVE_LOW>, /* A */<&gpio5 13 GPIO_ACTIVE_LOW>; /* B */linux,axis = <0>; /* REL_X */rotary-encoder,relative-axis;};pinctrl_rotary: rotarygrp {fsl,pins = <MX8MN_IOMUXC_ECSPI2_MISO_GPIO5_IO12	0x00000156MX8MN_IOMUXC_ECSPI2_SS0_GPIO5_IO13	0x00000156>;};

2,framework层适配
上面驱动层配置好设备树后,通过getevent能看到rotary encoder事件。
但是,应用App层却收不到。
旋转编码器input输入事件和鼠标滚轮类似,设备上鼠标滚轮事件是正常的。
于是,先看鼠标滚轮事件。

鼠标滚轮:
$ adb shell getevent -lpi
add device 2: /dev/input/event2bus:      0003vendor    093aproduct   2533version   0111name:     "Gaming Mouse"location: "usb-fc840000.usb-1/input0"id:       ""version:  1.0.1events:KEY (0001): BTN_MOUSE             BTN_RIGHT             BTN_MIDDLE            BTN_SIDE             BTN_EXTRA            REL (0002): REL_X                 REL_Y                 REL_WHEEL             REL_WHEEL_HI_RES     MSC (0004): MSC_SCAN             input props:<none>adb shell dumpsys input2: Gaming MouseClasses: CURSOR | EXTERNALPath: /dev/input/event2Enabled: trueDescriptor: 922b2be403d5734c3dacd1c480566209f0f39e80Location: usb-fc840000.usb-1/input0ControllerNumber: 0UniqueId: Identifier: bus=0x0003, vendor=0x093a, product=0x2533, version=0x0111KeyLayoutFile: KeyCharacterMapFile: ConfigurationFile: VideoDevice: <none>01-11 03:13:33.710   569   663 I EventHub: New device: id=6, fd=181, path='/dev/input/event2', name='Gaming Mouse', classes=CURSOR | EXTERNAL, configuration='', keyLayout='', keyCharacterMap='', builtinKeyboard=false, 
01-11 03:13:33.714   569   663 I InputReader: Device added: id=5, eventHubId=6, name='Gaming Mouse', descriptor='922b2be403d5734c3dacd1c480566209f0f39e80',sources=0x00002002

rotary encoder事件信息

rotary encoder事件信息
$ adb shell getevent -lpi
add device 2: /dev/input/event0bus:      0019vendor    0000product   0000version   0000name:     "rotary"  // 设备名是rotarylocation: ""id:       ""version:  1.0.1events:REL (0002): REL_X                input props:<none>

添加rotary.idc文件,用于framework层识别rotary encoder设备

framework层代码流程分析:
frameworks/native/services/inputflinger/reader/EventHub.cpp// Load the configuration file for the device.device->loadConfigurationLocked();// 要想rotary encoder旋转编码器被framework层识别到需要的条件:要有configuration文件且device.type为rotaryEncoder// See if this is a rotary encoder type device.String8 deviceType = String8();if (device->configuration &&device->configuration->tryGetProperty(String8("device.type"), deviceType)) {if (!deviceType.compare(String8("rotaryEncoder"))) {device->classes |= InputDeviceClass::ROTARY_ENCODER;}}// 根据设备名找configuration配置文件,adb shell getevent -lpi 查看到设备名是name:     "rotary"// Try device name.return getInputDeviceConfigurationFilePathByName(deviceIdentifier.getCanonicalName(), type);  先在android源代码device目录grep -r "rotaryEncoder"搜索看看是否有类似配置。
搜索到virtio_input_rotary.idc,所以,执行如下操作验证framework层就能识别到旋转编码器设备了,
cp device/generic/goldfish/input/virtio_input_rotary.idc rotary.idc
adb push rotary.idc /system/usr/idc/

添加rotary.idc文件,虽然framework层识别到了rotary encoder设备,但是事件还是报不到App层。继续分析。(原因是:rotary encoder报的事件是 EV_REL REL_X,而RotaryEncoderInputMapper没有解析REL_X事件。因此,需要适配解析REL_X事件)

打开DEBUG_INBOUND_EVENT_DETAILS log开关后,验证旋转编码器的input事件,
能看到log时,说明App层就能收到事件,如果打印不出该log,则App层收不到事件。
void InputDispatcher::notifyMotion(const NotifyMotionArgs* args) {
#if DEBUG_INBOUND_EVENT_DETAILSALOGD("notifyMotion - id=%" PRIx32 " eventTime=%" PRId64 ", deviceId=%d, source=0x%x, ""displayId=%" PRId32 ", policyFlags=0x%x, ""action=0x%x, actionButton=0x%x, flags=0x%x, metaState=0x%x, buttonState=0x%x, ""edgeFlags=0x%x, xPrecision=%f, yPrecision=%f, xCursorPosition=%f, ""yCursorPosition=%f, downTime=%" PRId64,args->id, args->eventTime, args->deviceId, args->source, args->displayId,args->policyFlags, args->action, args->actionButton, args->flags, args->metaState,args->buttonState, args->edgeFlags, args->xPrecision, args->yPrecision,args->xCursorPosition, args->yCursorPosition, args->downTime);for (uint32_t i = 0; i < args->pointerCount; i++) {ALOGD("  Pointer %d: id=%d, toolType=%d, ""x=%f, y=%f, pressure=%f, size=%f, ""touchMajor=%f, touchMinor=%f, toolMajor=%f, toolMinor=%f, ""orientation=%f",i, args->pointerProperties[i].id, args->pointerProperties[i].toolType,args->pointerCoords[i].getAxisValue(AMOTION_EVENT_AXIS_X),args->pointerCoords[i].getAxisValue(AMOTION_EVENT_AXIS_Y),args->pointerCoords[i].getAxisValue(AMOTION_EVENT_AXIS_PRESSURE),args->pointerCoords[i].getAxisValue(AMOTION_EVENT_AXIS_SIZE),args->pointerCoords[i].getAxisValue(AMOTION_EVENT_AXIS_TOUCH_MAJOR),args->pointerCoords[i].getAxisValue(AMOTION_EVENT_AXIS_TOUCH_MINOR),args->pointerCoords[i].getAxisValue(AMOTION_EVENT_AXIS_TOOL_MAJOR),args->pointerCoords[i].getAxisValue(AMOTION_EVENT_AXIS_TOOL_MINOR),args->pointerCoords[i].getAxisValue(AMOTION_EVENT_AXIS_ORIENTATION));}
#endif

RotaryEncoderInputMapper解析旋转编码器的input事件数据

RotaryEncoderInputMapper::sync() 函数解析旋转编码器的input事件
$ adb shell getevent -l
add device 1: /dev/input/event0name:     "rotary"/dev/input/event0: EV_REL       REL_X                00000001            
/dev/input/event0: EV_SYN       SYN_REPORT           00000000            /dev/input/event0: EV_REL       REL_X                ffffffff            
/dev/input/event0: EV_SYN       SYN_REPORT           00000000
 
86  void RotaryEncoderInputMapper::process(const RawEvent* rawEvent) {
87      mRotaryEncoderScrollAccumulator.process(rawEvent);
88  
89      if (rawEvent->type == EV_SYN && rawEvent->code == SYN_REPORT) {
90          sync(rawEvent->when, rawEvent->readTime);
91      }
92  }
93  
94  void RotaryEncoderInputMapper::sync(nsecs_t when, nsecs_t readTime) {
95      PointerCoords pointerCoords;
96      pointerCoords.clear();
97  
98      PointerProperties pointerProperties;
99      pointerProperties.clear();
100      pointerProperties.id = 0;
101      pointerProperties.toolType = AMOTION_EVENT_TOOL_TYPE_UNKNOWN;
102  
+    ALOGI("RotaryEncoderInputMapper::sync");// scroll 返回的是0,导致下面notifyMotion走不到。需要在getRelativeVWheel函数里适配
103      float scroll = mRotaryEncoderScrollAccumulator.getRelativeVWheel();
104      bool scrolled = scroll != 0;
105  
106      // This is not a pointer, so it's not associated with a display.
107      int32_t displayId = ADISPLAY_ID_NONE;
108  
109      // Moving the rotary encoder should wake the device (if specified).
110      uint32_t policyFlags = 0;
111      if (scrolled && getDeviceContext().isExternal()) {
112          policyFlags |= POLICY_FLAG_WAKE;
113      }
114  
115      if (mOrientation == DISPLAY_ORIENTATION_180) {
116          scroll = -scroll;
117      }
118  
119      // Send motion event.
120      if (scrolled) {
121          int32_t metaState = getContext()->getGlobalMetaState();
122          pointerCoords.setAxisValue(AMOTION_EVENT_AXIS_SCROLL, scroll * mScalingFactor);// 添加x的值,否则,无论正向旋转还是反向旋转,x值都是0,导致应用App无法识别方向
+        pointerCoords.setAxisValue(AMOTION_EVENT_AXIS_X, scroll);
123  
124          NotifyMotionArgs scrollArgs(getContext()->getNextId(), when, readTime, getDeviceId(),
125                                      mSource, displayId, policyFlags, AMOTION_EVENT_ACTION_SCROLL, 0,
126                                      0, metaState, /* buttonState */ 0, MotionClassification::NONE,
127                                      AMOTION_EVENT_EDGE_FLAG_NONE, 1, &pointerProperties,
128                                      &pointerCoords, 0, 0, AMOTION_EVENT_INVALID_CURSOR_POSITION,
129                                      AMOTION_EVENT_INVALID_CURSOR_POSITION, 0, /* videoFrames */ {});
130          getListener()->notifyMotion(&scrollArgs);
+    ALOGI("RotaryEncoderInputMapper::sync notifyMotion");
131      }
132  
133      mRotaryEncoderScrollAccumulator.finishSync();
134  }42  void CursorScrollAccumulator::process(const RawEvent* rawEvent) {
43      if (rawEvent->type == EV_REL) {
44          switch (rawEvent->code) {
45              case REL_WHEEL:
46                  mRelWheel = rawEvent->value;
47                  break;
48              case REL_HWHEEL:
49                  mRelHWheel = rawEvent->value;
50                  break;
+            case REL_X:  // 由于自己的旋转编码器报的事件是REL_X,所以,需要添加该类型解析
+                mRelWheel = rawEvent->value;
+                break;
51          }
52      }
53  }
log: 旋转编码器正向旋转:x=1.000000   SOURCE_ROTARY_ENCODER = 0x00400000  <==> source=0x400000
04-07 05:47:32.041   575   667 D InputDispatcher: notifyMotion - id=b0eb158 eventTime=1947954629000, deviceId=4, source=0x400000, displayId=-1, policyFlags=0x0, action=0x8, actionButton=0x0, flags=0x0, metaState=0x0, buttonState=0x0, edgeFlags=0x0, xPrecision=0.000000, yPrecision=0.000000, xCursorPosition=nan, yCursorPosition=nan, downTime=0
04-07 05:47:32.041   575   667 D InputDispatcher:   Pointer 0: id=0, toolType=0, x=1.000000, y=0.000000, pressure=0.000000, size=0.000000, touchMajor=0.000000, touchMinor=0.000000, toolMajor=0.000000, toolMinor=0.000000, orientation=0.000000
04-07 05:47:32.041   575   666 D InputDispatcher: dispatchMotion - eventTime=1947954629000, deviceId=4, source=0x400000, displayId=-1, policyFlags=0x62000000, action=0x8, actionButton=0x0, flags=0x0, metaState=0x0, buttonState=0x0,edgeFlags=0x0, xPrecision=0.000000, yPrecision=0.000000, downTime=0
04-07 05:47:32.041   575   666 D InputDispatcher:   Pointer 0: id=0, toolType=0, x=1.000000, y=0.000000, pressure=0.000000, size=0.000000, touchMajor=0.000000, touchMinor=0.000000, toolMajor=0.000000, toolMinor=0.000000, orientation=0.000000旋转编码器反向旋转:x=-1.000000  SOURCE_ROTARY_ENCODER = 0x00400000  <==> source=0x400000
04-07 05:47:35.923   575   667 I InputReader: lqy111 RotaryEncoderInputMapper::sync
04-07 05:47:35.923   575   667 I InputReader: lqy111 RotaryEncoderInputMapper::sync: scroll:-1.000000
04-07 05:47:35.923   575   667 I InputReader: lqy111 RotaryEncoderInputMapper::sync notifyMotion
04-07 05:47:35.923   575   667 D InputDispatcher: notifyMotion - id=45c55f eventTime=1951836828000, deviceId=4, source=0x400000, displayId=-1, policyFlags=0x0, action=0x8, actionButton=0x0, flags=0x0, metaState=0x0, buttonState=0x0, edgeFlags=0x0, xPrecision=0.000000, yPrecision=0.000000, xCursorPosition=nan, yCursorPosition=nan, downTime=0
04-07 05:47:35.923   575   667 D InputDispatcher:   Pointer 0: id=0, toolType=0, x=-1.000000, y=0.000000, pressure=0.000000, size=0.000000, touchMajor=0.000000, touchMinor=0.000000, toolMajor=0.000000, toolMinor=0.000000, orientation=0.000000
04-07 05:47:35.923   575   666 D InputDispatcher: dispatchMotion - eventTime=1951836828000, deviceId=4, source=0x400000, displayId=-1, policyFlags=0x62000000, action=0x8, actionButton=0x0, flags=0x0, metaState=0x0, buttonState=0x0,edgeFlags=0x0, xPrecision=0.000000, yPrecision=0.000000, downTime=0
04-07 05:47:35.923   575   666 D InputDispatcher:   Pointer 0: id=0, toolType=0, x=-1.000000, y=0.000000, pressure=0.000000, size=0.000000, touchMajor=0.000000, touchMinor=0.000000, toolMajor=0.000000, toolMinor=0.000000, orientation=0.000000core/java/android/view/InputDevice.java:    public static final int SOURCE_ROTARY_ENCODER = 0x00400000 | SOURCE_CLASS_NONE; <==> source=0x400000

App层监听旋转输入事件
在这里插入图片描述

App层监听旋转输入事件:
developer.android.google.cn/training/wearables/user-input/rotary-input
myView.setOnGenericMotionListener
onGenercMotion
或者 在Activity也可以。framework层监听旋转输入事件:
在NativeInputManager::interceptMotionBeforeQueueing()添加适配代码。
com_android_server_input_InputManagerService.cpp
void NativeInputManager::interceptMotionBeforeQueueing(const int32_t displayId, nsecs_t when,uint32_t& policyFlags) {

调试总结

make libinputreader -j3
make libinputflinger -j3 && make libinputflinger_base -j3while true; do echo "######$(date)######";adb logcat -b all | grep -i -E "EventHub|InputDispatcher|InputReader|WindowManager"; doneadb shell getevent
adb shell getevent -l
adb shell getevent -lip
adb shell dumpsys input

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/806313.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

日志监控思路分享,只监控日志内容,不存储

有一个这样的需求&#xff0c;就是实时监控日志文件的内容&#xff0c;不需要存储&#xff0c;仅当某行日志内容触发某个规则时调用一段业务逻辑就行了。比如用户触发限流规则&#xff0c;就将其封禁并发送钉钉通知到运维群。 看到这个需求首先想到的就是日志采集工具&#xff…

零售EDI:Princess Auto EDI对接

Princess Auto 是一家加拿大零售连锁店&#xff0c;专门从事农场、工业、车库、液压和剩余物品的销售。 Princess Auto 总部位于马尼托巴省温尼伯&#xff0c;截至 2024 年 1 月在 10 个省份拥有并经营 55 家商店以及三个配送中心。各种商品均以其“Powerfist”和“Pro.Point”…

Node.js 入门

一、什么是 Node.js&#xff1f; 定义&#xff1a; 作用&#xff1a;使用 Node.js 编写服务器端程序 编写数据接口&#xff0c;提供网页资源浏览功能等等 前端工程化&#xff1a;为后续学习 Vue 和 React 等框架做铺垫 二、什么是前端工程化&#xff1f; 前端工程化&#xf…

学浪视频怎么下载到手机相册

学浪视频如何下载到手机相册&#xff0c;很简单&#xff0c;首先将学浪课程下载到电脑&#xff0c;然后再由电脑传输到手机中&#xff0c;这样你就可以在手机相册里面观看。 但是将学浪课程下载到本地是不可以使用录屏的方式&#xff0c;所以这里要借助一款工具&#xff0c;小…

自动化测试-web(弹窗/滚动条/鼠标/等待等操作)

一、弹窗 为什么要处理弹窗&#xff1f; 如果页面操作过程中&#xff0c;有弹窗出现&#xff0c;不处理&#xff0c;无法继续对页面操作。 弹窗类型&#xff1a; js原生弹窗&#xff1a; 警告框、输入框、提示框&#xff0c;这些必须处理 如何处理&#xff1a; 1&#xff0…

Node.js cnpm的安装

百度搜索 cnpm,进入npmmirror 镜像站https://npmmirror.com/ cmd窗口输入 npm install -g cnpm --registryhttps://registry.npmmirror.com

dyld: Library not loaded: @rpath/SDK.framework/SDK错误问题

关于导入三方SDK.framework之后&#xff0c;启动崩溃之后如下报错的解决方式: 截屏2020-10-14 上午9.55.09.png 在正常导入framework之后&#xff0c;做如图示操作&#xff0c; image.png 以上步骤之后&#xff0c;重新启动运行xcode&#xff0c;即可成功运行。

Python 基于 OpenCV 视觉图像处理实战 之 OpenCV 简单视频处理实战案例 之八 简单视频素描效果

Python 基于 OpenCV 视觉图像处理实战 之 OpenCV 简单视频处理实战案例 之八 简单视频素描效果 目录 Python 基于 OpenCV 视觉图像处理实战 之 OpenCV 简单视频处理实战案例 之八 简单视频素描效果 一、简单介绍 二、简单指定视频某片段快放效果实现原理 三、简单指定视频某…

[leetcode]remove-duplicates-from-sorted-list

. - 力扣&#xff08;LeetCode&#xff09; 给定一个已排序的链表的头 head &#xff0c; 删除所有重复的元素&#xff0c;使每个元素只出现一次 。返回 已排序的链表 。 示例 1&#xff1a; 输入&#xff1a;head [1,1,2] 输出&#xff1a;[1,2]示例 2&#xff1a; 输入&…

安全风险攻击面管理如何提升企业网络弹性?

从研究人员近些年的调查结果来看&#xff0c;威胁攻击者目前非常善于识别和利用最具有成本效益的网络入侵方法&#xff0c;这就凸显出了企业实施资产识别并了解其资产与整个资产相关的安全态势的迫切需要。 目前来看&#xff0c;为了在如此复杂的网络环境中受到最小程度上的网络…

Unity Pro 2019 for Mac:专业级游戏引擎,助力创意无限延伸!

Unity Pro 2019是一款功能强大的游戏开发引擎&#xff0c;其特点主要体现在以下几个方面&#xff1a; 强大的渲染技术&#xff1a;Unity Pro 2019采用了新的渲染技术&#xff0c;包括脚本化渲染流水线&#xff0c;能够轻松自定义渲染管线&#xff0c;通过C#代码和材料材质&…

uniapp 上传视频到阿里云之后回显视频获取视频封面

uniapp 上传视频到阿里云之后回显视频获取视频封面 官网的解决方案 1.initial-time Number 指定视频初始播放位置&#xff0c;单位为秒&#xff08;s&#xff09;。 没什么卵用 2.使用 uni.createVideoContext(“myVideo”, this).seek(number)。 没什么卵用 <video :id&quo…

云计算重要概念之:虚拟机、网卡、交换机、路由器、防火墙

一、虚拟机 (Virtual Machine, VM) 1.主流的虚拟化软件&#xff1a; 虚拟化软件通过在单个物理硬件上创建和管理多个虚拟环境&#xff08;虚拟机&#xff09;&#xff0c;实现资源的高效利用、灵活部署、隔离安全以及便捷管理&#xff0c;是构建云计算和现代化数据中心的核心…

C语言面试题之返回倒数第 k 个节点

返回倒数第 k 个节点 实例要求 1、实现一种算法&#xff0c;找出单向链表中倒数第 k 个节点&#xff1b;2、返回该节点的值&#xff1b; 示例&#xff1a;输入&#xff1a; 1->2->3->4->5 和 k 2 输出&#xff1a; 4 说明&#xff1a;给定的 k 保证是有效的。实…

【机器学习300问】65、为什么Sigmoid和Tanh激活函数会导致梯度消失?

一、梯度消失现象 当神经网络的输入值较大或较小时&#xff0c;其导数&#xff08;梯度&#xff09;都会接近于0。在反向传播过程中&#xff0c;这些微小的梯度经过多层网络逐层传递时&#xff0c;会不断被乘以权重矩阵&#xff08;权重通常小于1&#xff09;&#xff0c;进一步…

智能时代中的工业应用中前所未有的灵活桥接和I/O扩展功能解决方案MachXO2系列LCMXO2-1200HC-4TG100I FPGA可编程逻辑IC

lattice莱迪斯 MachXO2系列LCMXO2-1200HC-4TG100I超低密度FPGA现场可编程门阵列&#xff0c;适用于低成本的复杂系统控制和视频接口设计开发&#xff0c;满足了通信、计算、工业、消费电子和医疗市场所需的系统控制和接口应用。 瞬时启动&#xff0c;迅速实现控制——启动时间…

Java项目:基于Springboot+vue实现的中国陕西民俗前后台管理系统设计与实现(源码+数据库+毕业论文)

一、项目简介 本项目是一套基于Springbootvue实现的中国陕西民俗管理系统设计与实现设 包含&#xff1a;项目源码、数据库脚本等&#xff0c;该项目附带全部源码可作为毕设使用。 项目都经过严格调试&#xff0c;eclipse或者idea 确保可以运行&#xff01; 该系统功能完善、界…

安全大脑与盲人摸象

21世纪是数字科技和数字经济爆发的时代&#xff0c;互联网正从网状结构向类脑模型进行进化&#xff0c;出现了结构和覆盖范围庞大&#xff0c;能够适应不同技术环境、经济场景&#xff0c;跨地域、跨行业的类脑复杂巨型系统。如腾讯、Facebook等社交网络具备的神经网络特征&…

Web漏洞-文件上传之内容逻辑数组

图片一句话制作方法&#xff1a; copy 1.png /b shell.php /a webshell.jpg 具体示例见upload-labs 的14-17 二次渲染----见Pass-18 用/.或者%00绕过&#xff1a;Pass-20----Pass-21 CVE-2017-12615复现 创好环境后打开环境&#xff0c;再访问ip8080 抓包发送数据 Shell的…

电介质材料(四)

本篇为西安交通大学本科课程《电气材料基础》的笔记。 本篇为这一单元的第四篇笔记&#xff0c;上一篇传送门。 复合电介质材料 是由多种成分共同组成&#xff0c;例如油纸复合绝缘、云母层压板、环氧浸渍玻璃纤维布等。即便是没有添加的材料&#xff0c;材料也会存在杂质和…