Android Input系统8 ANR原理分析

本文详细介绍了Android系统中ANR(Application Not Responding)的触发原理,特别是与Input事件处理相关的ANR。从InputReader、InputDispatcher和UI线程的角度梳理了输入事件的处理流程,分析了ANR如何在5秒超时时限内检测到输入事件的分发延迟,并触发相应处理。同时,文章还讨论了系统的输入死锁监测机制,以及ANR的监控和报告过程。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

一 概述

当 input 事件处理得慢就会触发 ANR,那 ANR 内部原理是什么,哪些场景会产生 ANR? “工欲善其事必先利其器”,为了理解 input ANR 原理,前面几篇文章疏通了整个 input 框架的处理流程,都是为了这篇文章而做铺垫。在正式开始分析 ANR 触发原理以及触发场景之前,先来回顾一下 input 流程。

1.1 InputReader

在这里插入图片描述
InputReader 的主要工作分两部分:

1.调用 EventHub 的 getEvents() 读取节点 /dev/input/eventX 下的输入事件,并把表示原始事件的 input_event 结构体转换成 RawEvent 结构体,RawEvent 根据不同 InputMapper 来转换成相应的 EventEntry,比如按键事件则对应 KeyEntry,触摸事件则对应 MotionEntry。

  • 转换结果:input_event -> EventEntry

2.将事件添加到 InputDispatcher 的 mInboundQueue 队列尾部,加入该队列前有以下两个过滤:

  • IMS.interceptKeyBeforeQueueing:事件分发前可增加业务逻辑
  • IMS.filterInputEvent:可拦截事件,当返回值为 false 的事件都直接拦截,没有机会加入 mInboundQueue 队列,不会再往下分发;否则进入下一步
  • enqueueInboundEventLocked:执行输入事件放入 mInboundQueue 队列尾部
  • mLooper->wake:并根据情况来唤醒 InputDispatcher 线程

3.KeyboardInputMapper.processKey() 的过程,记录下按下 down 事件的时间点

1.2 InputDispatcher

在这里插入图片描述

1.dispatchOnceInnerLocked():从 InputDispatcher 的 mInboundQueue 队列,取出事件 EventEntry。另外该方法开始执行的时间点 (currentTime) 便是后续事件 dispatchEntry 的分发时间 (deliveryTime)

2.dispatchKeyLocked():满足一定条件时会添加命令 doInterceptKeyBeforeDispatchingLockedInterruptible

3.enqueueDispatchEntryLocked():生成事件 DispatchEntry 并加入 connection 的 outbound 队列

4.startDispatchCycleLocked():从 outboundQueue 中取出事件 DispatchEntry,重新放入 connection 的 waitQueue 队列

5.runCommandsLockedInterruptible():通过循环遍历方式,依次处理 mCommandQueue 队列中的所有命令。而 mCommandQueue 队列中的命令是通过 postCommandLocked() 方式向该队列添加的。ANR 回调命令便是在这个时机执行

6.handleTargetsNotReadyLocked():该过程会判断是否等待超过 5s 来决定是否调用 onANRLocked()

流程15中 sendMessage 是将 input 事件分发到 app 端,当 app 处理完该事件后会发送 finishInputEvent() 事件。接下来又回到 pollOnce() 方法。

1.3 UI Thread

在这里插入图片描述

  • InputDispatcher 线程监听 socket 服务端,收到消息后回调 InputDispatcher.handleReceiveCallback()
  • UI 主线程监听 socket 客户端,收到消息后回调 NativeInputEventReceiver.handleEvent()

对于 ANR 的触发主要是在 InputDispatcher 过程,下面再从 ANR 的角度来说一说 ANR 触发过程。

二 ANR处理流程

ANR 时间区间便是指当前这次的事件 dispatch 过程中执行 findFocusedWindowTargetsLocked() 方法到下一次执行 resetANRTimeoutsLocked() 的时间区间。以下 5 个函数会 reset。都位于 InputDispatcher.cpp 文件中:

  • dispatchOnceInnerLocked
  • setInputDispatchMode
  • setFocusedApplication
  • releasePendingEventLocked
  • resetAndDropEverythingLocked

