Android O PackageInstaller 解析
Android O 8.0
1.src\com\android\packageinstaller\permission\mode\PermissionGroups.java
@Override
public List<PermissionGroup> loadInBackground() {
ArraySet<String> launcherPkgs = Utils.getLauncherPackages(getContext());
PermissionApps.PmCache pmCache = new PermissionApps.PmCache(
getContext().getPackageManager()); List<PermissionGroup> groups = new ArrayList<>();
Set<String> seenPermissions = new ArraySet<>(); PackageManager packageManager = getContext().getPackageManager();
List<PermissionGroupInfo> groupInfos = packageManager.getAllPermissionGroups(0); for (PermissionGroupInfo groupInfo : groupInfos) {
// Mare sure we respond to cancellation.
if (isLoadInBackgroundCanceled()) {
return Collections.emptyList();
} // 得到权限组里的所有权限
// Get the permissions in this group.
final List<PermissionInfo> groupPermissions;
try {
groupPermissions = packageManager.queryPermissionsByGroup(groupInfo.name, 0);
} catch (PackageManager.NameNotFoundException e) {
continue;
} boolean hasRuntimePermissions = false; // 查看某个权限组是否含有运行时权限
// Cache seen permissions and see if group has runtime permissions.
for (PermissionInfo groupPermission : groupPermissions) {
seenPermissions.add(groupPermission.name);
if ((groupPermission.protectionLevel & PermissionInfo.PROTECTION_MASK_BASE)
== PermissionInfo.PROTECTION_DANGEROUS
&& (groupPermission.flags & PermissionInfo.FLAG_INSTALLED) != 0
&& (groupPermission.flags & PermissionInfo.FLAG_REMOVED) == 0) {
hasRuntimePermissions = true;
}
} // 权限组里没有运行时权限,我们不感兴趣,继续循环查询
// No runtime permissions - not interesting for us.
if (!hasRuntimePermissions) {
continue;
} // 如果权限组里含有运行时权限,则继续往下执行 // 获取权限组的label
CharSequence label = loadItemInfoLabel(groupInfo);
Drawable icon = loadItemInfoIcon(groupInfo); // 根据权限组 新建 PermissionApps 对象
PermissionApps permApps = new PermissionApps(getContext(), groupInfo.name, null,
pmCache);
permApps.refreshSync(); // 创建 PermissionGroup 对象
// Create the group and add to the list.
PermissionGroup group = new PermissionGroup(groupInfo.name,
groupInfo.packageName, label, icon, permApps.getTotalCount(launcherPkgs),
permApps.getGrantedCount(launcherPkgs));
groups.add(group);
} // 获取所有的安装包权限
// Make sure we add groups for lone runtime permissions.
List<PackageInfo> installedPackages = getContext().getPackageManager()
.getInstalledPackages(PackageManager.GET_PERMISSIONS); // We will filter out permissions that no package requests.
Set<String> requestedPermissions = new ArraySet<>();
for (PackageInfo installedPackage : installedPackages) {
if (installedPackage.requestedPermissions == null) {
continue;
}
// 将安装包请求的权限加入到 列表中
for (String requestedPermission : installedPackage.requestedPermissions) {
requestedPermissions.add(requestedPermission);
}
} for (PackageInfo installedPackage : installedPackages) {
if (installedPackage.permissions == null) {
continue;
} // 自定义的权限
for (PermissionInfo permissionInfo : installedPackage.permissions) {
// If we have handled this permission, no more work to do.
if (!seenPermissions.add(permissionInfo.name)) {
continue;
} // 关心的是 运行时权限
// We care only about installed runtime permissions.
if ((permissionInfo.protectionLevel & PermissionInfo.PROTECTION_MASK_BASE)
!= PermissionInfo.PROTECTION_DANGEROUS
|| (permissionInfo.flags & PermissionInfo.FLAG_INSTALLED) == 0) {
continue;
} // 没有app使用那个自定义的权限
// If no app uses this permission,
if (!requestedPermissions.contains(permissionInfo.name)) {
continue;
} CharSequence label = loadItemInfoLabel(permissionInfo);
Drawable icon = loadItemInfoIcon(permissionInfo); PermissionApps permApps = new PermissionApps(getContext(), permissionInfo.name,
null, pmCache);
permApps.refreshSync(); // Create the group and add to the list.
PermissionGroup group = new PermissionGroup(permissionInfo.name,
permissionInfo.packageName, label, icon,
permApps.getTotalCount(launcherPkgs),
permApps.getGrantedCount(launcherPkgs));
groups.add(group);
}
} Collections.sort(groups);
return groups;
}
2.PermissionApps.java
/**
* Refresh the state and do not return until it finishes. Should not be called while an {@link
* #refresh async referesh} is in progress.
*/
public void refreshSync() {
mSkipUi = true;
createMap(loadPermissionApps());
}
3.
src\com\android\packageinstaller\permission\utils\EventLogger.java
/**
* For each permission there are four events. The events are in the order of
* #ALL_DANGEROUS_PERMISSIONS. The four events per permission are (in that order): "requested",
* "granted", "denied", and "revoked".
*/
public class EventLogger {
private static final String LOG_TAG = EventLogger.class.getSimpleName(); /** All dangerous permission names in the same order as the events in MetricsEvent */
private static final List<String> ALL_DANGEROUS_PERMISSIONS = Arrays.asList(
Manifest.permission.READ_CALENDAR,
Manifest.permission.WRITE_CALENDAR,
Manifest.permission.CAMERA,
Manifest.permission.READ_CONTACTS,
Manifest.permission.WRITE_CONTACTS,
Manifest.permission.GET_ACCOUNTS,
Manifest.permission.ACCESS_FINE_LOCATION,
Manifest.permission.ACCESS_COARSE_LOCATION,
Manifest.permission.RECORD_AUDIO,
Manifest.permission.READ_PHONE_STATE,
Manifest.permission.CALL_PHONE,
Manifest.permission.READ_CALL_LOG,
Manifest.permission.WRITE_CALL_LOG,
Manifest.permission.ADD_VOICEMAIL,
Manifest.permission.USE_SIP,
Manifest.permission.PROCESS_OUTGOING_CALLS,
Manifest.permission.READ_CELL_BROADCASTS,
Manifest.permission.BODY_SENSORS,
Manifest.permission.SEND_SMS,
Manifest.permission.RECEIVE_SMS,
Manifest.permission.READ_SMS,
Manifest.permission.RECEIVE_WAP_PUSH,
Manifest.permission.RECEIVE_MMS,
Manifest.permission.READ_EXTERNAL_STORAGE,
Manifest.permission.WRITE_EXTERNAL_STORAGE,
Manifest.permission.READ_PHONE_NUMBERS,
Manifest.permission.ANSWER_PHONE_CALLS,
Manifest.permission.ACCEPT_HANDOVER); private static final List<String> ALL_APPOP_PERMISSIONS = Arrays.asList(
Manifest.permission.ACCESS_NOTIFICATIONS,
Manifest.permission.SYSTEM_ALERT_WINDOW,
Manifest.permission.WRITE_SETTINGS,
Manifest.permission.REQUEST_INSTALL_PACKAGES); ...... }
4.
src\com\android\packageinstaller\permission\utils\Utils.java
public static final String OS_PKG = "android"; public static final String[] MODERN_PERMISSION_GROUPS = { // 目前平台支持的权限组
Manifest.permission_group.CALENDAR,
Manifest.permission_group.CAMERA,
Manifest.permission_group.CONTACTS,
Manifest.permission_group.LOCATION,
Manifest.permission_group.SENSORS,
Manifest.permission_group.SMS,
Manifest.permission_group.PHONE,
Manifest.permission_group.MICROPHONE,
Manifest.permission_group.STORAGE
}; private static final Intent LAUNCHER_INTENT = new Intent(Intent.ACTION_MAIN, null)
.addCategory(Intent.CATEGORY_LAUNCHER);
5.
public class DeviceUtils {
public static boolean isTelevision(Context context) { // 电视
int uiMode = context.getResources().getConfiguration().uiMode;
return (uiMode & Configuration.UI_MODE_TYPE_MASK) == Configuration.UI_MODE_TYPE_TELEVISION;
} public static boolean isWear(final Context context) {// 穿戴设备
return context.getPackageManager().hasSystemFeature(PackageManager.FEATURE_WATCH);
} public static boolean isAuto(Context context) {// 汽车、自动设备
return context.getPackageManager().hasSystemFeature(PackageManager.FEATURE_AUTOMOTIVE);
}
}
Android O PackageInstaller 解析的更多相关文章
- Android Service完全解析,关于服务你所需知道的一切(下)
转载请注册出处:http://blog.csdn.net/guolin_blog/article/details/9797169 在上一篇文章中,我们学习了Android Service相关的许多重要 ...
- Android Service完全解析,关于服务你所需知道的一切(上)
转载请注明出处:http://blog.csdn.net/guolin_blog/article/details/11952435 相信大多数朋友对Service这个名词都不会陌生,没错,一个老练的A ...
- [转] Android Volley完全解析(一),初识Volley的基本用法
版权声明:本文出自郭霖的博客,转载必须注明出处. 目录(?)[-] Volley简介 下载Volley StringRequest的用法 JsonRequest的用法 转载请注明出处:http ...
- Android OkHttp完全解析 --zz
参考文章 https://github.com/square/okhttp http://square.github.io/okhttp/ 泡网OkHttp使用教程 Android OkHttp完全解 ...
- Android IntentService完全解析 当Service遇到Handler
一 概述 大家都清楚,在Android的开发中,凡是遇到耗时的操作尽可能的会交给Service去做,比如我们上传多张图,上传的过程用户可能将应用置于后台,然后干别的去了,我们的Activity就很可能 ...
- Android Volley完全解析
1. Volley简介 我们平时在开发Android应用的时候不可避免地都需要用到网络技术,而多数情况下应用程序都会使用HTTP协议来发送和接收网络数据.Android系统中主要提供了两种方式来进行H ...
- Android OkHttp完全解析 是时候来了解OkHttp了
Android OkHttp完全解析 是时候来了解OkHttp了 标签: AndroidOkHttp 2015-08-24 15:36 316254人阅读 评论(306) 收藏 举报 分类: [an ...
- Android Bitmap 全面解析(四)图片处理效果对比 ...
对比对象: UIL Volley 官方教程中的方法(此系列教程一里介绍的,ImageLoader的处理方法和官方的差不多) -------------------------------------- ...
- android源码解析(十七)-->Activity布局加载流程
版权声明:本文为博主原创文章,未经博主允许不得转载. 好吧,终于要开始讲讲Activity的布局加载流程了,大家都知道在Android体系中Activity扮演了一个界面展示的角色,这也是它与andr ...
随机推荐
- php + ajax 避免重复提交
- CALayer: autoresizingMask
UIView 可以设定 autoresizingMask,当它的 superView 尺寸改变时,适应何种变化. 不过 CALayer 却没有这个属性,如和做到让 CALayer 和 UIView 一 ...
- objelement = event.target || event.srcElement;
objelement = event.target || event.srcElement; function updateProductVideo(e){ e = window.event || a ...
- XcenServer和XcenCenterter的安装
XcenServer安装 安装在客户端(作为服务器的电脑) 准备工具:U盘(4G以上).uiso9(iso刻录) 步骤一:下载ISO文件, 下载地址:https://xenserver.org/ove ...
- Linux 系统磁盘满处理方法
1.使用find命令,找出占用空间较大的文件 find / -type f -size +5G —找到大于5G的文件,有work权限的情况下可以使用 find . -type f -size +1 ...
- Android 布局学习之——Layout(布局)详解二(常见布局和布局参数)
[Android布局学习系列] 1.Android 布局学习之——Layout(布局)详解一 2.Android 布局学习之——Layout(布局)详解二(常见布局和布局参数) 3.And ...
- 4. Beego 框架之cookie与session
what is cookie? cookie是存储在客户端的,用于标识客户身份的! what is session session 是存储在服务端,也是用于客户身份标识,用于跟踪用户会话. BeeGo ...
- photoshop制作简单ico图标
新建16 * 16透明画布 字体20px 半径4px
- MvvmLight学习篇—— Mvvm Light Toolkit for wpf/silverlight系列(导航)
系列一:看的迷迷糊糊的 一.Mvvm Light Toolkit for wpf/silverlight系列之准备工作 二.Mvvm Light Toolkit for wpf/silverlight ...
- numpy, pandas, matplotlib等常用库的学习手册
pandas介绍: 待续 参考资料: 中文:https://www.cnblogs.com/skying555/p/5914391.html 英文:http://www.datadependence. ...