当前位置: 首页 > 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() {
    // 启动 InputDispatcher
    status_t result = mDispatcher->start();
    if (result) {
        ALOGE("Could not start InputDispatcher thread due to error %d.", result);
        return result;
    }

    // 启动 InputReader
    result = 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 层。

相关文章:

  • ubuntu下的node.js的安装
  • 带Label的韦恩图(vue)
  • 【Java】Maven
  • 【软件测试】自动化测试结合 CI/CD有哪些方案
  • Oracle 数据库查询表广播
  • 青蛙吃虫--dp
  • 【蓝桥杯】动态规划:线性动态规划
  • PhotoShop学习07
  • PostIn V1.0.8版本发布,IDEA 插件支持一键扫描上报,让接口定义不再繁琐
  • leetcode刷题记录44-208. 实现 Trie (前缀树)
  • 指针本质传递偏移动态申请空间 c语言(day05)
  • excel常见错误包括(#N/A、#VALUE!、#REF!、#DIV/0!、#NUM!、#NAME?、#NULL! )
  • 【蓝桥杯】动态规划:背包问题
  • 23种设计模式-行为型模式-模板方法
  • AtCoder 第400场初级竞赛 A~E题解
  • Redis客户端命令到服务器底层对象机制的完整流程?什么是Redis对象机制?为什么要有Redis对象机制?
  • 子串分值和(蓝桥杯)
  • 【MySQL 数据库】数据类型
  • Everything 安装教程与使用教程(附安装包)
  • java+postgresql+swagger-多表关联insert操作(七)
  • 陶石不语,玉见文明:临平玉架山考古博物馆明日开馆
  • 篮球培训机构东方启明星被指停摆,家长称已登记未退费用超百万
  • 俄乌谈判开始
  • 一图读懂丨创新创业人才最高补贴500万元!临港新片区发布创客新政“十二条”
  • 上海“城市文明开放麦”全城总动员,樊振东担任首位上海城市文明大使
  • 上海制造佳品汇大阪站即将启幕,泡泡玛特领潮出海