【转自前辈一部分[https://blog.csdn.net/qq_24349189/article/details/52300419] 再加上自己整理】
2)原始类型:
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 就是定义在该包中的一个对象。
内部类又如何在 smali 中:LpackageName/objectName s u b O b j e c t N a m e ; 也 就 是 在 内 部 类 前 加 “ subObjectName;也就是在内部类前加“ subObjectName;也就是在内部类前加“”符号。
3)方法的定义
Func-Name (Para-Type1Para-Type2Para-Type3…)Return-Type注意参数与参数之间没有任何分隔符,举例如下:
A ()V 这就是void A()。
B (II)Z 这个则是boolean B(int, int)。
C (Z[I[ILjava/lang/String;J)Ljava/lang/String; 这是String C (boolean, int[], int[], String, long) 。
4)Smali基本语法
.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 调用静态函数
5)条件跳转分支
"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_
二.Smali 文件
Smali中的包信息
.class public Lcom/aaaaa; (它是com.aaaaa这个package下的一个类)
.super Lcom/bbbbb; (继承自com.bbbbb这个类)
.source “ccccc.java” (一个由ccccc.java编译得到的smali文件)
Smali中的声明
annotations
.annotation system Ldalvik/annotation/MemberClasses;
value = {
Lcom/aaa q q q ; , L c o m / a a a qqq;, Lcom/aaa qqq;,Lcom/aaawww;
}.
end annotation
//这个声明是内部类的声明:aaa这个类它有两个成员内部类——qqq和www,内部类将在后面小节中会有提及。
3.关于寄存器
寄存器是什么意思呢?在 smali 里的所有操作都必须经过寄存器来进行:本地寄存器用 v 开头,数字结尾的符号来表示,如v0、v1、v2、…
参数寄存器则使用 p 开头,数字结尾的符号来表示,如p0、p1、p2、…
特别注意的是,p0 不一定是函数中的第一个参数,在非 static 函数中,p0 代指“this”,p1 表示函数的第一个参数,p2 代表函数中的第二个参数。
而在 static 函数中 p0 才对应第一个参数(因为 Java 的 static 方法中没有 this 方法。
4. 寄存器简单实例分析
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”的指令操作。
(put指令的使用和get指令是统一的)
const/4 v3, 0x0
sput-object v3, Lcom/aaa;->timer:Lcom/aaa/timer;
相当于:this.timer= null;
注意,这里因为是赋值object 所以是null
4) 简析四
.local v0, args:Landroid/os/Message;
const/4 v1, 0x12
iput v1, v0, Landroid/os/Message;->what:I
相当于:args.what = 18;(args 是 Message 的实例)
三.Smali函数分析
2)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"了。
3)invoke-super:
调用父类方法用的指令,一般用于调用onCreate、onDestroy等方法。
4)invoke-direct:
调用private函数:invoke-direct {p0}, Landroid/app/TabActivity;->()V
这里init()就是定义在TabActivity中的一个private函数
5)invoke-virtual:
用于调用 protected 或 public 函数,同样注意修改smali时不要错用 invoke-direct或 invoke-static:
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参数。
6)invoke-xxxxx/range:
当方法的参数多于5个时(含5个),不能直接使用以上的指令,而是在后面加上“/range”,range表示范围,使用方法也有所不同:
invoke-direct/range {v0 … v5}, Lcmb/pb/ui/PBContainerActivity;->h(ILjava/lang/CharSequence;Ljava/lang/String;Landroid/content/需要传递v0到v5一共6个参数,这时候大括号内的参数采用省略形式,且需要连续。
2. Smali中函数返回结果操作
在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字符串。
3. Smali中函数实体分析–if函数分析
.method private ifRegistered()Z
.locals 2 //在这个函数中本地寄存器的个数
.prologue
const/4 v0, 0x1 // v0赋值为1
.local v0, tempFlag:Z
if-eqz v0, :cond0 // 判断v0是否等于0,等于0则跳到cond0执行
const/4 v1, 0x1 // 符合条件分支
:goto_0 //标签
return v1 //返回v1的值
:cond_0 //标签
const/4 v1, 0x0 // cond_0分支
goto :goto0 //跳到goto0执行 即返回v1的值 这里可以改成return v1 也是一样的
.end method