Android15廣播ANR的源碼流程分析

Android15的廣播ANR源碼流程

跟了下實際代碼的流程,大概如下哈:

App.sendBroadcast() ?// 應用發起廣播→ AMS.broadcastIntentWithFeature() ?// 通過Binder IPC進入system_server進程→ AMS.broadcastIntentLocked() ?// 權限校驗+廣播分類(前臺/后臺)→ BroadcastQueueModernImpl.enqueueBroadcastLocked() ?// 按進程隔離的隊列管理→ BroadcastQueueModernImpl.updateRunningListLocked() ?// 檢查進程活躍狀態→ BroadcastQueueModernImpl.scheduleReceiverWarmLocked() ?// 預熱進程(若未啟動)→ BroadcastQueueModernImpl.dispatchReceivers() ?// 關鍵:啟動超時檢測(前臺10s/后臺60s)? ?├─ 通過Handler發送MSG_DELIVERY_TIMEOUT延遲消息? ?└─ 豁免條件:系統未就緒或廣播標記為timeoutExempt→ IApplicationThread.scheduleRegisteredReceiver() ?// Binder投遞到目標進程→ ActivityThread.post(Args.run()) ?// 主線程消息隊列調度→ Args.run() ?// 封裝廣播處理邏輯→ BroadcastReceiver.onReceive() ?// 開發者代碼執行點(耗時操作直接觸發ANR)→ AMS.finishReceiver() ?// 正常完成通知→ BroadcastQueueModernImpl.finishReceiverLocked() ?// 移除超時檢測(Handler取消MSG_DELIVERY_TIMEOUT)? ?└─ 若超時未完成→ AMS.appNotResponding() ?// 觸發ANR彈窗(含Intent/接收者類名等上下文)

0

源碼分析

→ App.sendBroadcast()?→ → AMS.broadcastIntentWithFeature()?→ → → AMS.broadcastIntentLocked() /broadcastIntentLockedTraced()→ → → → BroadcastQueueModernImpl.enqueueBroadcastLocked()→ → → → → BroadcastQueueModernImpl.updateRunningListLocked()?→ → → → → →BroadcastQueueModernImpl.scheduleReceiverWarmLocked()?→ → → → → → BroadcastQueueModernImpl.dispatchReceivers()?→ → → → → → → IApplicationThread.scheduleRegisteredReceiver()→→ → → → → → → → ActivityThread.post()→→ → → → → → → → → Args.run()?→ → → → → → → → → → BroadcastReceiver.onReceive()→ → → → → → → → → → → AMS.finishReceiver()?→ → → → → → → → → → → →BroadcastQueueModernImpl.finishReceiverLocked()?→ → → → → → → → → → → →→AMS.appNotResponding(queue.app, tr)

0. 廣播產生并發送給AMS

應用或系統廣播產生并發送給AMS

源碼路徑:frameworks/base/core/java/android/app/ContextImpl.java

