Smali语法
孔子:三人行,必有我师焉
语法记录
语法 | 含义 |
---|---|
nop | 没有操作 |
move vx,vy | 移动内容 vy 到 vx.寄存器范围必须在256以内 |
move/from16 vx,vy | 移动 vy 到 vx. vy寄存器范围只能在64k以内 vx 寄存器范围必须在256内. |
move-wide/from16 vx,vy | 移动long/double值从vy到vx. vy寄存器范围只能在64k以内 vx是一个 256 寄存器 |
move-object vx,vy | 自动对象引用从 vy 到 vx. |
move-object/from16 vx,vy | 移动对象引用从 vy 到 vx, vy 可以添加 64k 寄存器和 vx 可以添加 256 寄存器. |
move-result vx | 把前面运行的结果赋值给 vx. |
move-result-wide vx | 移动 long/double 方法调用结果赋值到vx,vx+1. |
move-result-object vx | 移动前面的方法调用结果到 vx |
move-exception vx | 移动异常对象 并抛出异常到 vx. |
return-void | 没有返回值返回 |
return vx | 返回值 vx |
return-wide vx | 返回double/long 结果 vx,vx+1. |
return-object vx | 返回vx对象引用值 |
const/4 vx,lit4 | 将4字节的参数赋值到 vx |
const/16 vx,lit16 | 将16进制数赋值到 vx |
const vx, lit32 | 将整数常量赋值到 vx |
const/high16 v0, lit16 | 将16位不变的最高位寄存器用于初始化浮点数 |
const-wide/16 vx, lit16 | 将整数常量vx和vx+1寄存器扩大整数为常量.. |
const-wide/32 vx, lit32 | 把 32 bit 常数 vx 和 vx+1 寄存器 |
const-wide vx, lit64 | 把64 bit 常数 vx 和 vx+1 放入寄存器. |
const-wide/high16 vx,lit16 | 将16位常数 vx 和 vx+1 的最高位寄存器终于初始化值的两倍 |
const-string vx,string_id | 设定一个字符串常量 vx 字符串id |
const-string/jumbo vx,string_id | 通过字符串索引(较大)构造一个字符串并赋给寄存器vx |
const-class vx,type_id 从 type_id | 移动类对象和类标识(e.g. Object.class) 到 vx. |
monitor-enter vx | 监控对象引用 vx. |
monitor-exit | 监视器对象引用 vx. |
check-cast vx, type_id | 检查是否对象引用 |
instance-of vx,vy,type_id | 检查是否 vy 是否是一个类的实例 type_id. |
array-length vx,vy | 计算vy引用的数组元素并赋值到 vx. |
new-instance vx,type | 实例化对象引用并将 vx 创建为新实例 |
new-array vx,vy,type_id | 生成新的数组 type_id 类型和 vy 元素大小并将数组赋值给 vx. |
throw vx | 抛出一个异常对象,异常对象vx. |
goto target | 无条件跳转. |
goto/16 target | 无条件跳转 |
packed-switch vx,table | 实例化switch语句,查找表与常量和偏移量为常数,如表里无匹配则执行下一条 |
sparse-switch vx,table | switch操作指令使用查找表与常量和偏移量为常数,如表里无匹配则执行下一条 |
cmpl-float | 在vy 比较浮点值后赋值到 vx |
cmpg-float vx, vy, vz | 比较浮点值 vy 和 vz 和设置整数赋值到 vx. |
cmpl-double vx,vy,vz | 比较 double 值 vy 和 vz 并设置整数 赋值到 vx. |
cmpg-double vx, vy, vz | 比较 double 值 vy 和 vz 并设置整数 赋值到 vx. |
cmp-long vx, vy, vz | 比较 long 值 vy 和 vz 并设置整数 赋值到 vx. |
if-eq vx,vy,目标 | 如果vx == vy注2,跳转到目标。 |
if-ne vx,vy,目标 | 如果vx != vy注2,跳转到目标 |
if-lt vx,vy,目标 | 如果vx < vy注2,跳转到目标 |
if-ge vx, vy,目标 | 如果vx >= vy注2,跳转到目标。 |
if-gt vx,vy,目标 | 如果vx > vy注2,跳转到目标 |
if-le vx,vy,目标 | 如果vx <= vy注2,跳转到目标 |
if-eqz vx,目标 | 如果vx == 0注2,跳转到目标。 |
if-nez vx,目标 | 如果vx != 0注2,跳转到目标。 |
if-ltz vx,目标 | 如果vx < 0跳转到目标 |
if-gez vx,目标 | 如果vx >= 0跳转到目标. |
if-gtz vx,目标 | 如果vx > 0跳转到目标. |
if-lez vx,目标 | 如果vx <= 0跳转到目标. |
aget vx,vy,vz | 获得vy对应的int数组中索引为vz的值并赋值给vx |
aget-object vx,vy,vz | 获得对象引用值和 n对象引用 数组赋值到 vx vy 引用数组和vz引索 |
aget-boolean vx,vy,vz | 获得 boolean值和boolean 数组赋值到 vx vy 引用数组和vz引索 |
aget-byte vx,vy,vz | 获得 byte值把byte 数组赋值到 vx vy 引用数组和vz引索 |
aget-char vx, vy,vz | 获得vy对应的的char数组中索引为vz的值并赋值给vx |
aput vx,vy,vz | 把integer 赋值到 vx得到整数元素组. . |
aput-wide vx,vy,vz | 把double/long 赋值到 vx,vx+1 成double/long 数组. |
aput-object vx,vy,vz | 把vy对象引用 赋值到 vx得到整数元素组 |
aput-boolean vx,vy,vz | 把boolean 赋值到 vx得到元素 boolean 数组 . |
aput-byte vx,vy,vz | 把 byte 赋值到 vx得到元素 byte 数组 . |
aput-char vx,vy,vz | 把 char 赋值到 vx 字符数组的一个元素被vy引用的数组对象. |
aput-short vx,vy,vz | 把 short 赋值到 vx得到元素 short 数组 . |
iget vx, vy, field_id | 读取一个实例字段到 vx. 实例从vy引用. |
iget-wide vx,vy,field_id | 读取一个实例字段到 vx1. 实例从vy引用. |
iget-object vx,vy,field_id | 读取一个对象引用 实例字段到 vx. 实例从vy引用. |
iget-boolean vx,vy,field_id | 读取 boolean 实例字段到 vx. 实例从vy引用. |
iget-byte vx,vy,field_id | 读取 byte 实例字段到 vx. 实例从vy引用. |
iget-char vx,vy,field_id | 读取 char 实例字段到 vx. 实例从vy引用. |
iget-short vx,vy,field_id | 读取 short 实例字段到 vx. 实例从vy引用. |
iput vx,vy, field_id | 将vx放入一个实例字段. 实例从vy引用. |
iput-wide vx,vy, field_id | 把 wide 数值 in vx and vx+1 注册到一个实例字段 实例从vy引用. |
iput-object vx,vy,field_id | 把 the对象引用在vx实际化字段. 实例从vy引用. |
iput-boolean vx,vy, field_id | 把 boolean 数值在vx实际化字段 实例从vy引用. |
iput-byte vx,vy,field_id | 把 byte 数值在vx实际化字段. 实例从vy引用. |
iput-char vx,vy,field_id | 把 char 数值在vx实际化字段. 实例从vy引用. |
iput-short vx,vy,field_id | 把 short 数值在vx实际化字段. 实例从vy引用. |
sget vx,field_id 从 field_id | 读取整个字段到 vx. |
sget-wide vx, field_id | 从 field_id 读取静态字段到 vx 和 vx+1 寄存器. |
sget-object vx,field_id | 读取对象引用 field_id 到 vx. |
sget-boolean vx,field_id | 读取 boolean 静态字段 field_id 到 vx. |
sget-byte vx,field_id | 读取 byte 静态字段 field_id 到 vx. |
sget-char vx,field_id | 读取 char 静态字段 field_id 到 vx. |
sget-short vx,field_id | 读取 short 静态字段 field_id 到 vx. |
sput vx, field_id | 把 vx静态字段 |
sput-wide vx, field_id | 把 vx and vx+1 转到一个静态字段 |
sput-object vx,field_id | 对象引用 in vx 转到一个静态字段 |
sput-boolean vx,field_id | 把 boolean 赋值到 vx转到一个静态字段 |
sput-byte vx,field_id | 把 byte 赋值到 vx转到一个静态字段 |
sput-char vx,field_id | 把 char 赋值到 vx 转到一个静态字段 |
sput-short vx,field_id | 把 short 赋值到 vx 转到一个静态字段 |
invoke-virtual { parameters },<br> methodtocall | 调用一个虚拟方法与参数. |
invoke-super {parameter},methodtocall | 调用直接父类的虚方法. |
invoke-direct { parameters },<br> methodtocall | 调用一个方法参数,没有虚拟方法解析 |
invoke-static {parameters}, methodtocall | 调用一个静态方法与参数 |
invoke-interface<br> {parameters},methodtocall | 调用接口方法 |
invoke-virtual/range<br> {vx..vy},methodtocall | 调用虚拟方法的寄存器.指定一个寄存器和寄存器数量被传递给方法 |
invoke-super/range | 直接调用父类的虚方法。指定一个寄存器和寄存器数量被传递给方法 |
invoke-direct/range<br> {vx..vy},methodtocall | 直接调用寄存器方法指定一个寄存器和寄存器数量被传递给方法 |
invoke-static/range<br> {vx..vy},methodtocall | 调用静态方法寄存器指定一个寄存器和寄存器数量被传递给方法 |
invoke-interface-range | 调用一个接口方法寄存器,指定一个寄存器和寄存器数量被传递给方法 |
neg-int vx,vy | 计算 vx = -vy. |
neg-long vx,vy | 计算 vx,vx+1=-(vy,vy+1) |
neg-float vx,vy | 计算 vx = -vy |
neg-double vx,vy | 计算 vx,vx+1=-(vy,vy+1) |
int-to-long vx, vy | 转换整数到 vy 并转成 long in vx,vx+1. |
int-to-float vx, vy | 转换整数到 vx 并转成 float in vx. |
int-to-double vx, vy | 转换vy到整数 double in vx,vx+1. |
long-to-int vx,vy | 转换 long vy 值 ,vy+1 到整数储存到 vx |
long-to-float vx, vy | 转换 long vy 值 ,vy+1 成 float in vx. |
long-to-double vx, vy | 转换 long vy 值 ,vy+1 成 double 赋值到 vx,vx+1. |
float-to-int vx, vy | 转换 float vy 值到整数并赋值到 vx. |
float-to-long vx,vy | 转换 float vy 值成long 赋值到 vx. |
float-to-double vx, vy | 转换 float vy 值成double 赋值到 vx,vx+1. |
double-to-int vx, vy | 转换 double vy 值 ,vy+1 到整数赋值到 vx. |
double-to-long vx, vy | 转换 double vy 值 ,vy+1 成long 赋值到 vx,vx+1. |
double-to-float vx, vy | 转换 double vy 值 ,vy+1 成float 赋值到 vx. |
int-to-byte vx,vy | 转换 int vy 值到 byte 值,并保存到vx. |
int-to-char vx,vy | 转换 int vy 值到char 值,并保存到vx. |
int-to-short vx,vy | 转换 int vy 值到 short 值,并保存到vx. |
add-int vx,vy,vz | 计算 vy + vz 并将结果放到 vx. |
sub-int vx,vy,vz | 计算 vy - vz 并将结果放到 vx. |
mul-int vx, vy, vz | vz 与 vy 相乘并将结果转int放到 vx. |
div-int vx,vy,vz | Divides vy 与 vz 并将结果赋值到 vx. |
rem-int vx,vy,vz | 计算 vy % vz 并将结果赋值到 vx. |
and-int vx, vy, vz | 计算 vy AND vz 并将结果赋值到vx. |
or-int vx, vy, vz | 计算 vy OR vz 并将结果赋值到vx. |
xor-int vx, vy, vz | 计算 vy XOR vz 并将结果赋值到vx. |
shl-int vx, vy, vz | 转换 vy 指定的位置留下后并储存结果 到 vx. |
shr-int vx, vy, vz | 从vy指定位置转换并储存到 vx. |
ushr-int vx, vy, vz | 无符号右移 (>>>) vy 指定位置后将结果储存到 vx |
add-long vx, vy, vz | 添加 vy 到 vz 并将结果赋值到 vx1. |
sub-long vx,vy,vz | 计算 vy - vz 并将结果赋值到 vx1. |
mul-long vx,vy,vz | 计算 vy * vz 并将结果赋值到 vx1. |
div-long vx, vy, vz | 计算 vy / vz 并将结果赋值到 vx1. |
rem-long vx,vy,vz | 计算 vy % vz 并将结果赋值到 vx1. |
and-long vx, vy, vz | 计算 the vy AND vz 并将结果赋值到 vx1. |
or-long vx, vy, vz | 计算 the vy OR vz 并将结果赋值到 vx1. |
xor-long vx, vy, vz | 计算 the vy XOR vz 并将结果赋值到 vx1. |
shl-long vx, vy, vz | Shifts left vy by vz positions 并将结果储存在 vx1. |
shr-long vx,vy,vz | Shifts right vy by vz positions 并将结果储存在 vx1. |
ushr-long vx, vy, vz | Unsigned shifts right vy by vz positions 并将结果储存在 vx1. |
add-float vx,vy,vz | 添加 vy 到 vz 并将结果赋值到 vx. |
sub-float vx,vy,vz | 计算 vy - vz 并将结果赋值到 vx. |
mul-float vx, vy, vz | 增加 vy 与 vz 并将结果赋值到 vx. |
div-float vx, vy, vz | 计算 vy / vz 并将结果赋值到 vx. |
rem-float vx,vy,vz | 计算 vy % vz 并将结果赋值到 vx. |
add-double vx,vy,vz | 添加 vy 到 vz 并将结果赋值到 vx1. |
sub-double vx,vy,vz | 计算 vy - vz 并将结果赋值到 vx1. |
mul-double vx, vy, vz | 增加 vy 与 vz 并将结果赋值到 vx1. |
div-double vx, vy, vz | 计算 vy / vz 并将结果赋值到 vx1. |
rem-double vx,vy,vz | 计算 vy % vz 并将结果赋值到 vx1. |
add-int/2addr vx,vy | 添加 vy 到 vx. |
sub-int/2addr vx,vy | 计算 vx - vy 并将结果赋值到 vx. |
mul-int/2addr vx,vy | 相乘 vx 和 vy. |
div-int/2addr vx,vy | Divides vx 和 vy 并将结果赋值到 vx. |
rem-int/2addr vx,vy | 计算 vx % vy 并将结果赋值到 vx |
and-int/2addr vx, vy | 计算 vx AND vy 并将结果赋值到 vx. |
or-int/2addr vx, vy | 计算 vx OR vy 并将结果赋值到 vx. |
xor-int/2addr vx, vy | 计算 vx XOR vy 并将结果赋值到 vx. |
shl-int/2addr vx, vy | Shifts vx left by vy positions. |
shr-int/2addr vx, vy | 从 vy 转换到 vx. |
ushr-int/2addr vx, vy | 无符号右移vy到vx的指定位置 |
add-long/2addr vx,vy | 添加 vy 到 vx1. |
sub-long/2addr vx,vy | 计算 vx - vy 并将结果赋值到 vx1. |
mul-long/2addr vx,vy | 计算 vx * vy 并将结果赋值到 vx1. |
div-long/2addr vx, vy | 计算 vx / vy 并将结果赋值到 vx1. |
rem-long/2addr vx,vy | 计算 vx % vy 并将结果赋值到 vx1. |
and-long/2addr vx, vy | 计算 vx AND vy 并将结果赋值到 vx1. |
or-long/2addr vx, vy | 计算 vx OR vy 并将结果赋值到 vx1. |
xor-long/2addr vx, vy | 计算 vx XOR vy 并将结果赋值到 vx1. |
shl-long/2addr vx, vy | Shifts left the 赋值到 vx,vx+1 从 vy 指定的位置 并将结果储存在 vx,vx+1. |
shr-long/2addr vx, vy | Shifts right the 赋值到 vx,vx+1 从 vy 指定的位置 并将结果储存在 vx,vx+1. |
ushr-long/2addr vx, vy | 无条件转移vy赋值到 vx,vx+1 并将结果储存在 vx,vx+1. |
add-float/2addr vx,vy | 添加 vy 到 vx. |
sub-float/2addr vx,vy | 计算 vx - vy 并将结果储存在 vx. |
mul-float/2addr vx, vy | 相乘 vx 和 vy. |
div-float/2addr vx, vy | 计算 vx / vy 并将结果赋值到 vx. |
rem-float/2addr vx,vy | 计算 vx / vy 并将结果赋值到 vx. |
add-double/2addr vx, vy | 添加 vy 到 vx1. |
sub-double/2addr vx, vy | 计算 vx - vy 并将结果赋值到 vx1. |
mul-double/2addr vx, vy | 相乘 vx 和 vy1. |
div-double/2addr vx, vy | 计算 vx / vy 并将结果赋值到 vx1. |
rem-double/2addr vx,vy | 计算 vx % vy 并将结果赋值到 vx1. |
add-int/lit16 vx,vy,lit16 | 添加 vy to lit16 并将结果储存到 vx |
sub-int/lit16 vx,vy,lit16 | 计算 vy - lit16 并将结果储存到 vx |
mul-int/lit16 vx,vy,lit16 | 计算 vy * lit16 并将结果储存到 vx |
div-int/lit16 vx,vy,lit16 | 计算 vy / lit16 并将结果储存到 vx |
rem-int/lit16 vx,vy,lit16 | 计算 vy % lit16 并将结果储存到 vx |
and-int/lit16 vx,vy,lit16 | 计算 vy AND lit16 并将结果储存到 vx |
or-int/lit16 vx,vy,lit16 | 计算 vy OR lit16 并将结果储存到 vx |
xor-int/lit16 vx,vy,lit16 | 计算 vy XOR lit16 并将结果储存到 vx |
add-int/lit8 vx,vy,lit8 | 添加 vy to lit8 并将结果储存到 vx |
sub-int/lit8 vx,vy,lit8 | 计算 vy - lit8 并将结果储存到 vx |
mul-int/lit8 vx,vy,lit8 | 增加 vy 与 lit8 8-bit 文字常量 并将结果赋值到 vx. |
div-int/lit8 vx,vy,lit8 | 计算 vy / lit8 并将结果储存到 vx |
rem-int/lit8 vx,vy,lit8 | 计算 vy % lit8 并将结果储存到 vx |
and-int/lit8 vx,vy,lit8 | 计算 vy AND lit8 并将结果储存到 vx |
or-int/lit8 vx, vy, lit8 | 计算 vy OR lit8 并将结果赋值到 vx. |
xor-int/lit8 vx, vy, lit8 | 计算 vy XOR lit8 并将结果赋值到 vx. |
shl-int/lit8 vx, vy, lit8 | 转换 v0 留下指定位置 |
shr-int/lit8 vx, vy, lit8 | 从指定位置读取文字常量并将结果储存到 vx. |
ushr-int/lit8 vx, vy, lit8 | 无符号右移 v0 (>>>) |
execute-inline {parameters},inline ID | 执行内联方法内联 ID. |
invoke-direct-empty | 占位符修改空对象.<init>. 期间使用nop正常执行6. |
iget-quick vx,vy,offset | 从vy实例的数据区获得的数据储存到vx |
iget-wide-quick vx,vy,offset | 获取对象引用偏移vy值到vx,vx+16. |
iget-object-quick vx,vy,offset | 获取对象引用 偏移vy值到vx6. |
iput-quick vx,vy,offset vy | 实例数据抵消并将值储存在 vx. |
iput-wide-quick vx,vy,offset | 将值储存在 vx,vx+1 to offset in vy instance's data area6. |
iput-object-quick vx,vy,offset | 将对象引用值储存在vx来抵消vy实例的数据区域 |
更多语法
vim
/* smali invoke */
invoke-virtual-quick {parameters},vtable<br> offset #调用方法目地址对象
invoke-virtual-quick/range {parameter<br> range},vtable offset #调用一个方法使用vtable的目标对象
invoke-super-quick {parameters},vtable<br> offset #调用一个虚方法直接在目标对象的父类直接使用.
invoke-super-quick/range {register<br> range},vtable offset #调用一个虚方法在目标对象父类使用.
提示
- filled-new-array {parameters},type_id
生成新数组 type_id 并填充 parameters5. 生成的数组可以通过 move-result-object 引用后使用 filled-new-array 指令 - filled-new-array-range {vx..vy},type_id
生成一个新数组 type_id 和填补参数 生成新数组引用可以通过 move-result-object 指令使用 filled-new-array fill-array-data vx,array_data_offset 填充数组引用到vx