GNU风格 ARM汇编语法指南
.ascii "Temp string \n" @对这一句的理解,我觉得应该是:将"Temp string \n"这个字符串存储在以标号strtemp:
@为起始地址的一段内存空间里
<2>汇编系统预定义的段名
l.text @代码段
l.data @初始化数据段.data Read-write initialized long data.
l.bss @未初始化数据段
l.sdata @ .sdata Read-write initialized short data.
l.sbss @
注意:源程序中.bss段应该在.text段之前。
4、GNU汇编语言定义入口点
汇编程序的缺省入口是_start标号,用户也可以在连接脚本文件中用ENTRY标志指明其它入口点。
例:定义入口点
.section .data
< initialized data here>
.section .bss
< uninitialized data here>
.section .text
.globl _start
_start:
5、GNU汇编程序中的宏定义
格式如下:
.macro 宏名参数名列表 @伪指令.macro定义一个宏
宏体
.endm @.endm表示宏结束
如果宏使用参数,那么在宏体中使用该参数时添加前缀“\”。宏定义时的参数还可以使用默认值。可以使用.exitm伪指令来退出宏。
例:宏定义
.macro SHIFTLEFT a, b
.if \b < 0
MOV \a, \a, ASR #-\b
.exitm
.endif
MOV \a, \a, LSL #\b
.endm
6、GNU汇编程序中的常数
<1>十进制数以非0数字开头,如:123和9876;
<2>二进制数以0b开头,其中字母也可以为大写;
<3>八进制数以0开始,如:0456,0123;
<4>十六进制数以0x开头,如:0xabcd,0X123f;
<5>字符串常量需要用引号括起来,中间也可以使用转义字符,如: “You are welcome!\n”;
<6>当前地址以“.”表示,在GNU汇编程序中可以使用这个符号代表当前指令的地址;
<7>表达式:在汇编程序中的表达式可以使用常数或者数值, “-”表示取负数, “~”表示取补,“<>”表示不相等,其他的符号如:+、-、*、 /、%、<、<、>、>>、|、&、^、!、==、>=、<=、&&、|| 跟C语言中的用法相似。
7、GNU ARM汇编的常用伪操作
在前面已经提到过了一些为操作,还有下面一些为操作:
l数据定义伪操作: .byte,.short,.long,.quad,.float,.string/.asciz/.ascii,重复定义伪操作.rept,赋值语句.equ/.set ;
l函数的定义;
l对齐方式伪操作 .align;
l源文件结束伪操作.end;
l.include伪操作;
lif伪操作;
l.global/ .globl 伪操作;
l.type伪操作;
l列表控制语句;
别于GNUAS汇编的通用伪操作,下面是ARM特有的伪操作:
.reg ,.unreq ,.code ,.thumb ,.thumb_func ,.thumb_set, .ltorg ,.pool
<1>数据定义伪操作
l.byte:单字节定义,如:.byte 1,2,0b01,0x34,072,s ;
l.short:定义双字节数据,如:.short 0x1234,60000 ;
l.long:定义4字节数据,如:.long 0x12345678,23876565
l.quad:定义8字节,如:.quad 0x1234567890abcd
l.float:定义浮点数,如:.float0f-314159265358979323846264338327\
95028841971.693993751E-40 @- pi
l.string/.asciz/.ascii:定义多个字符串,如:
.string "abcd", "efgh", "hello!"
.asciz "qwer", "sun", "world!"
.ascii "welcome"
注意:ascii伪操作定义的字符串需要自行添加结尾字符。
l.rept:重复定义伪操作, 格式如下:
.rept 重复次数
数据定义
.endr @结束重复定义
例:
.rept 3
.byte 0x23
.endr
l.equ/.set: 赋值语句, 格式如下:
.equ(.set) 变量名,表达式
例:
.equ abc, 3 @让abc=3
<2>函数的定义伪操作
l函数的定义,格式如下:
函数名:
函数体
返回语句
一般的,函数如果需要在其他文件中调用,需要用到.global伪操作将函数声明为全局函数。为了不至于在其他程序在调用某个C函数时发生混乱,对寄存器的使用我们需要遵循APCS准则。函数编译器将处理函数代码为一段.global的汇编码。
l函数的编写应当遵循如下规则:
a.a1-a4寄存器(参数、结果或暂存寄存器,r0到r3 的同义字)以及浮点寄存器f0-f3(如果存在浮点协处理器)在函数中是不必保存的;
b.如果函数返回一个不大于一个字大小的值,则在函数结束时应该把这个值送到 r0 中;
c.如果函数返回一个浮点数,则在函数结束时把它放入浮点寄存器f0中;
d.如果函数的过程改动了sp(堆栈指针,r13)、fp(框架指针,r11)、sl(堆栈限制,r10)、lr(连接寄存器,r14)、v1-v8(变量寄存器,r4 到 r11)和 f4-f7,那么函数结束时这些寄存器应当被恢复为包含在进入函数时它所持有的值。
<3>.align .end .include .incbin伪操作
l.align:用来指定数据的对齐方式,格式如下:
.align [absexpr1, absexpr2]
以某种对齐方式,在未使用的存储区域填充值.第一个值表示对齐方式,4, 8,16或32.第二个表达式值表示填充的值。
l.end:表明源文件的结束。
l.include:可以将指定的文件在使用.include 的地方展开,一般是头文件,例如:
GNU风格ARM汇编语法指 相关文章:
- Windows CE 进程、线程和内存管理(11-09)
- RedHatLinux新手入门教程(5)(11-12)
- uClinux介绍(11-09)
- openwebmailV1.60安装教学(11-12)
- Linux嵌入式系统开发平台选型探讨(11-09)
- Windows CE 进程、线程和内存管理(二)(11-09)