AMS handleApplicationCrash函數的代碼如下:
**ActivityManagerService.java::handleApplicationCrash**
~~~
public void handleApplicationCrash(IBinder app,
ApplicationErrorReport.CrashInfocrashInfo) {
//找到對應的ProcessRecord信息,后面那個字符串“Crash”用于打印輸出
ProcessRecord r = findAppProcess(app, "Crash");
finalString processName = app == null ? "system_server"
: (r == null ? "unknown": r.processName);
//添加crash信息到dropbox中
ddErrorToDropBox("crash", r, processName, null, null, null,null, null,
crashInfo);
//調用crashApplication函數
crashApplication(r, crashInfo);
}
~~~
上述代碼中的crashApplication 函數的代碼如下:
**ActivityManagerService.java::crashApplication**
~~~
private void crashApplication(ProcessRecord r,
ApplicationErrorReport.CrashInfocrashInfo) {
longtimeMillis = System.currentTimeMillis();
//從應用進程傳遞過來的crashInfo中獲取相關信息
StringshortMsg = crashInfo.exceptionClassName;
StringlongMsg = crashInfo.exceptionMessage;
StringstackTrace = crashInfo.stackTrace;
if(shortMsg != null && longMsg != null) {
longMsg = shortMsg + ": " + longMsg;
} elseif (shortMsg != null) {
longMsg = shortMsg;
}
AppErrorResult result = new AppErrorResult();
synchronized (this) {
if(mController != null) {
//通知監視者。目前僅MonkeyTest中會為AMS設置監聽者。測試過程中可設定是否一檢測
//到App Crash即停止測試。另外,Monkey測試也會將App Crash信息保存起來
//供開發人員分析
}
final long origId =Binder.clearCallingIdentity();
if (r!= null && r.instrumentationClass != null) {
......// instrumentationClass不為空的情況
}
//①調用makeAppCrashingLocked處理,如果返回false,則整個處理流程完畢
if (r == null || !makeAppCrashingLocked(r,shortMsg, longMsg,
stackTrace)) {
......
return;
}
Message msg = Message.obtain();
msg.what = SHOW_ERROR_MSG;
HashMap data = new HashMap();
data.put("result", result);
data.put("app", r);
msg.obj = data;
//發送SHOW_ERROR_MSG消息給mHandler,默認處理是彈出一個對話框,提示用戶某進程 //崩潰(Crash),用戶可以選擇“退出”或 “退出并報告”
mHandler.sendMessage(msg);
......
}//synchronized(this)結束
//下面這個get函數是阻塞的,直到用戶處理了對話框為止。注意,此處涉及兩個線程:
//handleApplicationCrash函數是在Binder調用線程中處理的,而對話框則是在mHandler所
//在線程中處理的
int res =result.get();
IntentappErrorIntent = null;
synchronized (this) {
if (r!= null)
mProcessCrashTimes.put(r.info.processName, r.info.uid,
SystemClock.uptimeMillis());
if (res== AppErrorDialog.FORCE_QUIT_AND_REPORT)
//createAppErrorIntentLocked返回一個Intent,該Intent的Action是
//"android.intent.action.APP_ERROR",指定接收者是app. errorReportReceiver
//成員,該成員變量在關鍵點makeAppCrashingLocked中被設置
appErrorIntent =createAppErrorIntentLocked(r, timeMillis,
crashInfo);
}//synchronized(this)結束
if(appErrorIntent != null) {
try {//啟動能處理APP_ERROR的應用進程,目前的源碼中還沒有地方處理它
mContext.startActivity(appErrorIntent);
} ......
}
}
~~~
以上代碼中還有一個關鍵函數makeAppCrashingLocked,其代碼如下:
**ActivityManagerService.java::makeAppCrashingLocked**
~~~
private booleanmakeAppCrashingLocked(ProcessRecord app,
String shortMsg, String longMsg, String stackTrace) {
app.crashing = true;
//生成一個錯誤報告,存放在crashingReport變量中
app.crashingReport = generateProcessError(app,
ActivityManager.ProcessErrorStateInfo.CRASHED, null, shortMsg,
longMsg, stackTrace);
/*
在上邊代碼中,我們知道系統會通過APP_ERROR Intent啟動一個Activity去處理錯誤報告,
實際上在此之前,系統需要為它找到指定的接收者(如果有)。代碼在startAppProblemLocked
中,此處簡單描述該函數的處理過程如下:
1 先查詢Settings Secure表中“send_action_app_error”是否使能,如果沒有使能,則
不能設置指定接收者
2 通過PKMS查詢安裝此Crash應用的應用是否能接收APP_ERROR Intent。必須注意
安裝此應用的應用(例如通過“安卓市場”安裝了該Crash應用)。如果沒有,則轉下一步處理
3 查詢系統屬性“ro.error.receiver.system.apps”是否指定了APP_ERROR處理者,如果
沒有,則轉下一步處理
4 查詢系統屬性“ro.error.receiver.default”是否指定了默認的處理者
5 處理者的信息保存在app的errorReportReceiver變量中
另外,如果該Crash應用正好是串行廣播發送處理中的一員,則需要結束它的處理流程以
保證后續廣播處理能正常執行。讀者可參考skipCurrentReceiverLocked函數
*/
startAppProblemLocked(app);
app.stopFreezingAllLocked();
//調用handleAppCrashLocked做進一步處理。讀者可自行閱讀
returnhandleAppCrashLocked(app);
}
~~~
當App的Crash處理完后,事情并未就此結束,因為該應用進程退出后,之前AMS為它設置的訃告接收對象將被喚醒。接下來介紹AppDeathRecipientbinderDied的處理流程。
- 前言
- 第1章 搭建Android源碼工作環境
- 1.1 Android系統架構
- 1.2 搭建開發環境
- 1.2.1 下載源碼
- 1.2.2 編譯源碼
- 1.2.3 利用Eclipse調試system_process
- 1.3 本章小結
- 第2章 深入理解Java Binder和MessageQueue
- 2.1 概述
- 2.2 Java層中的Binder架構分析
- 2.2.1 Binder架構總覽
- 2.2.2 初始化Java層Binder框架
- 2.2.3 addService實例分析
- 2.2.4 Java層Binder架構總結
- 2.3 心系兩界的MessageQueue
- 2.3.1 MessageQueue的創建
- 2.3.2 提取消息
- 2.3.3 nativePollOnce函數分析
- 2.3.4 MessageQueue總結
- 2.4 本章小結
- 第3章 深入理解SystemServer
- 3.1 概述
- 3.2 SystemServer分析
- 3.2.1 main函數分析
- 3.2.2 Service群英會
- 3.3 EntropyService分析
- 3.4 DropBoxManagerService分析
- 3.4.1 DBMS構造函數分析
- 3.4.2 dropbox日志文件的添加
- 3.4.3 DBMS和settings數據庫
- 3.5 DiskStatsService和DeviceStorageMonitorService分析
- 3.5.1 DiskStatsService分析
- 3.5.2 DeviceStorageManagerService分析
- 3.6 SamplingProfilerService分析
- 3.6.1 SamplingProfilerService構造函數分析
- 3.6.2 SamplingProfilerIntegration分析
- 3.7 ClipboardService分析
- 3.7.1 復制數據到剪貼板
- 3.7.2 從剪切板粘貼數據
- 3.7.3 CBS中的權限管理
- 3.8 本章小結
- 第4章 深入理解PackageManagerService
- 4.1 概述
- 4.2 初識PackageManagerService
- 4.3 PKMS的main函數分析
- 4.3.1 構造函數分析之前期準備工作
- 4.3.2 構造函數分析之掃描Package
- 4.3.3 構造函數分析之掃尾工作
- 4.3.4 PKMS構造函數總結
- 4.4 APK Installation分析
- 4.4.1 adb install分析
- 4.4.2 pm分析
- 4.4.3 installPackageWithVerification函數分析
- 4.4.4 APK 安裝流程總結
- 4.4.5 Verification介紹
- 4.5 queryIntentActivities分析
- 4.5.1 Intent及IntentFilter介紹
- 4.5.2 Activity信息的管理
- 4.5.3 Intent 匹配查詢分析
- 4.5.4 queryIntentActivities總結
- 4.6 installd及UserManager介紹
- 4.6.1 installd介紹
- 4.6.2 UserManager介紹
- 4.7 本章學習指導
- 4.8 本章小結
- 第5章 深入理解PowerManagerService
- 5.1 概述
- 5.2 初識PowerManagerService
- 5.2.1 PMS構造函數分析
- 5.2.2 init分析
- 5.2.3 systemReady分析
- 5.2.4 BootComplete處理
- 5.2.5 初識PowerManagerService總結
- 5.3 PMS WakeLock分析
- 5.3.1 WakeLock客戶端分析
- 5.3.2 PMS acquireWakeLock分析
- 5.3.3 Power類及LightService類介紹
- 5.3.4 WakeLock總結
- 5.4 userActivity及Power按鍵處理分析
- 5.4.1 userActivity分析
- 5.4.2 Power按鍵處理分析
- 5.5 BatteryService及BatteryStatsService分析
- 5.5.1 BatteryService分析
- 5.5.2 BatteryStatsService分析
- 5.5.3 BatteryService及BatteryStatsService總結
- 5.6 本章學習指導
- 5.7 本章小結
- 第6章 深入理解ActivityManagerService
- 6.1 概述
- 6.2 初識ActivityManagerService
- 6.2.1 ActivityManagerService的main函數分析
- 6.2.2 AMS的 setSystemProcess分析
- 6.2.3 AMS的 installSystemProviders函數分析
- 6.2.4 AMS的 systemReady分析
- 6.2.5 初識ActivityManagerService總結
- 6.3 startActivity分析
- 6.3.1 從am說起
- 6.3.2 AMS的startActivityAndWait函數分析
- 6.3.3 startActivityLocked分析
- 6.4 Broadcast和BroadcastReceiver分析
- 6.4.1 registerReceiver流程分析
- 6.4.2 sendBroadcast流程分析
- 6.4.3 BROADCAST_INTENT_MSG消息處理函數
- 6.4.4 應用進程處理廣播分析
- 6.4.5 廣播處理總結
- 6.5 startService之按圖索驥
- 6.5.1 Service知識介紹
- 6.5.2 startService流程圖
- 6.6 AMS中的進程管理
- 6.6.1 Linux進程管理介紹
- 6.6.2 關于Android中的進程管理的介紹
- 6.6.3 AMS進程管理函數分析
- 6.6.4 AMS進程管理總結
- 6.7 App的 Crash處理
- 6.7.1 應用進程的Crash處理
- 6.7.2 AMS的handleApplicationCrash分析
- 6.7.3 AppDeathRecipient binderDied分析
- 6.7.4 App的Crash處理總結
- 6.8 本章學習指導
- 6.9 本章小結
- 第7章 深入理解ContentProvider
- 7.1 概述
- 7.2 MediaProvider的啟動及創建
- 7.2.1 Context的getContentResolver函數分析
- 7.2.2 MediaStore.Image.Media的query函數分析
- 7.2.3 MediaProvider的啟動及創建總結
- 7.3 SQLite創建數據庫分析
- 7.3.1 SQLite及SQLiteDatabase家族
- 7.3.2 MediaProvider創建數據庫分析
- 7.3.3 SQLiteDatabase創建數據庫的分析總結
- 7.4 Cursor 的query函數的實現分析
- 7.4.1 提取query關鍵點
- 7.4.2 MediaProvider 的query分析
- 7.4.3 query關鍵點分析
- 7.4.4 Cursor query實現分析總結
- 7.5 Cursor close函數實現分析
- 7.5.1 客戶端close的分析
- 7.5.2 服務端close的分析
- 7.5.3 finalize函數分析
- 7.5.4 Cursor close函數總結
- 7.6 ContentResolver openAssetFileDescriptor函數分析
- 7.6.1 openAssetFileDescriptor之客戶端調用分析
- 7.6.2 ContentProvider的 openTypedAssetFile函數分析
- 7.6.3 跨進程傳遞文件描述符的探討
- 7.6.4 openAssetFileDescriptor函數分析總結
- 7.7 本章學習指導
- 7.8 本章小結
- 第8章 深入理解ContentService和AccountManagerService
- 8.1 概述
- 8.2 數據更新通知機制分析
- 8.2.1 初識ContentService
- 8.2.2 ContentResovler 的registerContentObserver分析
- 8.2.3 ContentResolver的 notifyChange分析
- 8.2.4 數據更新通知機制總結和深入探討
- 8.3 AccountManagerService分析
- 8.3.1 初識AccountManagerService
- 8.3.2 AccountManager addAccount分析
- 8.3.3 AccountManagerService的分析總結
- 8.4 數據同步管理SyncManager分析
- 8.4.1 初識SyncManager
- 8.4.2 ContentResolver 的requestSync分析
- 8.4.3 數據同步管理SyncManager分析總結
- 8.5 本章學習指導
- 8.6 本章小結