PMS安裝應(yīng)用程序流程
一葛躏、啟動時安裝
1、查看PMS的功能與啟動過程,首先PMS是在systemserver中啟動的讼积。之后分為兩大部分啟動
- 1)恢復(fù)之前的引用安裝信息:在main中new,開始調(diào)用readLP()恢復(fù)上一次引用程序安裝信息(讀取/data/system/packages.xml文件)----> ScanDirLP()掃描指定目錄---->readPackageLP()脚仔、addPackageLP()來讀取解析xml文件中的信息勤众。。鲤脏。们颜。。
- 2)應(yīng)用程序安裝過程:調(diào)用scanDirLI掃描指定目錄下(system/framework猎醇、system/app等)的文件時候有.apk的應(yīng)用窥突,----->scanPackageLI()解析.apk---> PackageParser.parsePackage()實(shí)現(xiàn)真正的解析工作(讀取AndroidManfest.xml文件等)----->解析正確---->調(diào)用另一個重載的scanPackageLI()來安裝應(yīng)用。
- 3)安裝好應(yīng)用之后硫嘶,updatePermissionsLPw()-----> grantPermissionsLPw()-------->requestedPermissions阻问,分配LINUX的用戶組ID,即資源訪問權(quán)限音半,最后writeLP()將應(yīng)用信息寫入到本地---->writePackage()將應(yīng)用安裝信息寫到/data/system/packages.xml中则拷,這正好一第一步中的readLP()與readPackageLP()形成閉環(huán)。
注意:
- (a) 重載的scanPackageLI()來安裝應(yīng)用后曹鸠,已經(jīng)安裝的應(yīng)用的四大組件都會記錄在PackageManagerService類的成員變量mActivitys煌茬,mReceivers,mServices彻桃,mProvidersByComponent中坛善。應(yīng)用使用Packages對象保存在mPackages所屬的HashMap中
- (b) AMS是負(fù)責(zé)管理應(yīng)用程序進(jìn)程的,在ActivityManagerService.java中會通過 startProcessLocked()函數(shù)中的 pm.getPackageGids(app.info.packageName)獲取需要創(chuàng)建應(yīng)用程序的LINUX用戶ID和LInux用戶組ID邻眷,此LINUX用戶ID和LInux用戶組ID正是PMS安裝應(yīng)用程序時創(chuàng)建的眠屎。
二、用戶安裝
adb install xxx.apk
PM : pm.java---->run()---->runInstall()
|
.....binder.....
|
PMS : installPackageAsUser()
|
消息機(jī)制
final Message msg = mHandler.obtainMessage(INIT_COPY);
msg.obj = new InstallParams(origin, observer, installFlags,
installerPackageName, verificationParams, user, packageAbiOverride);
mHandler.sendMessage(msg)
|
doHandleMessage(Message msg)
|
case INIT_COPY
|
connectToService()流程
然后再發(fā)送一個消息到 MCS_BOUND
|
case MCS_BOUND
|
params.startCopy()
|
handleReturnCode()
|
processPendingInstall(mArgs, mRet)
|
installPackageLI(args, res)
|
pkg = pp.parsePackage(tmpPackageFile, parseFlags) //用parsePackage解析APK
|
installNewPackageLI()
|
scanPackageLI() //重載
|
接下你就和啟動時安裝應(yīng)用流程一樣
.....
MCS_BOUND 消息發(fā)送流程
connectToService()的最后會在AMS中的publishServiceLocked()函數(shù)中通過ContextImpl傳過去的connection的IBinder接口肆饶,調(diào)用connected()函數(shù)
注意:
重載的scanPackageLI()函數(shù)
- 1改衩、private PackageParser.Package scanPackageLI(File scanFile, int parseFlags, int scanFlags, ...
- 2、private PackageParser.Package scanPackageLI(PackageParser.Package pkg, int parseFlags, ...
和啟動是被調(diào)用的scanPackageLI()不一樣驯镊,啟動是安裝要調(diào)用兩次scanPackageLI()重載函數(shù)竭鞍,而用戶安裝值調(diào)用2
發(fā)現(xiàn)問題:
Message消息循環(huán)
問題:sendMessage與handleMessage是怎么實(shí)現(xiàn)的?橄镜?偎快?
Android應(yīng)用程序在每一個線程啟動的時,都會在內(nèi)部創(chuàng)建一個消息隊(duì)列洽胶,然后進(jìn)入到無限循環(huán)中晒夹,不斷檢查消息隊(duì)列是否有新消息需要處理。如果有則會從消息隊(duì)列中取出來姊氓,處理丐怯。否則線程就會進(jìn)入睡眠狀態(tài)
connectToService()流程
doHandleMessage ----PMS
|
connectToService()
|
mContext.bindServiceAsUser() ----ContextImpl.java
|
ActivityManagerNative.getDefault().bindService() ----ContextImpl.java
|
{Singleton<IActivityManager> gDefault = new Singleton<IActivityManager>()
IBinder b = ServiceManager.getService("activity")
IActivityManager am = asInterface(b)}
|
IActivityManager.bindService() ----ActivityManagerNative
|
...binder...
|
ActivityManagerService.bindService() ----AMS
|
mServices.bindServiceLocked() ----ActiveServices.java
|
retrieveServiceLocked()
bringUpServiceLocked()
|
startProcessLocked()
|
啟動com.android.defcontainer/.DefaultContainerService 比較核心的拷貝/重命名/刪除都會在這個service中進(jìn)行