@Overridepublic void sendBroadcast(Intent intent) {? ? warnIfCallingFromSystemProcess(); // 防止系統進程誤用普通API? ? String resolvedType = intent.resolveTypeIfNeeded(getContentResolver());? ? try {? ? ? ? intent.prepareToLeaveProcess(this); // 清理不可序列化對象,確保跨進程傳輸? ? ? ? // 通過Binder調用AMS的廣播接口? ? ? ? ActivityManager.getService().broadcastIntentWithFeature(? ? ? ? ? ? mMainThread.getApplicationThread(), // 調用方線程標識? ? ? ? ? ? getAttributionTag(), ? ? ? ? ? ? ? ?// 權限追蹤標簽? ? ? ? ? ? intent, resolvedType, null, Activity.RESULT_OK,?? ? ? ? ? ? null, null, null, null, null, AppOpsManager.OP_NONE,?? ? ? ? ? ? null, false, false, getUserId());? ? } catch (RemoteException e) {? ? ? ? throw e.rethrowFromSystemServer();? ? }}

1. AMS投遞廣播到隊列并啟動超時檢測

源碼路徑:frameworks/base/services/core/java/com/android/server/am/BroadcastQueueModernImpl.java

@GuardedBy("mService")private boolean dispatchReceivers(@NonNull BroadcastProcessQueue queue,? ? ? ? @NonNull BroadcastRecord r, int index) throws BroadcastRetryException {? ? final ProcessRecord app = queue.app;? ? // 僅在滿足條件時啟動超時檢測? ? if (mService.mProcessesReady && !r.timeoutExempt && !r.isAssumedDelivered(index)) {? ? ? ? queue.setTimeoutScheduled(true);? ? ? ? // 區分前臺/后臺廣播超時閾值? ? ? ? final int softTimeoutMillis = (int) (r.isForeground() ? mFgConstants.TIMEOUT?? ? ? ? ? ? ? ? : mBgConstants.TIMEOUT);? ? ? ? startDeliveryTimeoutLocked(queue, softTimeoutMillis); // 啟動動態超時檢測? ? } else {? ? ? ? queue.setTimeoutScheduled(false); // 豁免超時檢測? ? }? ? // 處理后臺Activity啟動權限的超時(新增特性)? ? if (r.mBackgroundStartPrivileges.allowsAny()) {? ? ? ? final long bgStartTimeout = r.isForeground() ? mFgConstants.ALLOW_BG_ACTIVITY_START_TIMEOUT? ? ? ? ? ? ? ? : mBgConstants.ALLOW_BG_ACTIVITY_START_TIMEOUT;? ? ? ? // 通過Handler延遲發送超時消息? ? ? ? mLocalHandler.sendMessageDelayed(? ? ? ? ? ? Message.obtain(mLocalHandler, MSG_BG_ACTIVITY_START_TIMEOUT, args), bgStartTimeout);? ? }}

ActivityManagerService定義了廣播ANR的雙閾值動態控制:

前臺廣播(mFgConstants.TIMEOUT):默認10秒,保障用戶體驗。

后臺廣播(mBgConstants.TIMEOUT):默認60秒,放寬限制以節省資源

public class ActivityManagerService extends IActivityManager.Stub? ? ? ? implements Watchdog.Monitor, BatteryStatsImpl.BatteryCallback, ActivityManagerGlobalLock {? ? static final int BROADCAST_FG_TIMEOUT = 10 * 1000 * Build.HW_TIMEOUT_MULTIPLIER; // 前臺廣播默認10秒? ? static final int BROADCAST_BG_TIMEOUT = 60 * 1000 * Build.HW_TIMEOUT_MULTIPLIER; // 后臺廣播默認60秒? ? public BroadcastQueue getBroadcastQueue(ActivityManagerService service) {? ? ? ? // Broadcast policy parameters? ? ? ? final BroadcastConstants foreConstants = new BroadcastConstants(? ? ? ? ? ? ? ? Settings.Global.BROADCAST_FG_CONSTANTS);? ? ? ? foreConstants.TIMEOUT = BROADCAST_FG_TIMEOUT;? ? ? ? final BroadcastConstants backConstants = new BroadcastConstants(? ? ? ? ? ? ? ? Settings.Global.BROADCAST_BG_CONSTANTS);? ? ? ? backConstants.TIMEOUT = BROADCAST_BG_TIMEOUT;? ? ? ? return new BroadcastQueueModernImpl(service, service.mHandler,? ? ? ? ? ? ? ? ? ? foreConstants, backConstants);? ? }}

2. 目標進程接收并處理廣播

源碼路徑:frameworks/base/core/java/android/app/LoadedApk.ReceiverDispatcher.java

public void performReceive(Intent intent) {? ? // 封裝為Runnable投遞到主線程消息隊列? ? Args args = new Args(intent, ...);? ? mActivityThread.post(args); // 使用主線程Handler}

3.?超時檢測取消入口(finishReceiverLocked)

源碼路徑:frameworks/base/services/core/java/com/android/server/am/BroadcastQueueModernImpl.java

??????????????

BroadcastQueueModernImpl.finishReceiverLocked/**?* 處理廣播接收完成或超時的核心邏輯,更新狀態并觸發后續操作?*??* @param queue 目標進程的廣播隊列(包含當前活躍的廣播記錄)?* @param deliveryState 廣播投遞狀態(如超時/DELIVERY_TIMEOUT、正常完成等)?* @param reason 狀態變更原因(用于日志和調試)?*/@GuardedBy("mService")private void finishReceiverActiveLocked(@NonNull BroadcastProcessQueue queue,? ? ? ? @DeliveryState int deliveryState, @NonNull String reason) {? ? // 1. 校驗隊列有效性:若當前無活躍廣播,直接返回? ? if (!queue.isActive()) {? ? ? ? logw("Ignoring finishReceiverActiveLocked; no active broadcast for " + queue);? ? ? ? return;? ? }? ? // 2. 獲取當前廣播上下文? ? final int cookie = traceBegin("finishReceiver"); // 性能追蹤標記? ? final ProcessRecord app = queue.app; // 目標進程記錄? ? final BroadcastRecord r = queue.getActive(); // 當前處理的廣播記錄? ? final int index = queue.getActiveIndex(); // 接收者索引? ? final Object receiver = r.receivers.get(index); // 具體的接收者對象? ? // 3. 更新投遞狀態(如標記超時或完成)? ? setDeliveryState(queue, app, r, index, receiver, deliveryState, reason);? ? // 4. 處理超時場景? ? if (deliveryState == BroadcastRecord.DELIVERY_TIMEOUT) {? ? ? ? r.anrCount++; // 記錄ANR次數? ? ? ? if (app != null && !app.isDebugging()) { // 非調試進程才觸發ANR? ? ? ? ? ? final AutoCloseable timer = mAnrTimer.accept(queue);? ? ? ? ? ? final String packageName = getReceiverPackageName(receiver);? ? ? ? ? ? final String className = getReceiverClassName(receiver);? ? ? ? ? ? // 構造ANR報告記錄? ? ? ? ? ? TimeoutRecord tr = TimeoutRecord.forBroadcastReceiver(r.intent, packageName,? ? ? ? ? ? ? ? ? ? className).setExpiredTimer(timer);? ? ? ? ? ? mService.appNotResponding(queue.app, tr); // 觸發ANR彈窗? ? ? ? } else {? ? ? ? ? ? mAnrTimer.discard(queue); // 調試進程忽略ANR? ? ? ? }? ? }?? ? // 5. 正常完成時取消超時檢測? ? else if (queue.timeoutScheduled()) {? ? ? ? cancelDeliveryTimeoutLocked(queue); // 關鍵:移除超時消息? ? }? ? // 6. 檢查是否有等待條件被滿足(如有序廣播的鏈式觸發)? ? checkAndRemoveWaitingFor();? ? traceEnd(cookie); // 結束性能追蹤}

1.廣播處理完成 → finishReceiverActiveLocked(DELIVERY_DONE, "正常完成")

→ cancelDeliveryTimeoutLocked()

├─ mAnrTimer.cancel()

└─ (舊版) Handler.removeMessages()

2.廣播超時 → finishReceiverActiveLocked(DELIVERY_TIMEOUT, "超時")

→ mService.appNotResponding()

→ 觸發ANR彈窗或日志記錄

4.觸發AMS的ANR彈窗

@Overridepublic void appNotResponding(@NonNull String processName, int uid,? ? ? ? @NonNull TimeoutRecord timeoutRecord) {? ? ActivityManagerService.this.appNotResponding(processName, uid, timeoutRecord);}

廣播ANR中系統優化方案參考

0

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/web/90951.shtml
繁體地址,請注明出處:http://hk.pswp.cn/web/90951.shtml
英文地址,請注明出處:http://en.pswp.cn/web/90951.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

密碼學中的概率論與統計學:從頻率分析到現代密碼攻擊

在密碼學的攻防博弈中,概率論與統計學始終是破解密碼的“利器”。從古典密碼時期通過字母頻率推測凱撒密碼的密鑰,到現代利用線性偏差破解DES的線性密碼分析,再到側信道攻擊中通過功耗數據的統計特性還原密鑰,統計思維貫穿了密碼分…

力扣刷題977——有序數組的平方

977. 有序數組的平方 題目: 給你一個按 非遞減順序 排序的整數數組 nums,返回 每個數字的平方 組成的新數組,要求也按 非遞減順序 排序。示例 1: 輸入:nums [-4,-1,0,3,10] 輸出:[0,1,9,16,100] 解釋&…

應用加速游戲盾的安全作用

在數字娛樂產業蓬勃發展的今天,游戲已從單純的娛樂工具演變為連接全球數十億用戶的社交平臺與文化載體。然而,伴隨游戲市場的指數級增長,網絡攻擊的頻率與復雜性也呈爆發式上升。從DDoS攻擊導致服務器癱瘓,到外掛程序破壞公平競技…

linux安裝zsh,oh-my-zsh,配置zsh主題及插件的方法

這是一份非常詳細的指南,帶你一步步在 Linux 系統中安裝 Zsh、配置主題和安裝插件。 Zsh(Z Shell)是一個功能強大的 Shell,相比于大多數 Linux 發行版默認的 Bash,它提供了更強的自定義能力、更智能的自動補全、更漂亮…

【設計模式系列】策略模式vs模板模式

策略模式是什么?如何定義并封裝一系列算法策略模式 (Strategy Pattern)模板模式 (Template Pattern)模板模式與策略模式的深度對比與區分混合使用兩種模式的場景策略模式 (Strategy Pattern) 應用場景:當需要根據不同條件選擇不同算法或行為時&#xff…

aigc(1.1) opensora-2.0

open sora-2.0相關鏈接: arxiv鏈接 huggingface頁面 HunyuanVideo VAE open sora2.0的VAE模型復用了HunyuanVideo的3D VAE,HunyuanVideo的arxiv鏈接。下圖來自論文,可見VAE是一個因果注意力的3D結構。在配圖左側,視頻會被編碼為video token序列,而在配圖右側,去噪的vide…

Linux驅動21 --- FFMPEG 音頻 API

目錄 一、FFMPEG 音頻 API 1.1 解碼步驟 創建核心上下文指針 打開輸入流 獲取輸入流 獲取解碼器 初始化解碼器 創建輸入流指針 創建輸出流指針 初始化 SDL 配置音頻參數 打開音頻設備 獲取一幀數據 發送給解碼器 從解碼器獲取數據 開辟數據空間 初始化內存 音頻重采樣…

《計算機“十萬個為什么”》之 [特殊字符] 序列化與反序列化:數據打包的奇妙之旅 ??

《計算機“十萬個為什么”》之 📦 序列化與反序列化:數據打包的奇妙之旅 ??歡迎來到計算機“十萬個為什么”系列! 本文將以「序列化與反序列化」為主題,深入探討計算機世界中數據的打包與解包過程。 讓我們一起解開數據的神秘面…

機器學習與深度學習評價指標

機器學習與深度學習評價指標完全指南 ?? 為什么需要評價指標? 想象你是一位醫生,需要判斷一個診斷模型的好壞。如果模型說"這個病人有癌癥",你需要知道: 這個判斷有多準確? 會不會漏掉真正的癌癥患者? 會不會誤診健康的人? 評價指標就像是給AI模型打分的&…

Hugging Face-環境配置

打開anaconda promptconda activate pytorchpip install -i https://pypi.tuna.tsinghua.edu.cn/simple transformers datasets tokenizerspycharm找到pytorch下的python.exe#將模型下載到本地調用 from transformers import AutoModelForCausalLM,AutoTokenizer#將模型和分詞工…

cnn中池化層作用

一、池化層概述 在卷積神經網絡中,池化層是核心組件之一,主要作用是逐步降低特征圖的空間尺寸即寬和高,從而減少計算量、控制過擬合并增強模型的魯棒性。 核心作用 降維與減少計算量 壓縮特征圖的尺寸,顯著減少后續層的參數數量和…

寫一個音樂爬蟲

今天我們寫一個網易云音樂的爬蟲,爬取網易云音樂熱歌榜音樂鏈接并下載,這里用到了之前引用的BeautifulSoup和requests。 BeautifulSoup是一個Python庫,用于從HTML和XML文件中提取數據。它提供了一種簡單的方式來遍歷文檔樹和搜索文檔樹中的元…

戰斗公式和傷害走配置文件

故事背景,上次屬性計算用的配置,這次傷害計算也走配置,下面是測試代碼和測試數據local formulas {[100001]{id 100001,name "基礎傷害",formula "function (self,tag,ishit,iscritial,counterratio)\n if ishit1 then\n …

線性代數 上

文章目錄線性代數知識整理一、求行列式1、 套公式2、利用性質,化為可套公式3、抽象行列式4、抽象向量二、代數余子式的線性組合三、求AnA^nAn四、證明A可逆五、求A的逆1、定義法2、初等變換3、公式六、求秩七、線性表示的判定八、線性無關九、求極大線性無關組十、等…

紅帽AI推理服務器三大特點

生成式人工智能(Gen AI)的迅猛發展,對大型語言模型(LLM)的部署提出了更高的性能、靈活性和效率要求。無論部署在何種環境中,紅帽AI推理服務器都為用戶提供經過強化并獲得官方支持的vLLM發行版,配…

開始記錄一步步學習pcl

安裝參考,大神寫的非常詳細,一步到位 https://blog.csdn.net/qq_36812406/article/details/144307648?ops_request_misc%257B%2522request%255Fid%2522%253A%25220e215e6ac266b90ded12ed6b2eab1257%2522%252C%2522scm%2522%253A%252220140713.13010233…

Linux系統Centos7 安裝mysql5.7教程 和mysql的簡單指令

目錄 一. 安裝 MySQL 官方 Yum 倉庫 二. 安裝 MySQL 5.7 1.查看可用的mysql版本倉庫 2.啟用MySql5.7倉庫 3.禁用更高版本的倉庫(可選) 4.導入 MySQL GPG 公鑰 5.安裝MySql5.7 三. 啟動 MySQL 服務 1.啟動 MySQL 服務 2. 設置開機自啟 3.查看服…

嵌入式——C語言:指針③

一、函數指針和指針函數(一)指針函數:是函數,函數的返回值是指針1.不能返回局部變量的值2.指針函數返回的地址可以作為下一個函數調用的參數(二)函數指針:是指針,指針指向一個函數in…

OpenCV(05)直方圖均衡化,模板匹配,霍夫變換,圖像亮度變換,形態學變換

【OpenCV(01)】基本圖像操作、繪制,讀取視頻 【OpenCV(02)】圖像顏色處理,灰度化,二值化,仿射變換 【OpenCV(03)】插值方法,邊緣填充&#xff0…

常見的未授權訪問漏洞靶場-練習教程

一.Redis未授權訪問漏洞1.首先需要準備一個vps和vulhub靶場,然后進入目錄。命令:進入靶場目錄:cd /etc/vulhub-master/redis/4-unacc 啟動靶場:docker-compose up -d2.然后啟動我們kali,下載redis服務然后連接redis,并執行命令。…