参考资料:1. Richard Blum,Professional Assembly Language2. GNU ARM 汇编快速入门,/u/31996/showart.php?id=3261463. ARM GNU 汇编伪指令简介,/jb8164/archive/2008/01/22/41661.aspx4. GNU汇编使用经验,/u1/37614/showart_390095.html5. GNU的编译器和开发工具,/blog-htm-do-showone-uid-34335-itemid-81387-type-blog.html6. 用GNU工具开发基于ARM的嵌入式系统,/liren0@126/blog/static/32897598200821211144696/7. objcopy命令介绍,/junhua198310/archive/2007/06/27/1669545.aspx从网上找到一些关于ARM伪指令的资料,现整理如下:一. Linux汇编行结构任何汇编行都是如下结构:[:] [} @ comment[:] [} @ 注释Linux ARM 汇编中,任何以冒号结尾的标识符都被认为是一个标号,而不一定非要在一行的开始。
【例1】定义一个"add"的函数,返回两个参数的和。
.section .text, “x”.global add @ give the symbol add external linkageadd:ADD r0, r0, r1 @ add input argumentsMOV pc, lr @ return from subroutine@ end of program1. LDRLDR R0, =0X3FF5000 ;伪指令,把0X3FF5000这个地址送给R0LDR R0, 0XFF ; 把立即数0xff送给R0LDR R0, =&FF ; &相当于0XBIC R0, R0, #%1011 ;.#表示立即数,%表示二进制LDR R1, =0x3ff5000 ;伪指令 R1=0X3FF5000LDR R1, 0x3ff5000 ;存储器访问指令 R1= [0x3ff5000]2. adr与ldr比较adr r0, InitSystem ;ldr r1, =InitSystem ;伪指令adr r0,InitSystem 编译时汇编成:sub r0,PC,#offset to InitSystemLDR r1,=InitSystem ,这种方式读取的地址值在连接时已经被固定了,这种代码不是位置无关的。
遇到LDR伪指令时,汇编编译器将该地址值保存到一个缓冲区(l iteral pool)中,然后将该LDR 伪指令处理成一条基于PC到该数据缓冲区单元的LD R 指令,从而将该地址值读取到寄存器总,这时,要求该数据缓冲区到PC的距离小于4KB。
如果该目标地址值为一个外部地址值或者不在本数据段内,则汇编译器在目标文件中插入一个地址重定位伪操作,当连接器进行连接时生成该地址值。
LDR r1,=InitSystem 汇编成:LDR R1,[PC,#offset to Litpool1]------------------------------------------------------adr用来加载地址,例如adr r0,var1ldr用来加载地址处的内容,例如ldr r0,var1上面的这种语法只能从.text段中加载但ldr r0,=var1可从任意段中加载地址ldr有伪指令和非伪指令,伪指令后面的立即数前加=ADR在编译时会被替换成一条add或者sub指令,如果替换不了则报错。
相对PC寻址ADRL会被替换成两条指令,替换不了报错误。
相对PC或者积存器寻址这两条指令依据立即数的对齐方式不同,允许的立即数范围也不同。
LDR则是产生文字池的方式加载常量,基于PC的相对寻址,专用加载32bit立即数.通过反汇编可以很容易看出LDR和ADR区别:假设入口点地址为0x8000AREA LDRlabel, CODE, READONLYENTRY ; Mark first instruction to executestartBL func1 ; Branch to first subroutinestopMOV r0, #0x18 ; angel_SWIreason_ReportExceptionLDR r1, =0x20026 ; ADP_Stopped_ApplicationExitLDR r1,=0xffSWI 0x123456 ; ARM semihosting SWIfunc1LDR r0, =start ; => LDR R0,[PC, #offset to Litpool 1]ADR r2,startLDR r1, =Darea +12 ; => LDR R1,[PC, #offset to Litpool 1]ADR r3,Darea+12MOV pc,lr ; ReturnLTORG ; Literal Pool 1Darea SPACE 8000 ; Clears a 8000 byte area of memory,; starting at the current location,; to zero.END反汇编后:start [0xeb000003] bl func1stop [0xe3a00018] mov r0,#0x1800008008 [0xe59f1018] ldr r1,0x00008028 ; = #0x00020026 0000800c [0xe3a010ff] mov r1,#0xff00008010 [0xef123456] swi 0x123456func1 [0xe59f0010] ldr r0,0x0000802c ; = #0x0000800000008018 [0xe24f2020] sub r2,pc,#0x20 ; #0x80000000801c [0xe59f100c] ldr r1,0x00008030 ; = #0x00008040 00008020 [0xe28f3018] add r3,pc,#0x18 ; #0x804000008024 [0xe1a0f00e] mov pc,r1400008028 [0x00020026] dcd 0x00020026&...0000802c [0x00008000] dcd 0x00008000....00008030 [0x00008040] dcd 0x00008040@...Darea [0x00000000] dcd 0x00000000....00008038 [0x00000000] dcd 0x00000000....0000803c [0x00000000] dcd 0x00000000....00008040 [0x00000000] dcd 0x00000000....00008044 [0x00000000] dcd 0x00000000....……….3. MOVMOV加载8位立即数8位立即数即第2操作数,必须可由一个8位常数循环移位偶数位得到,如0xf0000000,0xf00000001都是合法的4. 数据回写例如:ldr r0,[r1]!stmdb sp!,{r0,r4}!用于前索引方式中表示数据回写,例如:ldr r0,[r1,#4]!后索引方式不用!, 数据始终回写,例如:ldr r0,[r1],#4对堆栈方式中用!, 表示堆栈自动增加或者减少访问内存的LDR/STR指令索引方式ldr r0,[r1,#4]前索引, 先加ldr r0,[r1],#4后索引, 后加5. DCD和SPACEDCD分配一个地址并初始化为指定的表达式如 DCD 0x8000就是分配一个32位的地址,其内容是0x8000如果用上标号label,那么这个label相当于一个变量,如label DCD 0X8000这样引用label就是使用了0x8000这个数值。
SPACE分配一段指定长度的内存空间并初始化为0如label SPACE 0x8000就是分配一段长度为0x8000的空间,并初始化为0Data1 DCD 1,2,3Data2 SPACE 12反汇编后的结果:Data1 [0x00000001] dcd 0x00000001 ....00008040 [0x00000002] dcd 0x00000002 ....00008044 [0x00000003] dcd 0x00000003 ....Data2 [0x00000000] dcd 0x00000000 ....0000804c [0x00000000] dcd 0x00000000 ....00008050 [0x00000000] dcd 0x00000000 ....6. LTORG与LDRLTORG是与LDR联合使用的literal pool,可以在函数尾部声明, 这样相对PC偏移最少, 如果不用LTORG, 则编译器自动在(整个)程序末尾声名,但这样偏移有可能太大而编译不通过.ARM立即数,LDR和MOV的区别Mov 是把立即数赋给一个寄存器,但对立即数的范围有要求。
只能是由8bit 连续有效位通过偶数次移位能得到的数。
如果立即数超出这个范围,就没办法用一条MOV指令给寄存器赋值。
LDR除了普通的读数之外,也有给寄存器赋立即数的功能。
你只要写LDR R0,=0xabcdef它没有立即数范围的限制。
因为这是一条伪指令。
如果立即数在MOV的要求内,那就用一条汇编来实现。
如果不在Mov的范围内,就用其它方式实现,如变成两条指令,或从PC偏移地址读一个32位数给寄存器。
MOV是从一个寄存器或者移位的寄存器或者立即数的值传递到另外一个寄存器从本质上是寄存器到寄存器的传递,为什么会有立即数,其实也是有限制的立即数,不是所有立即数都可以传递的这个立即数要符合一个8位数循环右移偶数位的取值原因是,MOV本身就是一条32bit指令,除了指令码本身,它不可能再带一个可以表示32bit的数字,所以用了其中的12bit来表示立即数,其中4bit表示移位的位数(循环右移,且数值x2),8bit用来表示要移位的一个基数。
还有一点是关于ldr的,其实ldr可以装载一个32bit立即数的说法并不确切,因为实际上并不是这一条语句装载了一个32bit立即数,比如ldr r1,=0x12345678其实真正的汇编代码是将某个地址的值传递给r1,就是说需要一个地址存放0x12345678这个立即数,实际上可以看作是一条伪指令而且如果这个立即数可以用mov指令的形式来表达,会被编译器实际用mov 来代替比如:ldr r1,=0x10会变成mov r1,#0x10ARM是RISC结构,数据从内存到CPU之间的移动只能通过L/S指令来完成,也就是ldr/str指令。