1. GNU ARM 汇编简介
任何汇编行都是如下结构:
[<label>:] [<instruction or directive>} @ comment[<标签>:] [<指令>} @ 注释
GNU ARM 汇编中,任何以冒号结尾的都被认为是一个标签,而不一定非要在一行的开始。下面是一个简单的例子,这段汇编程序定义了一个"add"的函数,该函数返回两个参数的和:
.section .text, “x”
.global add @ give the symbol add external linkage
add:
ADD r0, r0, r1 @ add input arguments
MOV pc, lr @ return from subroutine
@ end of program
2. GNU ARM汇编伪指令
GNU ARM汇编伪指令均以.开始。常用汇编伪指如下所示:
汇编伪指令 | 描述 | 举例 |
.ascii “<string>” | 插入字符串到目标文件,但不以NULL结束 | .ascii "JNZ" @插入字节0x4A 0x4E 0x5A |
.asciz “<string>” | 与.ascii类似,但以NULL结束 | .asciz "JNZ" @插入字节0x4A 0x4E 0x5A 0x00 |
.balign <power_of_2> {,<fill_value> {,<max_padding>} } |
使下面的位置<power_of_2>对齐 | .byte 0x55 @插入字节: 0x55 .balign 4,0 @插入3个对齐字节:0x00 0x00 0x00 .word 0xAA55EE11 @插入字节:0x11 0xEE 0x55 0xAA (LSB order) |
.byte <byte1> {,<byte2>} … | 插入字节到目标文件 | .byte 64, 'A' @ inserts the bytes 0x40 0x41 .byte 0x42 @ inserts the byte 0x42 .byte 0b1000011, 0104 @ inserts the bytes 0x43 0x44 |
.code <number_of_bits> | 设置指令位数,16位Thumb或32位ARM | .code 16 @设置以下汇编为Thumb指令 |
.else | 与 .if 和 .endif 配合使用 | |
.end | 标志汇编文件结束,汇编器不再读后面的内容 | |
.endif | 怀.if 配合使用 | |
.endm | 标志宏定义结束,与.macro配合使用 | |
.endr | 结束一个loop,与.rept 和 .irp 配合使用 | |
.equ <symbol name>, <value> | 设置一个符号的值 | .equ adams, (5 * 8) + 2 @ set adams to 42 .set adams, 0x2A @ set adams to 42 adams = 0b00101010 @ set adams to 42 |
.err | 汇编时如遇到.err,则停止编译 | |
.exitm | 中途退出宏定义 | |
.global <symbol> | 标识symbol被程序的其它模块(汇编或C)访问, 相当于Kernel中的EXPORT_SYMBOL |
.global _my_test_count @它可被其它模块R/W |
.hword <short1> {,<short2>} … |
插入16位(半字)值到目标文件 | .hword 0xAA55, 12345 @ 插入字节 0x55 0xAA 0x39 0x30 .2byte 0x55AA, -1 @插入字节 0xAA 0x55 0xFF 0xFF @ Least Significant Byte (LSB) ordering assumed |
.if <logical_expression> | 与 .endif 配合使用 | |
.ifdef <symbol> | 与 .endif 配合使用 | |
.ifndef <symbol> | 与 .endif 配合使用 | |
.include “<filename>” | 与C的#include类似 | |
.irp <param> {,<val_1>} {,<val_2>} … |
Repeats a block of code, once for each value in the value list. Mark the end of the block using a .endr directive. In the repeated code block, use \<param> to substitute the associated value in the value list. |
|
.macro <name> {<arg_1} {,<arg_2>} … {,<arg_N>} |
定义一个名为name的汇编宏,它有N个参数, 且必须以.endm结束。 |
.macro SHIFTLEFT a, b .if \b < 0 MOV \a, \a, ASR #-\b .exitm .endif MOV \a, \a, LSL #\b .endm |
.section <section_name> {,”<flags>”} |
开始一个新的section, section_name可为: .text: 代码段 .data: 已初始化的数据段 .bss: 未初始化的数据段 flags可为: a allowable section w writable section x executable section |
.section .text, “x” |
.set <variable_name>, <variable_value> |
设置变化的值,与.equ一样 | .set adams, 0x2A @ set adams to 42 |
.space <number_of_bytes> {,<fill_byte>} |
预留给定字节空间,并设置为0或fill_byte | .space 0x400,0 |
.word <word1> {,<word2>} … | 插入32位字列表到目标文件 | head_ptr: .word 0 @ Head pointer to within buffer (initially zero) tail_ptr: .word 0 @ Tail pointer to within buffer (initially zero) .word 0xDEADBEEF @inserts the bytes 0xEF 0xBE 0xAD 0xDE .4byte -42 @ inserts the bytes 0xD6 0xFF 0xFF 0xFF @ Least Significant Byte (LSB) ordering assumed |
.rept <number_of_times> | 重复执行代码块number_of_times次,与C中for 类似,以endr结束 |
|
<register_name> .req <register_name> |
寄存器重新命名 | acc .req r0 |
3. ARM特殊字符和语法
@ | 代码行中的注释符号 |
# | 整行注释符号 |
; | 语句分离符号 |
#或$ | 直接操作数前缀 |
.arm |
汇编使用ARM指令 |
.thumb |
汇编使用Thumb指令 |
.code16 |
汇编使用Thumb指令 |
.code32 |
汇编使用ARM指令 |
.force_thumb |
强制使用thumb模式,即使不支持 |
.thumb_func |
Mark entry point as thumb coded (force bx entry) |
.ltorg |
Start a new literal pool(文字池:嵌入在代码中的用以存放常数的区域) |
0 | 8进制 |
0x或0X | 16进制 |
0b或0B | 二进制 |
无前导符 | 10进制 |
.data | 标识把随后的语句放入目标文件数据段 |
.text |
标识把随后的语句放入目标文件的代码段 |
.extern symbol |
从其它模块引入符号,类似C中的extern |
.skip expression |
在目标文件中skip expression指定的字节数 buffer: .skip 512 @Buffer of 512 bytes, uninitialised |