arm32和arm64常用指令B BL BLX机器码计算

现在大部分手机cpu架构是ARM v7-A和ARMV8-A,,在ARM-v7A中常使用32位ARM指令集并且支持thumb指令集与arm的切换,而在ARMV8中使用的是64位ARM指令集且不再有thumb指令集状态的切换了。在调用函数时,会有常用的调用方式:BL和B,且分三种情况arm, thumb, aarch64,而对于BLX在arm64指令集中不再有。下面对这三种情况进行讨论。


ARM:



常见跳转指令机器码:

B:0xEA

BL:0xEB


偏移地址计算过程:


(目标地址 - 指令地址 - 8)/ 4 = 偏移

// 减8,指令流水造成。

// 除4,因为指令定长,存储指令个数差,而不是地址差。


完整指令:

.text:0000D11C D9 51 00 EB                             BL              __set_tls

.text:00021888                         __set_tls


计算偏移:

(21888-D11C - 8) /4 = 0x51D9

EB000000 | 0x51D9 =  EB0051D9


thumb:



thumb指令都是2字节。BL看起来像四字节指令,其实是个误解,因为长跳转是由两条跳转指令组成的。



0-11位表示11位地址,具体含义如下:

第11位为0,代表偏移高位

第11位为1,代表偏移低位


完整指令:

.text:0002E3C2 F2 F7 A2 ED                             BLX             nanosleep
.text:0002E3C6 38 B1                                   CBZ             R0, locret_2E3D8
.text:0002E3C8 E6 F7 66 F8                             BL              __errno
.text:0002E3CC 01 68                                   LDR             R1, [R0]

.text:00020F08                         nanosleep
.text:00014498                         __errno


先讲BL指令:


0002E3C8 E6 F7 66 F8                             BL              __errno

00014498                         __errno


如何得到目标地址的呢,计算方式如下:


解析偏移

F7E6(1111011111100110) 第11位为0,所以代表高位偏移

F866(1111100001100110)第11位为1,所以代表低位偏移


最高位:F7E6 取后11位 7E6

最低位:F866 取后11位 66

7E6 << 12(十进制) = 7E6000

66  << 1 =  CC

7E6000 | CC = 7E60CC


计算出来7E60CC最高位符号位为1,代表向前跳转,需要-1然后取反,得到数值为FFFFFFFFFF819F34,取19F34,

2E3C8+ 4 - 19F34 = 14498

若计算出来的数符号位为0,则直接保留该值,然后后面相加而不是减。


那偏移0xF866F7E6又该如何计算出来呢,计算方式如下:


offset = (目标地址- 源地址 -4) & 0x007fffff = 7E60CC

high = offset >> 12(十进制) = 7E6

low = ( offset & 0x00000fff )>>1  = 66


machineCode = ((0xF800 | low) << 16) | (0xF000 | high)
=F8660000 | F7E6 = 0xF866F7E6

注意F800相当于高偏移取第11位到第15位, F0000相当于取低偏移第11位到第15位。



再讲BLX指令:


0002E3C2 F2 F7 A2 ED                             BLX             nanosleep

00020F08                         nanosleep


那偏移0xEDA2 F7F2 如何得到的呢,跟BL算法稍微不同:


offset = (目标地址- 源地址 -4) & 0x007fffff = 7F2B42

high = offset >> 12 = 7F2

low = ( offset & 0x00000fff )>>1  = 5A1

if(low % 2 != 0) {

low++;

}//low=5A2


machineCode = ((0xE800 | low) << 16) | (0xF000 | high);
=EDA20000 | F7F2 = EDA2 F7F2


ARM64:



B:0x17向前跳转,0x14向后跳转

BL:0x97向前跳转  0x94向后跳转


偏移地址计算过程:

(目标地址 - 指令地址)/ 4 = 偏移

// 减8,指令流水造成。

// 除4,因为指令定长,存储指令个数差,而不是地址差。


完整指令:

.text:000000000008CC84 8D B3 FF 97                             BL              je_arena_malloc_hard

.text:0000000000079AB8                         je_arena_malloc_hard


计算偏移:

(79AB8-8CC84) / 4 = FFFFFFFFFFFFB38D

