标签(空格分隔): 安卓逆向java
文件夹 | 做用 |
---|---|
asset文件夹 | 资源目录1:asset和res都是资源目录但有所区别,见下面说明 |
lib文件夹 | so库存放位置,通常由NDK编译获得,常见于使用游戏引擎或JNI native调用的工程中 |
META-INF文件夹 | 存放工程一些属性文件,例如Manifest.MF |
res文件夹 | 资源目录2:asset和res都是资源目录但有所区别,见下面说明 |
AndroidManifest.xml | Android工程的基础配置属性文件 |
classes.dex | Java代码编译获得的DalvikVM能直接执行的文件,下面有介绍 |
resources.arsc | 对res目录下的资源的一个索引文件,保存了原工程中strings.xml等文件内容 |
其余文件夹 | etc. |
res目录下的资源文件在编译时会自动生成索引文件(R.java),在Java代码中用R.xxx.yyy来引用;
而asset目录下的资源文件不须要生成索引,在Java代码中须要用AssetManager来访问;
通常来讲,除了音频和视频资源(须要放在raw或asset下),使用Java开发的Android工程使用到的资源文件都会放在res下;使用C++游戏引擎(或使用Lua Unity3D等)的资源文件均须要放在asset下。android
其中在Davlik字节码中,寄存器都是32位的,可以支持任何类型,64位类型(Long/Double)用2个寄存器表示;Dalvik字节码有两种类型:原始类型;引用类型(包括对象和数组)数组
B---byte C---char D---double F---float I---int J---long S---short V---void Z---boolean [XXX---array Lxxx/yyy---object
数组的表示方式是:在基本类型前加上前中括号“[”,例如int数组和float数组分别表示为:[I、[F;对象的表示则以L做为开头,格式是LpackageName/objectName;
(注意必须有个分号跟在最后),例如String对象在smali中为:Ljava/lang/String;
,其中java/lang对应java.lang包,String就是定义在该包中的一个对象。或许有人问,既然类是用LpackageName/objectName;
来表示,那类里面的内部类又如何在smali中引用呢?答案是:LpackageName/objectName$subObjectName;
。也就是在内部类前加“$”符号,关于“$”符号更多的规则将在后面谈到。app
方法的定义通常为:Func-Name (Para-Type1Para-Type2Para-Type3...)Return-Type
注意参数与参数之间没有任何分隔符,一样举几个例子就容易明白了函数
hello ()V
。void hello()
hello (III)Z
。boolean hello(int, int, int)
hello (Z[I[ILjava/lang/String;J)Ljava/lang/String;
看出来这是String hello (boolean, int[], int[], String, long)
了吗?ui
.field private isFlag:z 定义变量
.method 方法
.parameter 方法参数
.prologue 方法开始
.line 123 此方法位于第123行
invoke-super 调用父函数
const/high16 v0, 0x7fo3 把0x7fo3赋值给v0
invoke-direct 调用函数
return-void 函数返回void
.end method 函数结束
new-instance 建立实例
iput-object 对象赋值
iget-object 调用对象
invoke-static 调用静态函数this
"if-eq vA, vB, :cond_**" 若是vA等于vB则跳转到:cond_**
"if-ne vA, vB, :cond_**" 若是vA不等于vB则跳转到:cond_**
"if-lt vA, vB, :cond_**" 若是vA小于vB则跳转到:cond_**
"if-ge vA, vB, :cond_**" 若是vA大于等于vB则跳转到:cond_**
"if-gt vA, vB, :cond_**" 若是vA大于vB则跳转到:cond_**
"if-le vA, vB, :cond_**" 若是vA小于等于vB则跳转到:cond_**
"if-eqz vA, :cond_**" 若是vA等于0则跳转到:cond_**
"if-nez vA, :cond_**" 若是vA不等于0则跳转到:cond_**
"if-ltz vA, :cond_**" 若是vA小于0则跳转到:cond_**
"if-gez vA, :cond_**" 若是vA大于等于0则跳转到:cond_**
"if-gtz vA, :cond_**" 若是vA大于0则跳转到:cond_**
"if-lez vA, :cond_**" 若是vA小于等于0则跳转到:cond_**翻译
.class public Lcom/aaaaa; .super Lcom/bbbbb; .source "ccccc.java"
这是一个由ccccc.java编译获得的smali文件(第3行)
它是com.aaaaa这个package下的一个类(第1行)
继承自com.bbbbb这个类(第2行)code
在smali里的全部操做都必须通过寄存器来进行:本地寄存器用v开头数字结尾的符号来表示,如v0、v一、v2。
参数寄存器则使用p开头数字结尾的符号来表示,如p0、p一、p二、...
特别注意的是,p0不必定是函数中的第一个参数,在非static函数中,p0代指“this”,p1表示函数的第一个参数,p2表明函数中的第二个参数…而在static函数中p0才对应第一个参数(由于Java的static方法中没有this方法。
寄存器简单实例分析:
const/4 v0, 0x1 iput-boolean v0, p0, Lcom/aaa;->IsRegistered:Z
咱们来分析一下上面的两句smali代码,首先它使用了v0本地寄存器,并把值0x1存到v0中,而后第二句用iput-boolean这个指令把v0中的值存放到com.aaa.IsRegistered这个成员变量中。
即至关于:this.IsRegistered = true;
(上面说过,在非static函数中p0表明的是“this”,在这里就是com.aaa实例)。视频
成员变量格式是:
.field public/private [static] [final] varName:<类型>
对于不一样的成员变量也有不一样的指令。
通常来讲,获取的指令有:
iget、sget、iget-boolean、sget-boolean、iget-object、sget-object
等。
操做的指令有:
iput、sput、iput-boolean、sput-boolean、iput-object、sput-object
等。
没有“-object”后缀的表示操做的成员变量对象是基本数据类型,带“-object”表示操做的成员变量是对象类型,特别地,boolean类型则使用带“-boolean”的指令操做。
sget-object v0, Lcom/aaa;->ID:Ljava/lang/String;
sget-object
就是用来获取变量值并保存到紧接着的参数的寄存器中,本例中,它获取ID这个String类型的成员变量并放到v0这个寄存器中。iget-object v0, p0, Lcom/aaa;->view:Lcom/aaa/view;
iget-object
指令比sget-object
多了一个参数,就是该变量所在类的实例,在这里就是p0即“this”.aget
和aget-object
,指令使用和上述一致put
指令的使用和get
指令是统一的以下:const/4 v3, 0x0 sput-object v3, Lcom/aaa;->timer:Lcom/aaa/timer;
至关于:this.timer = null;
注意,这里由于是赋值object 因此是null,如果boolean的话,你们想应该至关于什么呢?
.local v0, args:Landroid/os/Message; const/4 v1, 0x12 iput v1, v0, Landroid/os/Message;->what:I
至关于:args.what = 18;
(args
是Message
的实例)
smali中的函数和成员变量同样也分为两种类型,分别为direct
和virtual
之分。
那么direct method
和virtual method
有什么区别呢?
简单来讲,direct method
就是private
函数,其他的public
和protected
函数都属于virtual method
。因此在调用函数时,有invoke-direct
,invoke-virtual
,另外还有invoke-static
、invoke-super
以及invoke-interface
等几种不一样的指令。
固然其实还有invoke-XXX/range
指令的,这是参数多于4个的时候调用的指令,比较少见,了解下便可。
invoke-static
:用于调用static函数的例如:
invoke-static {}, Lcom/aaa;->CheckSignature()Z
这里注意到invoke-static
后面有一对大括号“{}”,实际上是调用该方法的实例+参数列表,因为这个方法既不需参数也是static
的,因此{}内为空,再看一个:
const-string v0, "NDKLIB" invoke-static {v0}, Ljava/lang/System;->loadLibrary(Ljava/lang/String;)V
这个是调用static void System.loadLibrary(String)
来加载NDK编译的so库用的方法,一样也是这里v0就是参数"NDKLIB"了。
invoke-direct {p0}, Landroid/app/TabActivity;-><init>()V
这里init()
就是定义在TabActivity
中的一个private
函数
sget-object v0, Lcom/dddd;->bbb:Lcom/ccc; invoke-virtual {v0, v1}, Lcom/ccc;->Messages(Ljava/lang/Object;)V
这里相信你们都已经很清楚了:
v0是bbb:Lcom/ccc
v1是传递给Messages
方法的Ljava/lang/Object
参数。
invoke-direct/range {v0 .. v5}, Lcmb/pb/ui/PBContainerActivity;->h(ILjava/lang/CharSequence;Ljava/lang/String;Landroid/content/Intent;I)Z
须要传递v0到v5一共6个参数,这时候大括号内的参数采用省略形式,且须要连续。
在Java代码中调用函数和返回函数结果能够用一条语句完成,而在Smali里则须要分开来完成,在使用上述指令后,若是调用的函数返回非void,那么还须要用到move-result
(返回基本数据类型)和move-result-object
(返回对象)指令:
const-string v0, "Eric" invoke-static {v0}, Lcmb/pbi;->t(Ljava/lang/String;)Ljava/lang/String; move-result-object v2
v2保存的就是调用t方法返回的String
字符串。
.method private ifRegistered()Z .locals 2 //在这个函数中本地寄存器的个数 .prologue const/4 v0, 0x1 // v0赋值为1 .local v0, tempFlag:Z if-eqz v0, :cond_0 // 判断v0是否等于0,等于0则跳到cond_0执行 const/4 v1, 0x1 // 符合条件分支 :goto_0 //标签 return v1 //返回v1的值 :cond_0 //标签 const/4 v1, 0x0 // cond_0分支 goto :goto_0 //跳到goto_0执行 即返回v1的值 这里能够改为return v1 也是同样的 .end method
const/4 v0, 0x0 //vo =0; .local v0, i:I :goto_0 if-lt v0, v3, :cond_0 // v0小于v3 则跳到cond_0并执行分支 :cond_0 return-void :cond_0 // 标签 iget-object v1, p0, Lcom/aaa/MainActivity;->listStrings:Ljava/util/List; // 引用对象 const-string v2, "Eric" invoke-interface {v1, v2}, Ljava/util/List;->add(Ljava/lang/Object;)Z // List是接口, 执行接口方法add add-int/lit8 v0, v0, 0x1 // 将第二个v0寄存器中的值,加上0x1的值放入第一个寄存器中, 实现自增加 goto :goto_0 // 回去:goto_0标签
.locals 4 const/4 v2, 0x1 const/16 v1, 0x10 .local v1, "length":I if-nez v1, :cond_1 :cond_0 :goto_0 return v2 :cond_1 const/4 v0, 0x0 .local v0, "i":I :goto_1 if-lt v0, v1, :cond_2 const/16 v3, 0x28 if-le v1, v3, :cond_0 const/4 v2, 0x0 goto :goto_0 :cond_2 xor-int/lit8 v1, v1, 0x3b add-int/lit8 v0, v0, 0x1 goto :goto_1