App自定义加固

项目传送门

一、加固常见的方案和手段

  • 反模拟器
  • 虚拟机保护
  • Dex文件加密

本项目主要使用Dex文件加密的方式对apk进行加固

二、项目主要包含Dex加密和Res加密两大块

2.1 dex文件的加密

  • dex文件的加密逻辑主要在Proxy_Guard_Tools模块的Main_Dex类中
  • dex文件的解密逻辑主要在Proxy_Guard_Core模块的ProxyApplicationDex类中

App自定义加固_第1张图片

2.2 res文件的加密

  • res文件的加密逻辑主要在Proxy_Guard_Tools模块的Main_Res类中
  • res文件的解密逻辑主要在Proxy_Guard_Core模块的ProxyApplicationRes类中

三、需要用到的原理

3.1 APK的打包流程和APK文件结构

App自定义加固_第2张图片
App自定义加固_第3张图片

理解APK的打包流程和文件结构,主要是为了让我们明白在反编译apk文件之后,如何重新修改加密各个文件,如何重新对apk文件进行重发包。

3.2 Framework层Application的启动流程

App自定义加固_第4张图片

理解Application的启动流程,主要是为了让我们明白:

  • 什么时候解密我们先前进行了加密的Dex和资源文件
  • 什么时候加载这些解密之后的Dex和资源文件
  • 什么时候使用我们真正的Application替换ProxyApplication对象

3.3 Dex文件结构解析

App自定义加固_第5张图片

理解Dex的文件结构,主要是为了让我们明白如何伪造一个Dex对象来为App提供一个外壳和入口

3.4 Android中的类加载器

App自定义加固_第6张图片

理解Android中的类加载器,这样我们才能明白我们应用程序的类是被那个类加载器加载的,通过分析该类的源码,以达到动态加载Dex的可行方案。

3.5 Dex文件的加载流程,ClassLoader类加载器的加载原理

Dex文件默认是被PathClassLoader加载到内存中的,默认存储在PathClassLoader对象的DexFileList属性的Element数组中,
在我们使用DexClassLoader加载我们经过加密解密之后的Dex文件,我们需要把通过反射获取这个Element数组,
然后把新加载的Dex字节数组插入到Element数组的前面,这样类加载器在寻找对象的时候就会先获取我们插入的对象

3.6 Resource.arsc文件结构解析

App自定义加固_第7张图片

理解Resource.arsc的结构,主要是为了我们在进行资源加密的时候伪造明文的resouce.arsc文件

3.7 Res资源文件的加载流程,Resource和AssetManager的加载原理

我们使用Resource通过资源id获取资源的时候,其本质是通过AssetManager来获取的,如此要想动态加载资源,我们其实有两个方向:

  • 方案1 我们可以新创建一个AssetManager, 并通过反射调用其【addAssetPath】方法进行资源加载,然后再替换Resouce中原来的AssetManager对象
  • 方案2 系统在创建AssetManager对象的时候,系统默认会加载LoadApk对象中的【mResDir】字段,我们可以通过反射修改这个字段

在我们新创建了Resouce对象之后,还需要找到Framework层中持有旧的Resouce引用的对象,把旧的Resouce对象的引用替换为新的

3.8 反射的使用,通过反射hook系统层类对象

我们需要在代理ProxyApplication的onCreate方法中把Application替换我们真正的Application,并替换掉Framework层中各个系统类对ProxyApplication的引用;在动态加载资源文件之后,我们还需要替换Framework层中各个系统类对Resouce的引用。

3.9 自定义加固和热修复的区别,热修复的原理,bisdiff算法产生差分包

目前资源加密只做了自定义AssetManager,并使用他成功进行了动态资源加载,但是目前的方案会导致很多兼容性问题,比如获取相册文件失败,
导致这些问题的原因主要是因为framework层存在着众多对象持有Resource和AssetManager的引用,我们要想真正实现实现动态加载资源,
就需要通过反射把这些对象里面关于Resouce的引用全部替换掉,具体可以参考腾讯的热修复开源框架Tinker

你可能感兴趣的:(Android技术进阶,android,app加固)