Android ART dex2oat

news/2024/11/24 6:01:44/

一、什么是dex2oat

Dex2oat (dalvik excutable file to optimized art file) ,是一个对 dex 文件进行编译优化的程序,在我们的 Android 手机中的位置是 /system/bin/dex2oat,对应的源码路径为 android/art/dex2oat/dex2oat.cc,通过编译优化,可以提升用户日常的使用体验(包含安装速度、启动速度、应用使用过程中的流畅度等),是 Android Art Runtime 中的一个重要的模块, 本文我们一起来了解下 dex2oat 的功能以及常用的场景。

二、为什么要进行dex2oat转换?

众所周知, Android 虚拟机可以识别的是dex文件,应用使用过程中如果每次将dex文件加载进行内存,解释性执行字节码,效率会很低, 严重影响用户体验。通过dex2oat 优化后, 可以在系统运行之前利用合适的时机将dex文件字节码提前转化为虚拟机可以执行运行的机器码,后续直接从效率更高的机器码中运行,则运行阶段更加流畅,优化用户体验。

Dex2oat的主要触发场景

三、几种dex2oat 相关的文件

Dex文件:Dex文件是Android 虚拟机识别的一种可执行文件,我们可以解压一个apk, 获取其中的class.dex文件, 通过dexdump 命令工具对dex 文件进行解析,查看文件内容,更多格式说明查看参考资料 1。

Oat文件:art执行的文件,dex2oat程序编译dex文件的产物。我们可以通过oatdump 查看oat文件具体内容。

Odex文件: Optimizied dexfile, dex文件已经dexopt操作优化后的产物,和dex文件类似,使用了一些优化操作码。

Art文件:Image文件,记录应用启动热点函数相关地址,方便寻址。

Vdex文件:Verified dex,主要包含dex和quicken info信息。Andorid 8.0新增机制产生的文件,其目的主要是为了跳过verified流程,减少dex2oat执行时间。

四、如何使用Dex2oat

4.1 Dex2oat用法

Dex2oat工具的常用参数如下:

4.2 Dex2oat日志解析

从日志中可以看出,在dex2oat发起时具体的编译类型、线程数以及编译原因等等。

常见的编译类型:verify、quicken、space-profile、space、speed-profile、speed、everything, 具体效果从字面上比较好理解, 越后面的类型编译时间越长,占用的空间也越大,运行时打开速度也越快,典型空间换时间思路的体现,其中profile类型的编译方式主要是根据JIT运行过程中热点函数的情况进行编译,JIT机制不进行展开,可以查阅相关资料。

4.3 和dex2oat相关的系统配置

[pm.dexopt.ab-ota]: [speed-profile]

[pm.dexopt.bg-dexopt]: [speed-profile]

[pm.dexopt.boot]: [verify]

[pm.dexopt.first-boot]: [quicken]

[pm.dexopt.inactive]: [verify]

[pm.dexopt.install]: [speed-profile]

[pm.dexopt.shared]: [speed]

4.4 如何手动发起dex2oat操作

通过以上介绍参数含义后,我们可以在adb shell 下通过命令行方式直接发起dex2oat操作,例如强制编译微信:

adb shell cmd package compile -m speed-profile -f com.tencent.mm

清除配置文件数据并移除经过编译的代码:

adb shell cmd package compile --reset com.tencent.mm

下面对具体调用的流程进行分析。

五、Dex2oat 流程分析

本次分析基于Android Q 代码。上述触发场景主要涉及PackageManagerService , 所以从该服务作为入口,分析dex2oat的相关流程。

应用在发起dex2oat时,主要通过PMS中接口调用installd触发的,相关调用函数performDexOpt ,该函数在上述应用安装以及启动的时候都会涉及,所以主要查看下这个函数的调用流程,详见代码:

传入参数DexoptOptions , 可以通过该参数指定编译包名,编译类型以及标志,返回编译是否成功。常见编译标志位:

接下来的调用流程:

以上是Framework中Dex2oat 的调用流程, 感兴趣的同学可以跟踪代码查看具体细节。

系统经过installd 的dexopt编译,通常会利用一些关键的日志查看dex2oat相关的信息, 比如计算dex2oat运行耗时以及最终的编译状态。

Installd dexopt代码小结:

1.检测dexopt classloader context 和相关的flag。

2.解析传入参数,生成dex2oat 命令, 最后通过RunDex2Oat 执行。

常见的dalvik参数控制属性值,

经过上述命令,最终调用到底层libart中相关代码,下面我们查看dex2oat调用的流程图, 了解dex2oat的相关流程。