FFB38D | 0x97000000 = 97FFB38D

  • 3
    点赞
  • 16
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: ARM汇编指令有:ADD,SUB,MOV,MVN,CMP,AND,ORR,EOR,MUL,MLA,UMULL,UMLAL,LDR,STR,B,BL,BX,BLX,SWI,RSB,ADC,SBC,RSC,LSL,LSR,ASR,ROR,CLZ,CMN,TEQ,LDA,STM,LDM,SWP,SWPB,PUSH,POP,BKPT,SMLAL,SMULL,QADD,QSUB,QDADD,QDSUB,SMULW,SMLAW,SMLALBB,SMLALBT,SMLALTB,SMLALTT,SMULBB,SMULBT,SMULTB,SMULTT,SMLABB,SMLABT,SMLATB,SMLATT,SMULWB,SMUAWB,SMULWT,SMUAWT,SMLALBB,SMLALBT,SMLALTB,SMLALTT,SMLAWB,SMLAWT,SMLSD,SMLSLD,SMMLA,SMMLS,SMMUL,SMUAD,SMUSD,SMULXY,SMLALXY,SMLSLXY,SMMLAR,SMMLSR,SMMULR,SMUADX,SMUSDX,SMLA,SMLS,SMUL,SMUAD,SMUSD,SMULW,SMLAW,SMULX,SMLALX,SMLSLX,SMMLA,SMMLS,SMMUL,SMUAD,SMUSD,SMULW,SMLAW,SMULX,SMLALX,SMLSLX,SMMLA,SMMLS,SMMUL,SMUAD,SMUSD,SMULW,SMLAW,SMULX,SMLALX,SMLSLX。 ### 回答2: ARM汇编语言是一种低级程序设计语言,用于控制ARM架构处理器的指令集。ARM指令集由多个指令组成,这些指令可以执行各种操作,包括算术运算、逻辑运算、数据传输、分支跳转等。以下是一些常见的ARM汇编指令: 1. 数据处理指令:用于执行算术和逻辑运算,例如加法、减法、与、或等。 2. 数据传输指令:用于读取和存储数据,例如将数据从内存传输到寄存器,或者将数据从寄存器传输到内存。 3. 分支和跳转指令:用于控制程序流程,例如条件分支、无条件分支、函数调用和返回等。 4. 位移和移位指令:用于对数据进行位移和移位操作,例如逻辑左移、逻辑右移、算术右移等。 5. 系统控制指令:用于与操作系统和硬件交互,例如中断处理、特权级别切换等。 6. 协处理器指令:用于执行与协处理器相关的操作,例如浮点运算。 此外,ARM汇编语言还包括一些指令的变种和修饰符来支持更多的功能和操作。不同的ARM处理器可能有略微不同的指令集,因此具体指令的数量和功能可能会有所不同。 总而言之,ARM汇编指令集非常丰富,可以满足各种程序的需求。掌握这些指令可以帮助开发者更好地理解和优化底层程序。 ### 回答3: ARM汇编语言是一种底层的程序设计语言,用于编写与ARM架构处理器直接交互的程序。ARM汇编指令非常多,我将列举一些常见的指令。 1. 数据传送指令:包括LDR(加载数据到寄存器)、STR(将寄存器中的数据存储到指定地址)等。 2. 算术运算指令:例如ADD(加法)、SUB(减法)、MUL(乘法)等。 3. 逻辑运算指令:如AND(与运算)、ORR(或运算)、EOR(异或运算)等。 4. 控制指令:包括B(无条件分支)、BL(带有链接的分支)、CMP(比较指令)、BEQ(相等时分支)等。 5. 中断处理指令:如SWI(软中断)和SVC(超级方法调用)等用于处理中断请求的指令。 6. 位操作指令:例如TST(测试位)、LSL(逻辑左移位)和ASR(算术右移位)等。 7. 存储器访问指令:如LDM(多寄存器加载)和STM(多寄存器存储)等,用于对内存进行批量读写。 8. 条件执行指令:包括IT(条件执行标记)和FNE(不等于时执行)等,用于根据条件灵活地执行指令。 以上只是ARM汇编指令的一小部分示例,实际上ARM汇编指令非常丰富,还有很多其他类型的指令。不同的ARM架构处理器可能会有不同的指令集,因此具体的指令会因处理器型号而异。掌握并熟练使用这些指令,可以编写高效的底层程序。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值