深入Android系统(十)PMS-2-初始化的一些细节

permission文件的处理

PMS的构造方法中初始化了PermissionManagerService,通过PermissionManagerService.create()方法,相关调用如下:

    public static PermissionManagerInternal create(......) {
   
   
        ......
        new PermissionManagerService(context, defaultGrantCallback, externalLock);
        ......
    }
    PermissionManagerService(......) {
   
   
        ......
        mHandlerThread = new ServiceThread(TAG,
                Process.THREAD_PRIORITY_BACKGROUND, true /*allowIo*/);
        mHandlerThread.start();
        mHandler = new Handler(mHandlerThread.getLooper());
        Watchdog.getInstance().addThread(mHandler);
        ......
        SystemConfig systemConfig = SystemConfig.getInstance();
        mSystemPermissions = systemConfig.getSystemPermissions();
        ......
    }

上面列出来PermissionManagerService初始化的过程中两个操作:

  • 注册Watchdog监听,此处列出它来主要是为了呼应前面的章节。哈哈哈,重点是第二点
  • 通过SystemConfiggetSystemPermissions方法来获取系统的Permission列表
    public SparseArray<ArraySet<String>> getSystemPermissions() {
         
         
        return mSystemPermissions;
    }
    

而对于mSystemPermissions,是在SystemConfig的构造方法中完成的,相关代码如下:

    //单例模式
    SystemConfig() {
   
   
        // 读取 /system/etc/sysconfig 目录下的 permission 文件
        readPermissions(Environment.buildPath(
                Environment.getRootDirectory(), "etc", "sysconfig"), ALLOW_ALL);
        // 读取 /system/etc/permissions 目录下的 permission 文件
        readPermissions(Environment.buildPath(
                Environment.getRootDirectory(), "etc", "permissions"), ALLOW_ALL);
        ......
        // 读取 /vendor/etc/sysconfig 目录下的 permission 文件
        readPermissions(Environment.buildPath(
                Environment.getVendorDirectory(), "etc", "sysconfig"), vendorPermissionFlag);
        // 读取 /vendor/etc/permissions 目录下的 permission 文件
        readPermissions(Environment.buildPath(
                Environment.getVendorDirectory(), "etc", "permissions"), vendorPermissionFlag);
        ...// 省略 /oem、/odm、/product对应分区目录下 readPermissions()
    }

不用说,核心是这个readPermissions()方法,看看:

    void readPermissions(File libraryDir, int permissionFlag) {
   
   
        // Read permissions from given directory.
        if (!libraryDir.exists() || !libraryDir.isDirectory()) {
   
   
            return;// 如果目录不存在或者libraryDir不是目录,直接返回
        }
        if (!libraryDir.canRead()) {
   
   
            return;// 如果目录不可读,直接返回
        }
        File platformFile = null;
        // 循环能处理目录文件
        for (File f : libraryDir.listFiles()) {
   
   
            if (f.getPath().endsWith("etc/permissions/platform.xml")) {
   
   
                // 检测到platform.xml文件,先记录,循环中不做处理
                platformFile = f;
                continue;
            }
            if (!f.getPath().endsWith(".xml")) {
   
   
                // 跳过非xml文件
                continue;
            }
            if (!f.canRead()) {
   
   
                // 跳过不可读文件
                continue;
            }
            readPermissionsFromXml(f, permissionFlag);// 解析xml
        }
        // 最后,解析platform.xml文件
        if (platformFile != null) {
   
   
            readPermissionsFromXml(platformFile, permissionFlag);
        }
    }

readPermissions()方法先检测指定目录下的xml文件,然后调用readPermissionsFromXml方法来解析文件并将解析结果填充到SystemConfig对应的数据结构中。

readPermissionsFromXml()方法内容如下:

    private void readPermissionsFromXml(File permFile, int permissionFlag) {
   
   
        FileReader permReader = new FileReader(permFile);
        ......
        try {
   
   
            XmlPullParser parser = Xml.newPullParser();
            parser.setInput(permReader);
            int type;
            // 查找 START_TAG 或  END_DOCUMENT
            while ((type=parser.next()) != parser.START_TAG
                       && type != parser.END_DOCUMENT) {
   
   
                ;
            }
            // 从 START_TAG 开始解析
            if (type != parser.START_TAG) {
   
   
                throw new XmlPullParserException("No start tag found");
            }
            // 只针对<permissions/>和<config/>标签进行解析
            if (!parser.getName().equals("permissions") && !parser
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值