流程图

Dex2oat逻辑小结:

1. 处理命令行参数;

2. 判断dex2oat的setup是否完成;

3. 根据是否为image类型,分别调用CompileImage或CompileApp的处理,CompileImage和CompileApp的主要功能逻辑类似,主要通过CompilerDriver对dexfile 进行编译。

六、Dex2oat常见修改思路

前面介绍了dex2oat一些优化以及相关的流程,虽然能够提高系统的流畅度,如果在不合适的时机发起,很有可能影响到其他用户操作,需要针对这一类情况进行修改。

以下是常见的修改思路:

1. 根据场景和负载情况调整dex2oat 编译参数,如编译类型,编译线程数量等。

2. 调整boot.img编译资源,预加载资源文件列表。

3. 后台并行编译。在系统空闲或者首次加载dex文件的时候预先触发dex2oat流程,从而加快后续使用dex文件的速度。

举例:后台应用安装导致大量资源被dex2oat占用导致前台进程卡顿

dex2oat 优化后虽然能够增加应用运行的流畅度, 但是如果在短时间内大量发起则会影响用户界面操作, 造成负面的影响。所以发现应用是因为后台自动更新时, 则可以限制dex2oat运行的线程数量,尽可能的减少对前面进程的影响。

七、总结

本文从dex2oat日志输出和使用命令出发,介绍了dex2oat常用的场景以及相关的调用流程,以此为根据简单讨论了常见dex2oat优化方面的思路,希望能起到抛砖引玉的作用,加深读者对dex2oat的原理流程的了解。

参考资料
1.https://source.android.google.cn/devices/tech/dalvik/dex-format
2.https://source.android.com/devices/tech/dalvik/configure#runtime_configuration
3.https://source.android.com/devices/tech/dalvik
4.https://blog.csdn.net/cosmoslhf/article/details/40380559


http://www.ppmy.cn/news/28778.html

相关文章

吉林大学 程序设计基础 2022级 实验复盘 2.23

本人能力有限,发出只为帮助有需要的人。 以下为实验课的复盘,内容会有大量失真,请多多包涵。 此次实验限时一个小时,时间很紧张,很多内容可能并不准确。 1.输出有规律的字母串 输入输出如下; 输入&…

docker-compose安装kafka和php简单测试

docker-compose.yml内容: version: 3.1 services: zookeeper: container_name: zookeeper image: zookeeper:3.6 ports: - 2181:2181 kafka: image: wurstmeister/kafka container_name: kafka depends_on: - zookeeper …

C++ string类(二)及深浅拷贝

一、string类方法使用举例1.迭代器迭代器本质:指针(理解)迭代器:正向迭代器: begin() | end() 反向迭代器: rbegin() | rend()2.find使用//找到s中某个字符 void TestString3() {string s("AAADEFNUIE…

2022年中职网络安全逆向题目整理合集

中职网络安全逆向题目整理合集逆向分析:PE01.exe算法破解:flag0072算法破解:flag0073算法破解:CrackMe.exe远程代码执行渗透测试天津逆向re1 re22023江苏省re12023年江苏省赛re2_easygo.exe逆向分析:PE01.exe FTPServ…

人工智能写的十段代码,九个通过测试了

“抢走你工作的不会是 AI ,而是先掌握 AI 能力的人” 编程测试 1. 我想用golang实现二叉树前序,请你帮我写一下代码。 // 定义二叉树节点 type TreeNode struct {Val intLeft *TreeNodeRight *TreeNode }// 前序遍历 func PreOrderTraversal(root *Tre…

docker常用命令

目录一、启动类命令二、镜像命令1、docker images2、docker search3、docker pull4、docker system df5、docker rmi6、什么是docker虚悬镜像?三、容器命令1、docker run2、docker ps3、退出容器4、容器重启、停止、删除5、容器日志、进程、进入命令行6、容器复制、…

C++STL详解(三)——vector的介绍和使用

文章目录vector的介绍vector的使用vector的定义方式vector的空间增长问题reserve和resizevector的迭代器使用begin 和endrbegin和rendinsert 和erasefind函数元素访问vector迭代器失效问题1:inserse插入扩容时空间销毁造成野指针问题2:erase删除或者inse…

Nmap的几种扫描方式以及相应的命令

Nmap是一款常用的网络扫描工具,它可以扫描目标网络上的主机和服务,帮助安全研究员了解目标网络的拓扑结构和安全情况。以下是Nmap的几种扫描方式以及相应的命令: 1.Ping扫描 Ping扫描可以用来探测网络上响应的主机,可以使用“-sn…