LEGU脱壳

news/2024/7/19 11:31:27 标签: 爬虫, 移动开发, 嵌入式

一、原理

腾讯乐固(http://legu.qcloud.com/)提供APK加壳技术,分析了一下,做了个简单的脱壳程序。

以某新华字典APP为例,APK下载地址:https://apkpure.com/%E6%96%B0%E5%8D%8E%E5%AD%97%E5%85%B8/com.qiushui.android.app.chdir

加壳的方法是,将原始的 class.dex 的文件头(DexHeader,长度为0x70个字节)使用某种方式进行加密,后面的数据保持不变,使用腾讯的代码(TxAppEntry)加固;运行时从文件的特定偏移地址开始,解密出原始的DEX,从内存运行。这里分析的壳的版本是0.0.3.

 

二、分析过程

打开APK,在lib/armeabi 下面发现这几个文件:

 

使用 linux下的 file 工具查看,可知:

 

libshella.so 是ARM平台用的,libshellx.so是X86平台用的,那么这两个so内的函数应该完全一样。

选择在 Android 平台调试,用 IDA 打开so文件,附加到进程(方法参考:http://www.cnblogs.com/shaoge/p/5425214.html ),经过若干次的尝试,终于找到解密的函数是:sub_5DCF4EA4。

目前可以这样找,搜索字符串 start mem dex,通过 xref graph 找到使用这个字符串的地方,可以找到是在函数sub_5DCEF760中,由于此时的 dex 是已经解密好的,如果不关心加壳的过程,可以在这里直接在该函数下断点,执行到这里时,用IDA的script来dump 内存,起始地址保存在寄存器 R0, 长度保存在寄存器R1.

 

如果关心解密过程,就向上查找,在函数  sub_5DE108E0 中,找到调用 sub_5DCF4EA4 之前的代码(有多个地方调用,可以在运行时查看实际执行的代码):

 

这里简单解释一下,过程大概如下:

  1. 获取进程的PID,打开 /proc/{PID}/mmap 文件,查找 XX.apk@classes.dex,找到内存的起始地址;
  2. 在这个地址偏移0x3028个字节,复制0x70个字节(DexHeader的大小)进行解密,解密的算法参见后面的地址;
  3. 解密后就是原始的DEX了。

 

在执行中,通过查看内存,可以知道 XX.apk@classes.dex 是 odex 的格式,因此偏移了 0x3028 个字节,如果是普通的 dex 格式,则只需偏移 0x3000 个字节。

 

DEX 文件的大小就是 DexHeader 里面包含了(DEX文件格式 http://www.retrodev.com/android/dexformat.html),只需向后直接复制相应的长度即可。脱壳后的文件头如下所示:

 

 

三、源代码

    https://github.com/zbqf109/txshellunpack

四、参考文档

  1. 腾讯应用加固的脱壳分析和修复http://www.52pojie.cn/thread-330022-1-1.html
  2. IDA远程调试 在内存中dump Dex文件 http://www.cnblogs.com/shaoge/p/5425229.html
  3. android某加固脱壳
    (一) http://www.xiaobaiyey.com/618.html
    (二) http://www.xiaobaiyey.com/634.html
  4. Dex File Format http://www.retrodev.com/android/dexformat.html

五、用到的工具

    apktool http://ibotpeaches.github.io/Apktool/

    dex2jar https://github.com/pxb1988/dex2jar

 

声明:仅供个人学习,不得用于其他用途。

转载于:https://www.cnblogs.com/zhangbaoqiang/p/5446457.html


http://www.niftyadmin.cn/n/916360.html

相关文章

第二章:硬件访问服务(2)-系统编写

通过上一节的介绍,对于android硬件访问服务器有了大致的了解,根据上一节的框图,我们在来梳理一下要点。一个饭店要正常的营业,顾客,菜单菜式,厨师缺一不可。我们的要点亮LED也是同样,APP&#x…

CPSR 和 SPSR

CPSR:程序状态寄存器(current program status register) (当前程序状态寄存器),在任何处理器模式下被访问。它包含了条件标志位、中断禁止位、当前处理器模式标志以及其他的一些控制和状态位。 CPSR在用户级编程时用于存储条件码。 SPSR:程序状态保存寄存器&#…

第一阶段 PHP基础.运算符2

一位初学php的随堂笔记,记录自己的成长! 一.运算符(Operator) 1.运算符:程序执行的运算符,要求一个以上的 操作数参加运算 表达式由运算符和操作数组成 2.运算符分类 (1)按功能分 (2)按操作数的个数分 3.按功能分 (1)算术运算符 (…

第二章:硬件访问服务(3)-APP修改编译

通过上一小节,我们硬件访问服务已经基本完成(还有少些部分需要修改,下小节为大家讲解),该小节我们实现APP部分。 APP修改 在第一章节中,我们在工程app/src/main/java/com/example/administrator/下创建了…

Linux网络子系统内核分析

转自 http://www.linuxidc.com/Linux/2016-01/127966.htm Linux网络体系结构由以下五个部分组成 1)系统调用接口 2)协议无关接口 3)网络协议 4)设备无关接口 5 设备驱动程序。下面分别简述五个部分: 1)系统…

第二章:硬件访问服务(4)-HAL编写

通过前面的学习,我们知道怎么通过andriod硬件访问服务控制硬件,再次贴出框图如下 其中红色圈出部分我们已经完成,我们知道在andriod系统中,java程序是无法直接访问硬件的,他访问硬件需要service_manager.c发出请求&am…

C语言函数之可变参数原理:va_start、va_arg及va_end

说到C语言函数可变参数,我们最先想到的可能就是printf、scanf、printk了。在Linux-2.6.24.7内核源码里,printk函数原型如下: asmlinkage int printk(constchar*fmt,...) asmlinkage表示通过堆栈传递参数。gcc编译器在汇编过程中调用c语言函数时传递参数…

软件工程与方法学——面向对象程序设计例子

题目: /*⑴ 有四个按钮输入,分别称为B1,B2,B3和B4; ⑵ 有两个灯泡作为输出,分别称为L1和L2; ⑶ B1是打开电源的按钮; ⑷ B4是关闭电源的按钮; ⑸ B2和B3 是操作按钮&…