作為核心服務,AMS的systemReady會做什么呢?由于該函數內容較多,我們將它分為三段。首先看第一段的工作。
1. systemReady第一階段的工作
**ActivityManagerService.java::systemReady**
~~~
public void systemReady(final RunnablegoingCallback) {
synchronized(this){
......
if(!mDidUpdate) {//判斷是否為升級
if(mWaitingUpdate) return; //升級未完成,直接返回
//準備PRE_BOOT_COMPLETED廣播
Intent intent = newIntent(Intent.ACTION_PRE_BOOT_COMPLETED);
List<ResolveInfo> ris = null;
//向PKMS查詢該廣播的接收者
ris= AppGlobals.getPackageManager().queryIntentReceivers(
intent, null,0);
......//從返回的結果中刪除那些非系統APK的廣播接收者
intent.addFlags(Intent.FLAG_RECEIVER_BOOT_UPGRADE);
//讀取/data/system/called_pre_boots.dat文件,這里存儲了上次啟動時候已經
//接收并處理PRE_BOOT_COMPLETED廣播的組件。鑒于該廣播的特殊性,系統希望
//該廣播僅被這些接收者處理一次
ArrayList<ComponentName>lastDoneReceivers =
readLastDonePreBootReceivers();
final ArrayList<ComponentName> doneReceivers=
newArrayList<ComponentName>();
......//從PKMS返回的接收者中刪除那些已經處理過該廣播的對象
for (int i=0; i<ris.size(); i++) {
ActivityInfo ai = ris.get(i).activityInfo;
ComponentName comp = newComponentName(ai.packageName, ai.name);
doneReceivers.add(comp);
intent.setComponent(comp);
IIntentReceiver finisher = null;
if (i == ris.size()-1) {
//為最后一個廣播接收者注冊一個回調通知,當該接收者處理完廣播后,將調用該
//回調
finisher = new IIntentReceiver.Stub() {
public voidperformReceive(Intent intent, int resultCode,
Stringdata, Bundle extras, boolean ordered,
booleansticky) {
mHandler.post(newRunnable() {
public void run(){
synchronized(ActivityManagerService.this) {
mDidUpdate = true;
}
//保存那些處理過該廣播的接收者信息
writeLastDonePreBootReceivers(doneReceivers);
showBootMessage(mContext.getText(
R.string.android_upgrading_complete),
false);
systemReady(goingCallback);
}//run結束
});// new Runnable結束
}//performReceive結束
};//finisher創建結束
}// if (i == ris.size()-1)判斷結束
//發送廣播給指定的接收者
broadcastIntentLocked(null, null, intent,null, finisher,
0, null, null, null, true, false, MY_PID, Process.SYSTEM_UID);
if (finisher != null) mWaitingUpdate = true;
}
if(mWaitingUpdate) return;
mDidUpdate= true;
}
mSystemReady = true;
if(!mStartRunning) return;
}//synchronized(this)結束
~~~
由以上代碼可知,systemReady第一階段的工作并不輕松,其主要職責是發送并處理與PRE_BOOT_COMPLETED廣播相關的事情。目前代碼中還沒有接收該廣播的地方,不過從代碼中的注釋中可猜測到,該廣播接收者的工作似乎和系統升級有關。
建議如有哪位讀者了解與此相關的知識,不妨和大家分享。
下面來介紹systemReady第二階段的工作。
2. systemReady第二階段的工作
**ActivityManagerService.java::systemReady**
~~~
ArrayList<ProcessRecord>procsToKill = null;
synchronized(mPidsSelfLocked) {
for(int i=mPidsSelfLocked.size()-1; i>=0; i--) {
ProcessRecord proc = mPidsSelfLocked.valueAt(i);
//從mPidsSelfLocked中找到那些先于AMS啟動的進程,哪些進程有如此能耐,
//在AMS還未啟動完畢就啟動完了呢?對,那些聲明了persistent為true的進程有可能
if(!isAllowedWhileBooting(proc.info)){
if (procsToKill == null)
procsToKill = new ArrayList<ProcessRecord>();
procsToKill.add(proc);
}
}//for結束
}// synchronized結束
synchronized(this){
if(procsToKill != null) {
for (int i=procsToKill.size()-1; i>=0; i--) {
ProcessRecord proc = procsToKill.get(i);
//把這些進程關閉,removeProcessLocked函數比較復雜,以后再分析
removeProcessLocked(proc, true, false);
}
}
//至此,系統已經準備完畢
mProcessesReady = true;
}
synchronized(this) {
if(mFactoryTest == SystemServer.FACTORY_TEST_LOW_LEVEL) {
}//和工廠測試有關,不對此進行討論
}
//查詢Settings數據,獲取一些配置參數
retrieveSettings();
~~~
systemReady第二階段的工作包括:
- 殺死那些竟然在AMS還未啟動完畢就先啟動的應用進程。注意,這些應用進程一定是APK所在的Java進程,因為只有應用進程才會向AMS注冊,而一般Native(例如mediaserver)進程是不會向AMS注冊的。
- 從Settings數據庫中獲取配置信息,目前只取4個配置參數,分別是:"debug_app"(設置需要debug的app的名稱)、"wait_for_debugger"(如果為1,則等待調試器,否則正常啟動debug_app)、"always_finish_activities"(當一個activity不再有地方使用時,是否立即對它執行destroy)、"font_scale"(用于控制字體放大倍數,這是Android 4.0新增的功能)。以上配置項由Settings數據庫的System表提供。
3. systemReady第三階段的工作
**ActivityManagerService.java::systemReady**
~~~
//調用systemReady傳入的參數,它是一個Runnable對象,下節將分析此函數
if (goingCallback != null) goingCallback.run();
synchronized (this) {
if(mFactoryTest != SystemServer.FACTORY_TEST_LOW_LEVEL) {
try{
//從PKMS中查詢那些persistent為1的ApplicationInfo
List apps = AppGlobals.getPackageManager().
getPersistentApplications(STOCK_PM_FLAGS);
if (apps != null) {
int N = apps.size();
int i;
for (i=0; i<N; i++) {
ApplicationInfo info = (ApplicationInfo)apps.get(i);
//由于framework-res.apk已經由系統啟動,所以這里需要把它去除
//framework-res.apk的packageName為"android"
if (info != null && !info.packageName.equals("android"))
addAppLocked(info);//啟動該Application所在的進程
}
}
}......
}
mBooting= true; //設置mBooting變量為true,其作用后面會介紹
try {
if(AppGlobals.getPackageManager().hasSystemUidErrors()) {
......//處理那些Uid有錯誤的Application
}......
//啟動全系統第一個Activity,即Home
mMainStack.resumeTopActivityLocked(null);
}// synchronized結束
}
~~~
systemReady第三階段的工作有3項:
- 調用systemReady設置的回調對象goingCallback的run函數。
- 啟動那些聲明了persistent的APK。
- 啟動桌面。
先看回調對象goingCallback的run函數的工作。
(1) goingCallback的run函數分析
**SystemServer.java::ServerThread.run**
~~~
ActivityManagerService.self().systemReady(newRunnable() {
publicvoid run() {
startSystemUi(contextF);//啟動SystemUi
//調用其他服務的systemReady函數
if(batteryF != null) batteryF.systemReady();
if(networkManagementF != null) networkManagementF.systemReady();
......
Watchdog.getInstance().start();//啟動Watchdog
......//調用其他服務的systemReady函數
}
~~~
run函數比較簡單,執行的工作如下:
- 執行startSystemUi,在該函數內部啟動SystemUIService,該Service和狀態欄有關。
- 調用一些服務的systemReady函數。
- 啟動Watchdog。
startSystemUi的代碼如下:
`SystemServer.java::startSystemUi`
~~~
static final void startSystemUi(Context context) {
Intentintent = new Intent();
intent.setComponent(new ComponentName("com.android.systemui",
"com.android.systemui.SystemUIService"));
context.startService(intent);
}
~~~
SystemUIService由SystemUi.apk提供,它實現了系統的狀態欄。
* * * * *
**注意**:在精簡ROM時,也不能刪除SystemUi.apk。
* * * * *
(2) 啟動Home界面
如前所述,resumeTopActivityLocked將啟動Home界面,此函數非常重要也比較復雜,故以后再詳細分析。我們提取了resumeTopActivityLocked啟動Home界面時的相關代碼,如下所示:
**ActivityStack.java::resumeTopActivityLocked**
~~~
final booleanresumeTopActivityLocked(ActivityRecord prev) {
//找到下一個要啟動的Activity
ActivityRecord next = topRunningActivityLocked(null);
finalboolean userLeaving = mUserLeaving;
mUserLeaving = false;
if (next== null) {
//如果下一個要啟動的ActivityRecord為空,則啟動Home
if(mMainStack) {//全系統就一個ActivityStack,所以mMainStack永遠為true
//mService指向AMS
return mService.startHomeActivityLocked();//mService指向AMS
}
}
......//以后再詳細分析
}
~~~
下面來看AMS的startHomeActivityLocked函數,代碼如下:
**ActivityManagerService.java::startHomeActivityLocked**
~~~
boolean startHomeActivityLocked() {
Intentintent = new Intent( mTopAction,
mTopData != null ? Uri.parse(mTopData) :null);
intent.setComponent(mTopComponent);
if(mFactoryTest != SystemServer.FACTORY_TEST_LOW_LEVEL)
intent.addCategory(Intent.CATEGORY_HOME);//添加Category為HOME類別
//向PKMS查詢滿足條件的ActivityInfo
ActivityInfo aInfo =
intent.resolveActivityInfo(mContext.getPackageManager(),
STOCK_PM_FLAGS);
if(aInfo != null) {
intent.setComponent(new ComponentName(
aInfo.applicationInfo.packageName,aInfo.name));
ProcessRecordapp = getProcessRecordLocked(aInfo.processName,
aInfo.applicationInfo.uid);
//在正常情況下,app應該為null,因為剛開機,Home進程肯定還沒啟動
if(app == null || app.instrumentationClass == null) {
intent.setFlags(intent.getFlags()|
Intent.FLAG_ACTIVITY_NEW_TASK);
//啟動Home
mMainStack.startActivityLocked(null, intent, null, null, 0, aInfo,
null, null, 0, 0, 0, false, false,null);
}
}//if(aInfo != null)判斷結束
return true;
}
~~~
至此,AMS攜諸位Service都啟動完畢,Home也靚麗登場,整個系統就準備完畢,只等待用戶的檢驗了。不過在分析邏輯上還有一點沒涉及,那會是什么呢?
(3) 發送ACTION_BOOT_COMPLETED廣播
由前面的代碼可知,AMS發送了ACTION_PRE_BOOT_COMPLETED廣播,可系統中沒有地方處理它。在前面的章節中,還碰到一個ACTION_BOOT_COMPLETED廣播,該廣播廣受歡迎,卻不知道它是在哪里發送的。
當Home Activity啟動后,ActivityStack的activityIdleInternal函數將被調用,其中有一句代碼頗值得注意:
**ActivityStack.java::activityIdleInternal**
~~~
final ActivityRecord activityIdleInternal(IBindertoken, boolean fromTimeout,
Configuration config) {
booleanbooting = false;
......
if(mMainStack) {
booting = mService.mBooting; //在systemReady的第三階段工作中設置該值為true
mService.mBooting = false;
}
......
if(booting) mService.finishBooting();//調用AMS的finishBooting函數
}
~~~
**ActivityManagerService.java::finishBooting**
~~~
final void finishBooting() {
IntentFilter pkgFilter = new IntentFilter();
pkgFilter.addAction(Intent.ACTION_QUERY_PACKAGE_RESTART);
pkgFilter.addDataScheme("package");
mContext.registerReceiver(new BroadcastReceiver() {
public void onReceive(Context context, Intent intent) {
......//處理Package重啟的廣播
}
}, pkgFilter);
synchronized (this) {
finalint NP = mProcessesOnHold.size();
if (NP> 0) {
ArrayList<ProcessRecord> procs =
new ArrayList<ProcessRecord>(mProcessesOnHold);
for(int ip=0; ip<NP; ip++)//啟動那些等待啟動的進程
startProcessLocked(procs.get(ip), "on-hold", null);
}
if(mFactoryTest != SystemServer.FACTORY_TEST_LOW_LEVEL) {
//每15鐘檢查系統各應用進程使用電量的情況,如果某進程使用WakeLock時間
//過長,AMS將關閉該進程
Message nmsg =
mHandler.obtainMessage(CHECK_EXCESSIVE_WAKE_LOCKS_MSG);
mHandler.sendMessageDelayed(nmsg, POWER_CHECK_DELAY);
//設置系統屬性sys.boot_completed的值為1
SystemProperties.set("sys.boot_completed","1");
//發送ACTION_BOOT_COMPLETED廣播
broadcastIntentLocked(null, null,
newIntent(Intent.ACTION_BOOT_COMPLETED, null),
null, null, 0, null,null,
android.Manifest.permission.RECEIVE_BOOT_COMPLETED,
false, false, MY_PID,Process.SYSTEM_UID);
}
}
}
~~~
原來,在Home啟動成功后,AMS才發送ACTION_BOOT_COMPLETED廣播。
4. ASM的 systemReady總結
systemReady函數完成了系統就緒的必要工作,然后它將啟動Home Activity。至此,Android系統就全部啟動了。
- 前言
- 第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 本章小結