PackageManagerService(PmS)包管理服务运行在SystemServer进程中,是一个安卓系统服务,主要用于实现应用安装卸载,组件查询匹配,权限管理等功能。
主要功能
根据Intent匹配到具体的Activity,Provider,Service,即当应用程序调用startActivity(intent)方法时,能够把Intent转换成一个具体的包含程序名称及Component的信息,以便类加载器加载具体的Component。
权限检查,当应用程序调用某个需要特定权限的接口时,判断调用者是否有该权限
提供安装删除应用程序的接口
实现原理
应用安装时读取应用程序AndroidManifest.xml中的标签比如,request-feature、permission并将其保存在指定目录文件下,PmS在启动时会读取这些xml文件建立起一个包信息树,应用程序可以间接的从信息树种查到所需要的程序包信息。
两个目录
1./data/system/package.xml
文件记录系统中所有应用程序包管理相关信息,比如程序包名称是什么,安装包路径在哪里,程序都使用了哪些权限,等等。
2./system/etc/permissions/
文件夹下保存的xml文件用于应用程序权限管理。
PmS在启动时会读取这两个文件来构建应用程序包信息树。PmS读取的数据会存在其内部类变量中。
数据结构
PmS的内部类Settings基本上包含了包管理所需要的全部信息,该类主要包含几类变量:
- 包属性信息,包括packages.xml配置文件,配置文件备份,应用程序列表文件,包管理信息。
- 用户Id相关信息,所有用户Id共享Id等,
- 权限管理相关信息,保存所有的签名,所有的权限。
- 删除信息,应用程序被卸载后,如果该程序的数据保存在外部存储空间中,则其数据目录默认不被删除。
以上信息,均是从packages.xml配置文件中解析而来。
PmS类中还包括以下重要数据:
- 扫描应用程序目录得到的程序包信息
- 系统所有权限名称
- 系统所依赖的共享Java库
- 从应用程序AndroidManifest.xml中解析出来的Activity,Service,BroadcastReceiver列表
PmS在启动时会遍历应用程序目录下的所有程序,并从AndroidManifest.xml中提取出Intent-filter数据并将其保存在一个列表中,用于进行Intent-filter(startXXX(intent)
)匹配。
关键方法
1.readPermission()
从/system/etc/permissions/
目录下读取系统中定义的所有feature列表,给系统一些native分配权限信息。
2.mSettings.readLP()
从packages.xml
中读取所有安装包信息。其中会涉及到package-backup.xml
文件的处理。packages.xml
中使用各种各样的标签存储应用程序信息。
3.mSettings.writeLP()
应用程序包扫描已经应用程序安装。应用程序安装分为两步,第一是把原始APK复制到相应程序目录下,第二则是为应用程序创建相应的数据目录及提取dex文件,并修改系统包管理信息等。mSettings.writeLP()主要完成第二步。
4.scanPackageLI()
将mSettings.mPackages的数据写到packages.xml和packages.list文件中。
创建PmS
SystemServer启动时,PmS在从其静态main函数中创建,并将自己添加到系统服务中:
public static final IPackageManager main(Context context,boolean factoryTest){
PackageManagerService m = new PackageManagerService();
SystemManager.addService("package",m);
return m;
}
启动PmS
- 创建PmS.Settings数据对象并赋值
- 创建Installer对象,该对象主要用于辅佐应用程序安装
- 为几个静态数据文件路径变量赋值
- 调用readPermission()方法从
/system/etc/permissions
目录下读取并解析全部XML文件 - 调用mSettings.readLP()方法从
/data/system/packages.xml
文件中读取应用程序包管理相关信息。 - 提取或者转换Java系统库中的dex文件
- 为
/system/framwworks
,/system/app
,/vendor/app
目录添加FileObserver,FileObserver用于检测目录天剑删除的问价事件。 - 调用scanDieLP()扫描上述三个目录中的所有应用程序,并将扫描结果保存到PmS的mPackages变量中。
- 删除已经不存在的应用程序对应的数据记录
- 清除没有安装成功的数据记录
- 为
/data/app
添加FileObserver - 检测是否系统升级,如果升过则重新为应用程序设置权限
- 将mSettings.mPackages中的数据从新写入packages.xml中
获取PmS
通过ContextImpl.getPackageManager()
返回一个PackageManager
对象,然后就可以调用改对象提供的各种API接口。
getPackageManager
内部获取过程和getSystemService
的过程基本相似,都是通过ServiceManager获取指定名称的IBinder对象进而获取PmS服务。(普通服务通过context.getSystemService("serviceName")
获取)