简单来说,主要是以下 4 个场景,会有机会执行 resetANRTimeoutsLocked:

  • 解冻屏幕,系统开/关机的时刻点 (thawInputDispatchingLw,setEventDispatchingLw,最后调用 setInputDispatchMode)
  • wms 聚焦 app 的改变 (WMS.setFocusedApp,IMS.setFocusedApplication,setFocusedApplication)
  • 设置 input filter 的过程 (IMS.setInputFilter,进而调用 resetAndDropEverythingLocked)
  • 再次分发事件的过程 (dispatchOnceInnerLocked)
  • dispatch 结束的时候 (dispatchOnceInnerLocked 最后 done 为 true,最终调用 releasePendingEventLocked)

当 InputDispatcher 线程,执行 findFocusedWindowTargetsLocked() 过程调用到 handleTargetsNotReadyLocked,且满足超时 5s 的情况则会调用 onANRLocked()。

2.1 onANRLocked

void InputDispatcher::onANRLocked(nsecs_t currentTime,
    const sp<InputApplicationHandle>& applicationHandle,
    const sp<InputWindowHandle>& windowHandle,
    nsecs_t eventTime, nsecs_t waitStartTime, const char* reason) {
   
    
    float dispatchLatency = (currentTime - eventTime) * 0.000001f;
    float waitDuration = (currentTime - waitStartTime) * 0.000001f;

    ALOGI("Application is not responding: %s. "
"It has been %0.1fms since event, %0.1fms since wait started. Reason: %s",
getApplicationWindowLabelLocked(applicationHandle, windowHandle).string(),
dispatchLatency, waitDuration, reason);

    // 捕获 ANR 的现场信息
    time_t t = time(NULL);
    struct tm tm;
    localtime_r(&t, &tm);
    char timestr[64];
    strftime(timestr, sizeof(timestr), "%F %T", &tm);
    mLastANRState.clear();
    mLastANRState.append(INDENT "ANR:\n");
    mLastANRState.appendFormat(INDENT2 "Time: %s\n", timestr);
    mLastANRState.appendFormat(INDENT2 "Window: %s\n",
    getApplicationWindowLabelLocked(applicationHandle, windowHandle).string());
    mLastANRState.appendFormat(INDENT2 "DispatchLatency: %0.1fms\n", dispatchLatency);
    mLastANRState.appendFormat(INDENT2 "WaitDuration: %0.1fms\n", waitDuration);
    mLastANRState.appendFormat(INDENT2 "Reason: %s\n", reason);
    dumpDispatchStateLocked(mLastANRState);

    // 将 ANR 命令加入 mCommandQueue
    CommandEntry* commandEntry = postCommandLocked(
            & InputDispatcher::doNotifyANRLockedInterruptible);
    commandEntry->inputApplicationHandle = applicationHandle;
    commandEntry->inputWindowHandle = windowHandle;
    commandEntry->reason = reason;
}

onANRLocked() 中会对 ANR 信息进行收集,然后构建一个回调函数为 doNotifyANRLockedInterruptible 的 CommandEntry ,并加入 mCommandQueue 队列。

这样,当循环执行到下一轮 InputDispatcher.dispatchOnce 的过程中,会先执行 runCommandsLockedInterruptible() 方法,取出 mCommandQueue 队列的所有命令逐一执行。那么就会执行 ANR 所对应的函数 doNotifyANRLockedInterruptible

2.2 doNotifyANRLockedInterruptible

InputDispatcher.cpp

void InputDispatcher::doNotifyANRLockedInterruptible(
        CommandEntry* commandEntry) {
   
    mLock.unlock();
    
    nsecs_t newTimeout = mPolicy->notifyANR(
        commandEntry->inputApplicationHandle, commandEntry->inputWindowHandle,
        commandEntry->reason);

    mLock.lock();
    // newTimeout = 5s
    resumeAfterTargetsNotReadyTimeoutLocked(newTimeout,
            commandEntry->inputWindowHandle != NULL
            ? commandEntry->inputWindowHandle->getInputChannel() : NULL);
}

我们已经知道这里的 mPolicy,就是 NativeInputManager。

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值