当前位置: 首页 > news >正文

Android Input——IMS启动流程(二)

        InputManagerService(IMS)作为 Android 系统中的一个关键服务,在系统的启动过程中由 SystemServer 进程启动。SystemServer 负责启动和管理系统中几乎所有的核心服务,包括 AMS、PMS、WMS 和 IMS 等。

一、InputManagerService启动流程

1、SystemServer

源码位置:/frameworks/base/services/java/com/android/server/SystemServer.java

private void startOtherServices(@NonNull TimingsTraceAndSlog t) {WindowManagerService wm = null;InputManagerService inputManager = null;……try {……// 创建IMS实例inputManager = new InputManagerService(context);……// 创建并初始化WMS实例wm = WindowManagerService.main(context, inputManager, !mFirstBoot, mOnlyCore,new PhoneWindowManager(), mActivityManagerService.mActivityTaskManager);// 将WMS和IMS注册到系统服务列表中ServiceManager.addService(Context.WINDOW_SERVICE, wm, /* allowIsolated= */ false,DUMP_FLAG_PRIORITY_CRITICAL | DUMP_FLAG_PROTO);ServiceManager.addService(Context.INPUT_SERVICE, inputManager,/* allowIsolated= */ false, DUMP_FLAG_PRIORITY_CRITICAL);……// 设置从WMS返回的回调接口。这允许IMS在处理输入事件时可以通知WMS有关状态的变化inputManager.setWindowManagerCallbacks(wm.getInputManagerCallback());// 启动IMS,使其开始监听和处理来自各种输入设备的事件inputManager.start();……} catch (Throwable e) {throw e;}……
}

        这里在 SystemServer 的初始化过程中,InputManagerService 和 WindowManagerService 被创建出来,并将 WindowManagerService 中的 monitor 传给了 InputManagerService ,作为回调。最后调用 InputManagerService 的 start() 方法启动服务。

2、InputManagerService构造

InputManagerService

源码位置:/frameworks/base/services/core/java/com/android/server/input/InputManagerService.java

public InputManagerService(Context context) {this.mContext = context;this.mHandler = new InputManagerHandler(DisplayThread.get().getLooper());……mPtr = nativeInit(this, mContext, mHandler.getLooper().getQueue());……
}private static native long nativeInit(InputManagerService service,Context context, MessageQueue messageQueue);

        InputManagerService 的构造方法中执行了 nativeInit,这是个 native 方法,属于 jni 调用。

nativeInit

源码位置:/frameworks/base/services/core/jni/com_android_server_input_InputManagerService.cpp

static const JNINativeMethod gInputManagerMethods[] = {{"nativeInit","(Lcom/android/server/input/InputManagerService;Landroid/content/Context;Landroid/os/""MessageQueue;)J", (void*)nativeInit},……
}static jlong nativeInit(JNIEnv* env, jclass /* clazz */,jobject serviceObj, jobject contextObj, jobject messageQueueObj) {sp<MessageQueue> messageQueue = android_os_MessageQueue_getMessageQueue(env, messageQueueObj);if (messageQueue == nullptr) {jniThrowRuntimeException(env, "MessageQueue is not initialized.");return 0;}NativeInputManager* im = new NativeInputManager(contextObj, serviceObj,messageQueue->getLooper());im->incStrong(0);return reinterpret_cast<jlong>(im);
}

        可以看到该方法中创建了一个 NativeInputManager 实例,并且和 Java 层使用的是同一个 looper。

NativeInputManager::NativeInputManager(jobject contextObj,jobject serviceObj, const sp<Looper>& looper) :mLooper(looper), mInteractive(true) {……InputManager* im = new InputManager(this, this);mInputManager = im;defaultServiceManager()->addService(String16("inputflinger"), im);
}

        通过这个构造函数,NativeInputManager 不仅初始化了自身,还创建了一个核心的本地服务——InputManager,并将其注册到系统服务管理器中。 

InputManager

源码位置:/frameworks/native/services/inputflinger/InputManager.cpp

InputManager::InputManager(const sp<InputReaderPolicyInterface>& readerPolicy,const sp<InputDispatcherPolicyInterface>& dispatcherPolicy) {// 创建 InputDispatcher 实例mDispatcher = createInputDispatcher(dispatcherPolicy);mClassifier = new InputClassifier(mDispatcher);// 创建 InputReader 实例mReader = createInputReader(readerPolicy, mClassifier);
}

        InputManager 的构造函数展示了如何初始化输入系统的两个主要组件:InputDispatcher 和 InputReader,并通过 InputClassifier 来增强对输入事件的处理能力。 

createInputReader

源码位置:/frameworks/native/services/inputflinger/reader/InputReaderFactory.cpp

sp<InputReaderInterface> createInputReader(const sp<InputReaderPolicyInterface>& policy,const sp<InputListenerInterface>& listener) {return new InputReader(std::make_unique<EventHub>(), policy, listener);
}

        这里创建了 InputReader 实例,它是 Android 输入系统中负责从底层硬件读取原始输入事件并进行解析的核心组件之一。通过 createInputReader 函数,EventHub、策略接口和监听器接口被组合在一起,以初始化一个 InputReader 实例。 其中 EventHub 就是将数据从硬件驱动上读出来然后传递上来的通道。

3、InputManagerService启动

        在 SystemServer 中还调用了 InputManagerService 的 start() 方法,这里我们在来看下。

InputManagerService.start

public void start() {Slog.i(TAG, "Starting input manager");// 调用本地代码,启动输入管理的核心功能nativeStart(mPtr);// 将服务添加到看门狗中监控Watchdog.getInstance().addMonitor(this);// 注册设置观察者\设置广播接收器、即时更新等……
}private static native void nativeStart(long ptr);

        该方法负责启动输入管理服务并注册一些必要的设置观察者和广播接收器,以便在系统状态或用户设置发生变化时更新输入管理服务的行为。

nativeStart

源码位置:/frameworks/base/services/core/jni/com_android_server_input_InputManagerService.cpp

static const JNINativeMethod gInputManagerMethods[] = {……{"nativeStart", "(J)V", (void*)nativeStart},……
}static void nativeStart(JNIEnv* env, jclass /* clazz */, jlong ptr) {NativeInputManager* im = reinterpret_cast<NativeInputManager*>(ptr);status_t result = im->getInputManager()->start();if (result) {jniThrowRuntimeException(env, "Input manager could not be started.");}
}

        这里获取 InputManager 并调用它的 start() 方法。

InputManager.start

status_t InputManager::start() {// 启动 InputDispatcherstatus_t result = mDispatcher->start();if (result) {ALOGE("Could not start InputDispatcher thread due to error %d.", result);return result;}// 启动 InputReaderresult = mReader->start();if (result) {ALOGE("Could not start InputReader due to error %d.", result);mDispatcher->stop();return result;}return OK;
}

        InputManager::start() 是 Android 输入系统中用于启动输入管理器(InputManager)的关键方法。它的主要职责是依次启动 InputDispatcher 和 InputReader,确保输入事件的分发和读取流程能够正常运行。

二、总结

        InputManagerService(IMS)在 Android 系统中的启动流程,涉及到从系统服务的初始化到具体的输入事件读取与处理线程的启动,覆盖了从 Java 层到 Native 层的多个步骤。

  • 系统级服务启动:InputManagerService 是由 SystemServer 启动的核心系统服务之一,负责管理所有输入设备和事件。
  • Java层与Native层协作:InputManagerService 通过 JNI 与本地层的 InputManager 交互,后者整合了 EventHub、InputReader 和 InputDispatcher,实现了从硬件事件读取到应用分发的全流程。
  • 多线程处理:InputReader 创建了一个单独的线程来异步处理输入事件,保证了系统的响应性和效率。
  • 动态配置调整:通过监听系统设置变化和用户切换事件,InputManagerService 能够实时调整其行为,提供更灵活和个性化的用户体验。

        根据以上的初始化过程总结整个层次关系:

        从这个初始化过程来看,InputManagerService 的逻辑基本上都是通过 jni 实现在 native 层。


http://www.mrgr.cn/news/97598.html

相关文章:

  • 每日OJ题_剑指offer数组篇(剑指offer04+剑指offer11+剑指offer21)
  • IntelliJ IDEA下开发FPGA——FPGA开发体验提升__上
  • 【蓝桥杯】搜索算法:剪枝技巧+记忆化搜索
  • [蓝桥杯] 求和(C语言)
  • 剑指Offer(数据结构与算法面试题精讲)C++版——day7
  • 【蓝桥杯】动态规划:线性动态规划
  • IntelliJ IDEA下开发FPGA——FPGA开发体验提升__下
  • JVM基础架构:内存模型×Class文件结构×核心原理剖析
  • PythonJSON解析如何优雅处理嵌套JSON字符串
  • springboot中使用async实现异步编程
  • 【蓝桥杯】动态规划背包问题
  • Go语言从零构建SQL数据库(5)-Pratt解析算法:SQL表达式解析的核心引擎
  • 算法与数据结构线性表之栈和队列
  • Docker与VNC的使用
  • PPTAgent:一款开源免费生成和评估幻灯片的项目
  • Linux信号——信号的保存(2)
  • Linux信号——信号的处理(3)
  • QT6(12)3.3.1 Qt元对象系统概述:QObject 类与 QMetaObject 类,类型转换 qobject_cast<T>()。3.3.3 属性系统:帮助文档,
  • 【题解-Acwing】798. 差分矩阵
  • 【第十三届“泰迪杯”数据挖掘挑战赛】【2025泰迪杯】【代码篇】A题解题全流程(持续更新)