5.1 dex文件解析
1.DexHeader结构体占用0x70字节,源码位置 dalvik\libdex\DexFile.h文件中269/*
270 * Direct-mapped "header_item" struct.
271 */
273 u1 magic[8]; /* dex版本标示 */
274 u4 checksum; /* adler32检验 */
275 u1 signature[kSHA1DigestLen]; /* SHA-1哈希值 */
276 u4 fileSize; /* 整个文件的大小 */
277 u4 headerSize; /* DexHeader结构大小 */
281 u4 mapOff; /* DexMapList文件偏移 */
282 u4 stringIdsSize; /* DexStringId的个数 */
283 u4 stringIdsOff; /* DesStringId的文件偏移 */
284 u4 typeIdsSize; /* DexTypeId的个数 */
285 u4 typeIdsOff; /* DexTypeId的文件偏移 */
286 u4 protoIdsSize; /* DexProtoId的个数 */
287 u4 protoIdsOff; /* DexProtoId的文件偏移 */
288 u4 fieldIdsSize; /* DexFileldId的个数 */
289 u4 fieldIdsOff; /* DexFileldId的文件偏移 */
290 u4 methodIdsSize; /* DexMethodId的个数 */
291 u4 methodIdsOff; /* DexMethodId的文件偏移 */
292 u4 classDefsSize; /* DexClassDefs的个数 */
293 u4 classDefsOff; /* DexClassDefs的文件偏移 */
295 u4 dataOff; /* 数据段的文件偏移 */
296};
补充:
magic:目前固定值dex035 “64 65 78 0A 30 33 35 00”
checksum:dex文件校验和,判断dex文件的完整性和是否被修改
signature:识别最佳化之前的dex文件
fileSize:整个dex文件大小
headerSize:DexHeader结构体本身的大小,目前固定为0x70
endianTag:指定dex运行环境的CPU字节序,预设值ENDIAN_CONSTANT 0x12345678 默认采用Little-Endian字节序列
linkSize和linkOff:指定链接段的大小与文件偏移,大部分情况下默认值0

将实例文件Hello.dex使用010editor打开

查看源码
255struct DexMapList {
256 u4 size; /* DexMapItem的个数 */
257 DexMapItem list[1]; /* DexMapItem结构 */
258};
根据mapOff指向DexMapList的数据结构:
245struct DexMapItem {
246 u2 type; /* kDexType开头类型 */
247 u2 unused; /* 未使用,用于字节对齐 */
248 u4 size; /* 指定类型的个数 */
249 u4 offset; /* 指定类型数据的文件偏移,起始位置 */
250};

跳转到DexMapList结构

在DexHeader部分

2.分析StringIdItem结构体中的字段
源码:
263struct DexStringId {
264 u4 stringDataOff; /* 字符串数据偏移 */
265};
通过stringDataOff找到stringdata字符串的保存位置

Data区存放的字符串并非是ASCII字符串而是由MUTF-8编码表示的Modified UTF-8
特点:
1. MUTF-8使用1~3字节编码长度
2. 大于16位的Unicode编码U+10000~U+10ffff使用3字节编码
3. U+0000采用2字节编码
4. 以null作为字符串结尾
分析DexTypeIds结构体:

地址跳转Ctril+G B0

DexTypeId对应DexHeader中的typeIdsSzie与typeIdsOff
270struct DexTypeId {
271 u4 descriptorIdx; /* 指向DexStringId列表索引 */
272};
3.分析DexTypeProtoIdItem

295struct DexProtoId {
296 u4 shortyIdx; /* 指向DexStringId列表的索引 */
297 u4 returnTypeIdx; /* 指向DexTypeId列表的索引 */
298 u4 parametersOff; /* 指向DexTypeList的偏移 */
299};
补充:
shortyIdx是方法声明字符串
returnTypeIdx是方法返回类型字符串
DexTypeList存放了方法的参数列表
4.DexTypeList的偏移地址


DexTypeList结构声明:325struct DexTypeList {
326 u4 size; /* DexTypeItem的个数 */
327 DexTypeItem list[1]; /* DexTypeItem结构 */
328};
318struct DexTypeItem {
319 u2 typeIdx; /* 指向DexTypeId列表的索引 */
320};
方法声明由返回类型和参数列表组成,并且返回类型位于参数列表的前面
5.分析TypeFieldIdItem

