网站大量收购独家精品文档,联系QQ:2885784924

iOS 逆向工程与ARM汇编.docVIP

  1. 1、本文档共9页,可阅读全部内容。
  2. 2、有哪些信誉好的足球投注网站(book118)网站文档一经付费(服务费),不意味着购买了该文档的版权,仅供个人/单位学习、研究之用,不得用于商业用途,未经授权,严禁复制、发行、汇编、翻译或者网络传播等,侵权必究。
  3. 3、本站所有内容均由合作方或网友上传,本站不对文档的完整性、权威性及其观点立场正确性做任何保证或承诺!文档内容仅供研究参考,付费前请自行鉴别。如您付费,意味着您自己接受本站规则且自行承担风险,本站不退款、不进行额外附加服务;查看《如何避免下载的几个坑》。如果您已付费下载过本站文档,您可以点击 这里二次下载
  4. 4、如文档侵犯商业秘密、侵犯著作权、侵犯人身权等,请点击“版权申诉”(推荐),也可以打举报电话:400-050-0827(电话支持时间:9:00-18:30)。
  5. 5、该文档为VIP文档,如果想要下载,成为VIP会员后,下载免费。
  6. 6、成为VIP后,下载本文档将扣除1次下载权益。下载后,不支持退款、换文档。如有疑问请联系我们
  7. 7、成为VIP后,您将拥有八大权益,权益包括:VIP文档下载权益、阅读免打扰、文档格式转换、高级专利检索、专属身份标志、高级客服、多端互通、版权登记。
  8. 8、VIP文档为合作方或网友上传,每下载1次, 网站将根据用户上传文档的质量评分、类型等,对文档贡献者给予高额补贴、流量扶持。如果你也想贡献VIP文档。上传文档
查看更多
iOS 逆向工程与ARM汇编.doc

iOS 逆向之ARM汇编 我们先讲一些ARM汇编的基础知识。(我们以ARMV7为例,必威体育精装版iPhone5s上的64位暂不讨论) 基础知识部分:? 首先介绍一下寄存器: R0-R3:用于函数参数及返回值的传递 R4-R6, R8,?R10-R11:没有特殊规定,就是普通的通用寄存器 R7:栈帧指针(Frame Pointer).指向前一个保存的栈帧(stack frame)和链接寄存器(link register, lr)在栈上的地址。 R9:操作系统保留 R12:又叫IP(intra-procedure scratch?), R13:又叫SP(stack pointer),是栈顶指针 R14:又叫LR(link register),存放函数的返回地址。 R15:又叫PC(program counter),指向当前指令地址。 CPSR:当前程序状态寄存器(Current Program State Register),在用户状态下存放像condition标志中断禁用等标志的。    ? 在其它系统状态中断状等状态下与CPSR对应还有一个SPSR,在这里不详述了。 另外还有VFP(向量浮点运算)相关的寄存器,在此我们略过,感兴趣的可以从后面的参考链接去查看。? 基本的指令: add 加指令 sub 减指令 str 把寄存器内容存到栈上去 ldr ?把栈上内容载入一寄存器中 .w是一个可选的指令宽度说明符。它不会影响为此指令的行为,它只是确保生成 32 位指令。 bl 执行函数调用,并把使lr指向调用者(caller)的下一条指令,即函数的返回地址 blx 同上,但是在ARM和thumb指令集间切换。 bx ?bx lr返回调用函数(caller)。? 接下来是函数调用的一些规则。 一. 在iOS中你需要使用BLX,BX这些指令来调用函数,不能使用MOV指令(具体意义下面会说) 二. ARM使用一个栈来来维护函数的调用及返回。ARM中栈是向下生长(由高地址向低地址生长的)。 函数调用前后栈的布局如图一(引用的苹果iOS ABI Reference):               图(一)SP(stack pointer)指向栈顶(栈低在高地址)。栈帧(stack frame)其实就是通过R7及存在栈上的旧R7来标识的栈上的一块一块的存储空间。栈帧包括: 参数区域(parameter area),存放调用函数传递的参数。对于32位ARM,前4个参数通过r0-r3传递,多余的参数通过栈来传递,就是存放在这个区域的。 链接区域(linkage area),存放调用者(caller)的下一条指令。 栈帧指针存放区域(saved frame pointer),存放调用函数的栈帧的底部,标识着调用者(caller)栈帧的结束及被调用函数(callee)的栈帧开始。 局部变量存储区(local storage area)。用于存被调函数(callee)的局部变量及在被调用函数(callee)结束后反回调用函数(call)之前需要恢复的寄存器内容。 寄存器存储区(saved registers area)。Apple的文档中是这样说的。但我认为这个区域和local storage area相邻且干的事也是存放需要恢复的寄存器内容,因此我觉得要不就把这个区域在概念上不区分出来,要不就把存放需要恢复的寄存器这项功能从local storage area中分出来。 当然这些都只是概念上的,其实实质上是没有区别的。 接下来看看在调用子函数开始及结尾时所要做的事情。(官方叫序言和结语, prologs and epilogs) 调用开始: LR入栈 R7入栈 R7 = SP地址。在经过前面两条入栈指令后,SP指向的地址向下移动,再把SP赋值给R7, 标志着caller栈帧的结束及callee的栈帧的开始 将callee会修改且在返回caller时需要恢复的寄存器入栈。 分配栈空间给子程序使用。由于栈是从高地址向低地址生长,所以通常使用sub sp, #size来分配。 调用结尾: 释放栈空间。add sp, #size指令。 恢复所保存的寄存器。 恢复R7 将之前存放的LR从栈上弹出到PC,这样函数就返回了。 -----------------------------------------------------------华丽的分割线------------------------------------------------------------- 实战部分(一): 用XCode创建一个Test工程,新建一个.c文件,添加如下函数: 1 2 3 4 5 6 7 #include stdio.h ? int func(int a, int b, in

文档评论(0)

mghkfg58 + 关注
实名认证
文档贡献者

该用户很懒,什么也没介绍

1亿VIP精品文档

相关文档