当前位置: 首页 > news >正文

android 14 apexd分析(2)apexd 启动


1. class main进程一起启动,  apexservice是他提供的binderservice,这也第二阶段的最主要的作用
/system/apex/apexd/apexd.rc?r=3c8e8603c640fc41e0406ddcf981381803447cfb#1
1 service apexd /system/bin/apexd
2     interface aidl apexservice     ----------> aidl service
3     class core
4     user root
5     group system
6     oneshot
7     disabled # does not start with the core class
8     reboot_on_failure reboot,apexd-failed
9     # CAP_CHOWN, CAP_DAC_OVERRIDE, CAP_DAC_READ_SEARCH required for apexdata snapshot & restore
10     # CAP_SYS_ADMIN is required to access device-mapper and to use mount syscall
11     capabilities CHOWN DAC_OVERRIDE DAC_READ_SEARCH FOWNER SYS_ADMIN

 
2. main()函数的的内容
 
/system/apex/apexd/apexd_main.cpp?r=3c8e8603c640fc41e0406ddcf981381803447cfb#106
106  int main(int /*argc*/, char** argv) {
.......
155    android::base::Result<android::apex::VoldCheckpointInterface>
156        vold_service_st = android::apex::VoldCheckpointInterface::Create();   ----> 获得Ivold binder service
157    android::apex::VoldCheckpointInterface* vold_service = nullptr;
158    if (!vold_service_st.ok()) {
159      LOG(ERROR) << "Could not retrieve vold service: "
160                 << vold_service_st.error();
161    } else {
162      vold_service = &*vold_service_st;
163    }
164    android::apex::Initialize(vold_service);                           ----> 获得使用Ivold的一些初始化操作
165  
166    if (booting) {         
167      if (auto res = android::apex::MigrateSessionsDirIfNeeded(); !res.ok()) {
168        LOG(ERROR) << "Failed to migrate sessions to /metadata partition : "
169                   << res.error();
170      }
171      android::apex::OnStart();                              ----> onstart 这其中就会 decompress compressedapex 和 ActivateApexPackages()
172    } else {
173      // TODO(b/172911822): Trying to use data apex related ApexFileRepository
174      //  apis without initializing it should throw error. Also, unit tests should
175      //  not pass without initialization.
176      // TODO(b/172911822): Consolidate this with Initialize() when
177      //  ApexFileRepository can act as cache and re-scanning is not expensive
178      android::apex::InitializeDataApex();
179    }
180    // start apexservice before ApexdLifecycle::WaitForBootStatus which waits for
181    // IApexService::markBootComplete().
182    android::apex::binder::CreateAndRegisterService();  ----> 把apexservice 添加到servicemanager中,这是个lazy service
183    android::apex::binder::StartThreadPool();
184  


题外话:
想要使用apexservice命令时,需要加上-w参数,因为他是个lazyservice,-w 是start and wait的意思,调用的是sm->waitforservice这个方法:
 

Example:
cmd -w apexservice getAllPackages
cmd -w apexservice deactivatePackage /system/apex/com.android.virt.apex
cmd -w apexservice activatePackage /system/apex/com.android.virt.apex

compressed apex解压如下:(实际上是对origin_apex又包了一层)
com.google.android.art_compressed$ ls
AndroidManifest.xml   apex_build_info.pb   apex_manifest.pb   apex_pubkey   META-INF   original_apex  original  stamp-cert-sha256

com.google.android.art_compressed/original_apex $ ls
AndroidManifest.xml  apex_build_info.pb  apex_manifest.pb  apex_payload.img  apex_pubkey  assets  META-INF  stamp-cert-sha256

3.ProcessCompressedApex() 把compressed apex搞到/data/apex/decompressed/下面
其中有一个验证比较有意思拉出来单聊,其他我也没兴趣,就没细看
 
2990  Result<void> ValidateDecompressedApex(const ApexFile& capex,
2991                                        const ApexFile& apex) {
2992    // Decompressed APEX must have same public key as CAPEX
2993    if (capex.GetBundledPublicKey() != apex.GetBundledPublicKey()) {=========> 1 压缩包的apexpubkey和original_apex的aepxpubkey是一样的
2994      return Error()
2995             << "Public key of compressed APEX is different than original "
2996             << "APEX for " << apex.GetPath();
2997    }
2998    // Decompressed APEX must have same version as CAPEX                        2 ======>压缩包的version和original_apex的version是一样的
2999    if (capex.GetManifest().version() != apex.GetManifest().version()) {
3000      return Error()
3001             << "Compressed APEX has different version than decompressed APEX "
3002             << apex.GetPath();
3003    }
3004    // Decompressed APEX must have same root digest as what is stored in CAPEX
3005    auto apex_verity = apex.VerifyApexVerity(apex.GetBundledPublicKey());      =======> 3 首先验证decompressed apex自己的avb特性
3006    if (!apex_verity.ok() ||
3007        capex.GetManifest().capexmetadata().originalapexdigest() !=
3008            apex_verity->root_digest) {
3009      return Error() << "Root digest of " << apex.GetPath()  =======> 4 再和compressed apex的manifest的root digest对比一下,见CompressedApexMetadata
3010                     << " does not match with"
3011                     << " expected root digest in " << capex.GetPath();
3012    }
3013    return {};
3014  }
秘钥的结果一模一样,就问你牛不牛逼
python3 avbtool.py info_image --image apex_payload.img
Footer version:           1.0
Image size:               831488 bytes
Original image size:      811008 bytes
VBMeta offset:            823296
VBMeta size:              2240 bytes
--
Minimum libavb version:   1.0
Header Block:             256 bytes
Authentication Block:     576 bytes
Auxiliary Block:          1408 bytes
Public key (sha1):        184474f2094c5dcbfc923af813bebddbf3f712e8
Algorithm:                SHA256_RSA4096
Rollback Index:           0
Flags:                    0
Rollback Index Location:  0
Release String:           'avbtool 1.2.0'
Descriptors:Hashtree descriptor: ------------------------------------- apex 用的是hashtreeVersion of dm-verity:  1Image Size:            811008 bytes                         -----> 198 blocksTree Offset:           811008Tree Size:             12288 bytes                          ------> 3 blocksData Block Size:       4096 bytesHash Block Size:       4096 bytesFEC num roots:         0FEC offset:            0FEC size:              0 bytesHash Algorithm:        sha256Partition Name:        Salt:                  b1c8aab92b646d2d85d0553fa1d3e0bea4c7289e2b740ae6079d40a490631bc0Root Digest:           25aaef0ecde6d8d66458d20f8ff4ef808c2630410443ffec18d82d8647aa332aFlags:                 0Prop: apex.key -> 'com_google_android_ipsec-image'      $ cat ./apex_pubkey |sha1sum
184474f2094c5dcbfc923af813bebddbf3f712e8hashtree的size的计算从layer 1开始,所以是3 blocks, 对于sha256算法,128个data block对应一个hashtree blocklayer 2                   root_hash_blk0/              \
layer 1         hash_blk_0              hash_blk_1/       \               /        \
layer 0    [blk_0] ....[blk_127]  [blk_128]....[blk_197]


4. ActivateApexPackages()  同上一章


5.PMS和apex的联系,用adb install apex
PMS在安装apex的时候,其实是通过apexservice 调用apexservice的
installAndActivatePackage()方法
/frameworks/base/services/core/java/com/android/server/pm/InstallingSession.java?r=3491936bce332909e2b20f7891ced303b9806925#495
495      private void processInstallRequests(boolean success, List<InstallRequest> installRequests) {
496          List<InstallRequest> apexInstallRequests = new ArrayList<>();
497          List<InstallRequest> apkInstallRequests = new ArrayList<>();
498          for (InstallRequest request : installRequests) {
499              if ((request.getInstallFlags() & PackageManager.INSTALL_APEX) != 0) {
500                  apexInstallRequests.add(request);
501              } else {
502                  apkInstallRequests.add(request);
503              }
504          }
505          // Note: supporting multi package install of both APEXes and APKs might requir some
506          // thinking to ensure atomicity of the install.
507          if (!apexInstallRequests.isEmpty() && !apkInstallRequests.isEmpty()) {
508              // This should've been caught at the validation step, but for some reason wasn't.
509              throw new IllegalStateException(
510                      "Attempted to do a multi package install of both APEXes and APKs");
511          }
512          if (!apexInstallRequests.isEmpty()) {
513              if (success) {
514                  // Since installApexPackages requires talking to external service (apexd), we
515                  // schedule to run it async. Once it finishes, it will resume the install.
516                  Thread t = new Thread(() -> installApexPackagesTraced(apexInstallRequests),  ====> 其实是调用的installAndActivatePackage()
517                          "installApexPackages");
518                  t.start();
519              } else {
520                  // Non-staged APEX installation failed somewhere before
521                  // processInstallRequestAsync. In that case just notify the observer about the
522                  // failure.
523                  InstallRequest request = apexInstallRequests.get(0);
524                  mPm.notifyInstallObserver(request);
525              }
526              return;
527          }
528   
529          processApkInstallRequests(success, installRequests);          ====> 普通apk的安装
530      }adb install apex命令和结果:
$ adb install com.android.ipsec@340818020.decompressed.apex
Performing Streamed Install
Success. Reboot device to apply staged session

相关文章:

android 14 apexd分析(2)apexd 启动

1. class main进程一起启动&#xff0c; apexservice是他提供的binderservice&#xff0c;这也第二阶段的最主要的作用 /system/apex/apexd/apexd.rc?r3c8e8603c640fc41e0406ddcf981381803447cfb#1 1 service apexd /system/bin/apexd 2 interface aidl apexservice …...

微信小程序怎么制作?制作一个微信小程序需要多少钱?

随着移动互联网的快速发展&#xff0c;微信小程序已成为连接用户与服务的重要桥梁。它以其便捷性和易用性&#xff0c;为各类企业和个人提供了一个全新的展示和交易平台。那么&#xff0c;如何制作一个微信小程序&#xff1f;又需要投入多少资金呢&#xff1f;本文将为您提供全…...

WPS二次开发专题:如何获取应用签名SHA256值

作者持续关注WPS二次开发专题系列&#xff0c;持续为大家带来更多有价值的WPS开发技术细节&#xff0c;如果能够帮助到您&#xff0c;请帮忙来个一键三连&#xff0c;更多问题请联系我&#xff08;QQ:250325397&#xff09; 在申请WPS SDK授权版时候需要开发者提供应用包名和签…...

Flink SQL系列之:基于Flink SQL查询Topic中序列化的Debezium数据格式字段

Flink SQL系列之:基于Flink SQL查询Topic中序列化的Debezium数据格式字段 一、表结构二、查询Topic中表的数据三、反序列化字段一、表结构 CREATE TABLE IF NOT EXISTS record_rt (id decimal(20,0) COMMENT "主键",follow_entity_type <...

【WPF应用30】WPF中的ListBox控件详解

WPF&#xff08;Windows Presentation Foundation&#xff09;是.NET框架的一个组成部分&#xff0c;用于构建桌面应用程序的用户界面。ListBox是WPF中一个非常常用的控件&#xff0c;用于显示一系列的项&#xff0c;用户可以选择单个或多个项。 1.ListBox的基本概念 ListBox…...

Chatgpt掘金之旅—有爱AI商业实战篇(二)

演示站点&#xff1a; https://ai.uaai.cn 对话模块 官方论坛&#xff1a; www.jingyuai.com 京娱AI 一、前言&#xff1a; 成为一名商业作者是一个蕴含着无限可能的职业选择。在当下数字化的时代&#xff0c;作家们有着众多的平台可以展示和推广自己的作品。无论您是对写书、文…...

AGI时代,LLM可以在AutoML哪些环节进行增强?

当下大模型技术发展如火如荼&#xff0c;颇有改变各行业和各领域的架势。那么对于AutoML来讲&#xff0c;LLM对其有哪些助力&#xff1f;对于这个问题&#xff0c;我们来问一问kimi chat&#xff0c;看看它怎么回答&#xff1f; 大型语言模型&#xff08;LLM&#xff09;可以在…...

算法练习—day1

title: 算法练习—day1 date: 2024-04-03 21:49:55 tags: 算法 categories:LeetCode typora-root-url: 算法练习—day1 网址&#xff1a;https://red568.github.io 704. 二分查找 题目&#xff1a; 题目分析&#xff1a; 左右指针分别为[left,right]&#xff0c;每次都取中…...

关于ansible的模块 ③

转载说明&#xff1a;如果您喜欢这篇文章并打算转载它&#xff0c;请私信作者取得授权。感谢您喜爱本文&#xff0c;请文明转载&#xff0c;谢谢。 接《关于Ansible的模块①》和《关于Ansible的模块②》&#xff0c;继续学习ansible的user模块。 user模块可以增、删、改linux远…...

Spring Boot--文件上传和下载

文件上传和下载 前言文件上传1、以MultipartFile 接口流文件&#xff0c;流的名称需要和前台传过来的名称对应上2、获取到文件名称截取后缀3、为了放置文件名重复使用uuid来随机生成id后缀4、判断转存路径中是否有这个文件夹如果没有就创建5、将文件存储到转存的目录中 文件下载…...

hexo博客7:构建简单的多层安全防御体系

【hexo博客7】构建简单的多层安全防御体系 写在最前面理解全面安全策略的重要性防御常见的网络攻击1. SQL注入攻击2. 文件上传漏洞3. 跨站脚本攻击&#xff08;XSS&#xff09;4. 跨站请求伪造&#xff08;CSRF&#xff09;5. 目录遍历/本地文件包含&#xff08;LFI/RFI&#x…...

《捕鱼_ue4-5输出带技能的透明通道素材到AE步骤》

《捕鱼_ue4-5输出带技能的透明通道素材到AE步骤》 2022-05-17 11:06 先看下带透明的特效素材效果1、首先在项目设置里搜索alpha&#xff0c;在后期处理标签设置最后一项allow through tonemapper2、在插件管理器中&#xff0c;搜索movie render &#xff0c;加载movie render q…...

(免费分享)基于微信小程序自助停取车收费系统

本项目的开发和制作主要采用Java语言编写&#xff0c;SpringBoot作为项目的后端开发框架&#xff0c;vue作为前端的快速开发框架&#xff0c;主要基于ES5的语法&#xff0c;客户端采用微信小程序作为开发。Mysql8.0作为数据库的持久化存储。 获取完整源码&#xff1a; 大家点赞…...

Vue3_2024_7天【回顾上篇watch常见的后两种场景】___续

Vue3中监听多条数据的两种使用 1.watch【使用上一章写法&#xff0c;监听两个属性&#xff0c;然后执行相应操作…】 2.watchEffect【相对于使用watch&#xff0c;watchEffect默认页面初始加载&#xff0c;有点类似加配置&#xff1a;立即执行 immediate】 代码&#xff1a; …...

Gemini即将收费,GPT无需注册?GPT3.5白嫖和升级教程

&#x1f310;Gemini 即将开始收费 开发者“白嫖”的好日子到头了 - Gemini将开始收费&#xff0c;影响使用Google AI for Developers提供的Gemini API的用户。 - Gemini API将引入按量付费定价&#xff0c;需要注意新的服务条款。 - 用户需在5月2日之前停止使用Gemini API和Go…...

【协议篇:Http与Https】

1. Http 1.1 Http的定义 超文本传输协议&#xff08;Hypertext Transfer Protocol&#xff0c;HTTP&#xff09;是用于分布式、协作式和超媒体信息系统的应用层协议。它是互联网上最广泛应用的数据通信协议之一&#xff0c;尤其对于万维网&#xff08;WWW&#xff09;服务而言…...

WPS二次开发系列:WPS SDK初始化

作者持续关注WPS二次开发专题系列&#xff0c;持续为大家带来更多有价值的WPS开发技术细节&#xff0c;如果能够帮助到您&#xff0c;请帮忙来个一键三连&#xff0c;更多问题请联系我&#xff08;QQ:250325397&#xff09; 本文将详细介绍WPS SDK初始化&#xff0c;帮您能够更…...

EXCEL地理数据处理工具(地图任务)

版本号 作者 修订内容 发布日期 1.0 小O 更新至0705版 2022-4-28 1.1 小O 更新至0772版 2024年4月3日 一、概述 小O地图EXCEL插件版提供基于EXCEL表格进行地理数据处理、地图可视化、地图绘图等功能&#xff0c;地理工具是用户使用频率很高的功能模块。地理工具能…...

软件设计原则:迪米特法则

定义 迪米特法则&#xff08;Law of Demeter, LoD&#xff09;&#xff0c;又称最少知识原则&#xff0c;它指导我们在设计软件时&#xff0c;应当尽量减少对象之间的交互&#xff0c;一个对象应该对其他对象有尽可能少的了解。具体来说&#xff0c;一个对象应该只调用属于以下…...

MongoDB聚合运算符:$max

文章目录 语法使用空值和缺失值的处理数组操作数的处理 举例在$group阶段使用在$setWindowFields阶段使用在$project阶段使用 $max聚合运算符用于返回最大值。 $max对于不同的类型的值使用BSON的比较顺序。 $max可以用于下面的这些阶段&#xff1a; $addFields$bucket$bucket…...

神经网络学习笔记10——RNN、ELMo、Transformer、GPT、BERT

系列文章目录 参考博客1 参考博客2 文章目录 系列文章目录前言一、RNN1、简介2、模型结构3、RNN公式分析4、RNN的优缺点及优化1&#xff09;LSTM是RNN的优化结构2&#xff09;GRU是LSTM的简化结构 二、ELMo1、简介2、模型结构1&#xff09;输入2&#xff09;左右双向上下文信…...

Java23种设计模式

本文主要是对Java中一些常用的设计模式进行讲解 后期会进行不断的更新&#xff0c;欢迎浏览 23种设计模式 创建型模式&#xff0c;共五种&#xff1a;工厂方法模式、抽象工厂模式、建造者模式、原型模式、单例模式。结构型模式&#xff0c;共七种&#xff1a;适配器模式、桥接…...

pieces of cake concerning torchtorchvision

1. version match torchvision的版本对应关系 2. utilize tqdm to present process bar lay a pbar from tqdm import tqdm pbar tqdm(unit"batch", filesys.stdout,totallen(self.training_dataloader)) #处理单位为batch pbar2 tqdm(range(20), descIt\s a t…...

如何在Python中处理JSON数据?

如何在Python中处理JSON数据&#xff1f; 在Python中处理JSON数据是一个常见的任务&#xff0c;因为JSON&#xff08;JavaScript Object Notation&#xff09;是一种轻量级的数据交换格式&#xff0c;它易于人阅读和编写&#xff0c;同时也易于机器解析和生成。Python的内置库…...

站群服务器如何提高搜索引擎排名

站群服务器是一种专门为多个相关联的网站提供支持的服务器&#xff0c;旨在通过网站集合的形式提高搜索引擎排名和曝光度。那么站群服务器如何提高搜索引擎排名呢?Rak部落小编为您整理发布。 站群服务器提高搜索引擎排名的原理主要在于以下几个方面&#xff1a; - **提高网站…...

Redis安装-Docker

安装redis的docker容器 1、创建redis挂载目录 mkdir -p /liuchaoxu/redis/{data,conf}2、复制配置文件 在 /liuchaoxu/redis/conf 目录中创建文件 redis.conf&#xff0c;文件从 redis-6.2.7.tar.gz 中解压获取 修改默认配置(从上至下依次)&#xff1a; #bind 127.0.0.1 …...

day16-二叉树part03

104.二叉树的最大深度 &#xff08;优先掌握递归&#xff09; 根节点的高度就是二叉树的最大深度,后序遍历到叶子节点&#xff0c;对遍历高度取最小 class solution {/*** 递归法*/public int maxDepth(TreeNode root) {if (root null) {return 0;}int leftDepth maxDepth(ro…...

上位机图像处理和嵌入式模块部署(qmacvisual亮度检测)

【 声明&#xff1a;版权所有&#xff0c;欢迎转载&#xff0c;请勿用于商业用途。 联系信箱&#xff1a;feixiaoxing 163.com】 前面我们说过&#xff0c;在机器视觉当中&#xff0c;对于光源的处理要非常小心。这里面不仅包括了选择什么样的光源&#xff0c;还取决于怎样使用…...

防止推特Twitter账号被冻结,应该选什么代理类型IP?

在处理多个 Twitter 帐号时&#xff0c;选择合适的代理IP对于避免大规模帐户暂停至关重要。现在&#xff0c;问题出现了&#xff1a;哪种类型的代理是满足您需求的最佳选择&#xff1f;下面文章将为你具体讲解推特账号冻结原因以及重点介绍如何选择代理IP。 一、推特账号被冻结…...

【二叉树】Leetcode 114. 二叉树展开为链表【中等】

二叉树展开为链表 给你二叉树的根结点 root &#xff0c;请你将它展开为一个单链表&#xff1a; 展开后的单链表应该同样使用 TreeNode &#xff0c;其中 right 子指针指向链表中下一个结点&#xff0c;而左子指针始终为 null 。展开后的单链表应该与二叉树 先序遍历 顺序相同…...

无锡网站备案/专门看网站的浏览器

...

旅游网站作用/百度网盘资源链接入口

AQS是JUC包中各种CAS同步器的基类,核心原理就是aqs维护了一个volatile的int类型的变量state,不同的同步器state代表的意义不同. 比如: CountDownLatch的实现中state变量指代的是一个计数. Semaphore的实现state代表的是一个令牌的数量. ReentrantLock的实现state代表的是冲入的…...

wordpress网站源码分享/哈尔滨seo关键词优化

很多时候&#xff0c;我们需要自己去定义dialog&#xff0c;目前我们就遇见了这样一个需求&#xff0c;我的想法是自己定义一个dialog&#xff0c;如果有list的话就使用listview&#xff0c;如果有msg的话就使用msg&#xff0c;并且取消和确定按钮也可自己定义。自定义一个dial…...

wordpress部署https/廊坊首页霸屏优化

集群配置&#xff1a;1个nsqlookupd, 1个nsqadmin&#xff0c;3个nsqd 分区&#xff1a;1个order-topic&#xff0c;分区数为100&#xff0c;副本数为3 扩容时&#xff0c;新增一个nsqd-4。刚开始&#xff0c;nsqd-4没有任何分区副本。 接下来通过nsqadmin页面发现&#xff…...

有赞微商城官网登录/关键词seo公司推荐

异常处理汇总-前端系列 http://www.cnblogs.com/dunitian/p/4523015.html 调节文字间的距离&#xff0c;发现》word-spacing汉字不起作用 研究发现&#xff0c;使用&#xff1a;letter-spacing才可以&#xff08;可以这样理解&#xff1a;word-spacing>单词间隔&#xff0c;…...

网站备案地点/学seo如何入门

最近在看《深入理解计算机系统》&#xff0c;学习编程的一些基本的东西&#xff0c;还是觉得很有意思的。 今天看到了补码编码这一节。 最常见的有符号数的计算机表示方式就是补码(twos-complement)形式。在这个定义中&#xff0c;将字的最高有效为解释为负权(negative weight)…...