根据偏移

277struct DexFieldId {
278 u2 classIdx; /* 类的声明,指向DexTypeId列表索引 */
279 u2 typeIdx; /* 声明类型,指向DexProtoId列表索引 */
280 u4 nameIdx; /* 方法名,指向DexStringId列表索引 */
281};
DexFieldId结构中的数据全部是索引值,指明字段所在的类,字段的类型,字段名
6.分析DexTypeMethodIdItem

根据偏移

286struct DexMethodId {
287 u2 classIdx; /* 类的声明,指向DexTypeId列表索引 */
288 u2 protoIdx; /* 声明类型,指向DexProtoId列表索引 */
289 u4 nameIdx; /* 方法名,指向DexStringId列表索引 */
290};
7.分析TypeClassDefItem

根据偏移

304struct DexClassDef {
305 u4 classIdx; /*类的类型,指向DexTypeId列表的索引 */
306 u4 accessFlags; /* 访问标示,以ACC_开头的一个枚举值*/
307 u4 superclassIdx; /* 父类类型,指向DexTypeId列表的索引 */
308 u4 interfacesOff; /* 接口,指向DexTypeList的偏移 */
309 u4 sourceFileIdx; /*源文件名,指向DexStringId列表的索引结构*/
310 u4 annotationsOff; /* 注解,指向DexAnnotationDirectoryItem结构*/
311 u4 classDataOff; /* 指向DexClassData结构的偏移 */
312 u4 staticValuesOff; /* 指向DexEncodedArray结构的偏移 */
313};
根据classDataOff偏移

classDataOff指向的DexClassData结构声明 在DexClass.h文件中
51struct DexClassData {
52 DexClassDataHeader header; /* 指定字段与方法的个数 */
53 DexField* staticFields; /* 静态字段,DexField结构 */
54 DexField* instanceFields; /* 实例字段,DexField结构 */
55 DexMethod* directMethods; /* 直接方法,DexMethod结构 */
56 DexMethod* virtualMethods; /* 虚方法,DexMethond结构 */
57};

根据DexClassData分析DexClassDataHeader结构
28struct DexClassDataHeader {
29 u4 staticFieldsSize; /* 静态字段个数 */
30 u4 instanceFieldsSize; /* 实例字段个数 */
31 u4 directMethodsSize; /* 直接方法 */
32 u4 virtualMethodsSize; /* 虚方法 */
33};
DexField结构描述了字段的类型与访问标志
36struct DexField {
37 u4 fieldIdx; /* 指向DexFieldId的索引 */
38 u4 accessFlags; /* 访问标志 */
39};
DexMethod描述方法的原型,名称,访问标志和代码数据块
42struct DexMethod {
43 u4 methodIdx; /* 指向DexMethodId的索引 */
44 u4 accessFlags; /* 访问标志 */
45 u4 codeOff; /* 指向DexCode结构的偏移 */
46};
根据codeOff偏移定位

DexCode在DexFile.h文件中
337struct DexCode {
338 u2 registersSize; /* 使用寄存器的数量 */
339 u2 insSize; /* 参数的个数 */
340 u2 outsSize; /* 调用其它方法时使用的寄存器个数 */
341 u2 triesSize; /* Try/Catch个数 */
342 u4 debugInfoOff; /* 指向调试信息的偏移 */
343 u4 insnsSize; /* 指令集个数,以2字节为单位 */
344 u2 insns[1]; /* 指令集 */
345 /* 2字节空间用于结构对齐 */
346 /* try_item[triesSize] DexTry结构 */
347 /* Try/Catch中handler的个数 */
348 /* catch_handler_item[handlersSize],DexCatchHandler结构 */
349};
至此Dex文件分析完毕
附录StringId表,DexTypeld表,DexProtoId表
DexStringId结构体

DexTypeId列表

DexProtoId列表

