Android 包管理服务

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基本上包含了包管理所需要的全部信息,该类主要包含几类变量:

  1. 包属性信息,包括packages.xml配置文件,配置文件备份,应用程序列表文件,包管理信息。
  2. 用户Id相关信息,所有用户Id共享Id等,
  3. 权限管理相关信息,保存所有的签名,所有的权限。
  4. 删除信息,应用程序被卸载后,如果该程序的数据保存在外部存储空间中,则其数据目录默认不被删除。

以上信息,均是从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

  1. 创建PmS.Settings数据对象并赋值
  2. 创建Installer对象,该对象主要用于辅佐应用程序安装
  3. 为几个静态数据文件路径变量赋值
  4. 调用readPermission()方法从/system/etc/permissions目录下读取并解析全部XML文件
  5. 调用mSettings.readLP()方法从/data/system/packages.xml文件中读取应用程序包管理相关信息。
  6. 提取或者转换Java系统库中的dex文件
  7. /system/framwworks,/system/app,/vendor/app目录添加FileObserver,FileObserver用于检测目录天剑删除的问价事件。
  8. 调用scanDieLP()扫描上述三个目录中的所有应用程序,并将扫描结果保存到PmS的mPackages变量中。
  9. 删除已经不存在的应用程序对应的数据记录
  10. 清除没有安装成功的数据记录
  11. /data/app添加FileObserver
  12. 检测是否系统升级,如果升过则重新为应用程序设置权限
  13. 将mSettings.mPackages中的数据从新写入packages.xml中

获取PmS

通过ContextImpl.getPackageManager()返回一个PackageManager对象,然后就可以调用改对象提供的各种API接口。

getPackageManager内部获取过程和getSystemService的过程基本相似,都是通过ServiceManager获取指定名称的IBinder对象进而获取PmS服务。(普通服务通过context.getSystemService("serviceName")获取)

    原文作者:Gooooood
    原文地址: https://www.jianshu.com/p/4c40237e605c
    本文转自网络文章,转载此文章仅为分享知识,如有侵权,请联系博主进行删除。
点赞