5.1 dex文件解析的更多相关文章
- DEX文件解析---1、dex文件头解析
DEX文件解析---1.dex文件头解析 一.dex文件 dex文件是Android平台上可执行文件的一种文件类型.它的文件格式可以下面这张图概括: dex文件头一般固定为0x70个字 ...
- class文件与dex文件解析
关于Android的热修复与插件化技术在如今基本上已经成为了“时髦技术”的标配了,或者说用来进行“炫技”的一种方式,毕境如今Android已经发展得非常之成熟了,基本上APP用的到东东都差不多,除了业 ...
- DEX文件解析--7、类及其类数据解析(完结篇)
一.前言 前置技能链接: DEX文件解析---1.dex文件头解析 DEX文件解析---2.Dex文件checksum(校验和)解析 DEX文件解析--3.d ...
- DEX文件解析--3、dex文件字符串解析
一.前言 前两篇文章链接: 1.DEX文件头解析 2.DEX文件校验和解析 PS:前几天检查文件夹的时候发现DEX文件解析还只写了开头,正好找点事情来做,就去接着解析DEX ...
- DEX文件解析--4、dex类的类型解析
一.前言 前几篇系列文章链接: DEX文件解析---1.dex文件头解析 DEX文件解析---2.Dex文件checksum(校验和)解析 DEX文件解析--3.dex文件 ...
- DEX文件解析--5、dex方法原型解析
一.前言 前几篇文章链接: DEX文件解析---1.dex文件头解析 DEX文件解析---2.Dex文件checksum(校验和)解析 DEX文件解析--3.dex文件字 ...
- DEX文件解析--6、dex文件字段和方法定义解析
一.前言 前几篇文章链接: DEX文件解析---1.dex文件头解析 DEX文件解析---2.Dex文件checksum(校验和)解析 DEX文件解析--3. ...
- DEX文件解析---2、Dex文件checksum(校验和)解析
一.checksum介绍 checksum(校验和)是DEX位于文件头部的一个信息,用来判断DEX文件是否损坏或者被篡改,它位于头部的0x08偏移地址处,占用4个字节,采用小端序存储. ...
- Android Dex文件格式解析
Dex文件是Android虚拟机下的可执行文件,包含了应用程序所用到所有操作指令和运行时数据.在程序编译过程中,java源文件先被编译成class文件,然后通过dx工具将多个class文件整合为一个d ...
随机推荐
- vue-cli3和ts建立vue项目
第一步,如果你之前没有装vuecli,可以直接执行下面命令 npm install --global @vue/cli 注:这里我install 的时候不成,于是我用的是 cnpm install - ...
- (HK1-0)激活与配置摄像机
HK使用手册 网络连接 激活与配置摄像机 网络摄像机可通过 SADP 软件.客户端软件和浏览器三种方式激活, 具体激活操作方式可参见<网络摄像机操作手册>. 1. 安装随机光盘或从官网下载 ...
- Pandas | 12 选项和自定义
Pandas提供API来自定义其行为的某些方面,大多使用来显示. API由五个相关函数组成.它们分别是: get_option() set_option() reset_option() descri ...
- proc介绍,free命令查看内存
proc介绍 https://www.cnblogs.com/dongzhuangdian/p/11366910.html https://blog.csdn.net/majianting/artic ...
- yugabyte 安装pg_hashids 扩展
主要目的比较简单,就是测试下yugabyte 对于pg 扩展的支持,今天在测试plv8的时候发现有问题(以及提交issue了,很期待官方的解决) 所以测试下一个其他的扩展,看看是否像官方介绍的那样 环 ...
- React - 组件:函数组件
目录: . 组件名字首字母一定是大写的 . 返回一个jsx . jsx依赖React,所以组件内部需要引入React . 组件传参 a. 传递. <Component list={ arrDat ...
- springboot修改页面不用重启的设置(idea)
1) “File” -> “Settings” -> “Build,Execution,Deplyment” -> “Compiler”,选中打勾 “Build project ...
- JS 常见问题
JavaScript 是一种有趣的语言,我们都喜欢它,因为它的性质.浏览器是JavaScript的主要运行的地方,两者在我们的服务中协同工作.JS有一些概念,人们往往会对它掉以轻心,有时可能会忽略不计 ...
- BurpSuite pro v2.0 使用入门教程
BurpSuite简介 BurpSuite是进行Web应用安全测试集成平台.它将各种安全工具无缝地融合在一起,以支持整个测试过程中,从最初的映射和应用程序的攻击面分析,到发现和利用安全漏洞.Burps ...
- CnblogAndroid使用反馈 & PureMan6留言板
我们的话: 您可以在这篇博客下评论您使用CnblogAndroid时遇到的问题和您的意见与建议: 或是留言给PureMan6团队,我们会定期查看并进行回复. 同时,关于app的问题,您也可以在Cnbl ...