15.1 NASM 简介
NASM 全称 The Netwide Assembler ,是一款基于 x86 平台的汇编语言编译程序,其设计初衷是为了实现编译器程序跨平台和模块化的特性。 NASM 支持大量的文件格式,包括 Linux , BSD , a.out , ELF , COFF , Mach−O , Microsoft 16−bit OBJ , Win32 以及 Win64 ,同时也支持简单的二进制文件生成。它的语法被设计的简单易懂,相较 Intel 的语法更为简单,支持目前已知的所有 x86 架构之上的扩展语法,同时也拥有对宏命令的良好支持。
用 NASM 编写 Linux 下的 hello world 示例程序 hello.nasm 如下:
- GLOBAL _start
- [SECTION .TEXT]
- _start:
- MOV EAX, 4 ; write
- MOV EBX, 1 ; stdout
- MOV ECX, msg
- MOV EDX, len
- INT 0x80 ; write(stdout, msg, len)
- MOV EAX, 1 ; exit
- MOV EBX, 0
- INT 0x80 ; exit(0)
- [SECTION .DATA]
- msg: DB "Hello, world!", 10
- len: EQU $-msg
编译和运行的命令如下( Debian-8.4-amd64 环境下):
- $ nasm -f elf32 -o hello.o hello.nasm
- $ ld -m elf_i386 -o hello hello.o
- $ ./hello
- Hello, world!
Linux 32位可执行程序中,用 “INT 0x80” 指令来执行一个系统调用,用 “EAX” 指定系统调用编号,用 “EBX, ECX, EDX” 来传递系统调用需要的参数。上面这段汇编代码中,首先执行了编号为 4 的系统调用(write),向 stdout 写了一个长为 len 的字符串(msg),之后,执行编号为 1 的系统调用(exit)。
NASM 拥有对宏命令的良好支持,可以简化很多重复代码的编写。对于上面这个程序,可以编写两个名为 print 和 exit 的宏用来重复使用。新建一个 macro.inc 文件,内容如下:
- %MACRO print 1
- [SECTION .DATA]
- %%STRING: DB %1, 10
- %%LEN: EQU $-%%STRING
- [SECTION .TEXT]
- MOV EAX, 4 ; write
- MOV EBX, 1 ; stdout
- MOV ECX, %%STRING
- MOV EDX, %%LEN
- INT 0x80 ; write(stdout, %%STRING, %%LEN)
- %ENDMACRO
- %MACRO exit 1
- MOV EAX, 1
- MOV EBX, %1
- INT 0x80
- %ENDMACRO
- GLOBAL _start
- [SECTION .TEXT]
- _start:
新的 hello.nasm 如下:
- %include "macro.inc"
- print "Hello world!"
- print "Hello again!"
- exit 0
后面这段代码够简洁吧。
上面这段代码中的 %include 命令和 C 语言中的 #inlucde 的作用是一样的,就是把 %include 后面的文件名对应的文件的内容原样的拷贝进来。
下面再来解释一下 NASM 宏的使用。首先看简单一点的 exit 宏。 NASM 中: %MACRO 是宏定义的开始; %MACRO 后面接宏的名称;此处是 “exit” ;宏名后面是宏的参数数量,此处是 “1” ,表示该宏带有一个参数,宏内部中可以用 “%1, %2, %3, …” 来引用宏的第 1 、 2 、 3 、 … 个参数; %ENDMACRO 是宏定义的结束。
宏定义好后,若后面的代码中遇到这个宏,则会用宏定义中的内容来替换这个宏。如 hello.nasm 中的 第 5 行 “exit 0”,会被替换成:
- MOV EAX, 1
- MOV EBX, 0
- INT 0x80
注意宏定义中的 %1 将被替换为 exit 后面的参数 0。
print 宏定义稍微复杂一点,多了 %%STRING 和 %%LEN ,它们可以看成是宏定义中的局部名称,在每个 print 宏被展开的时候, NASM 会为这种类型的名称生成一个唯一的标志符。我们可以用 nasm -e hello.nasm 来查看 hello.nasm 文件经过预处理后的代码,如下(以下代码经过的适当的缩进和注释处理):
- [global _start]
- [SECTION .TEXT]
- _start:
- ; print "Hello world!"
- [SECTION .DATA]
- ..@1.STRING: DB "Hello world!", 10
- ..@1.LEN: EQU $-..@1.STRING
- [SECTION .TEXT]
- MOV EAX, 4
- MOV EBX, 1
- MOV ECX, ..@1.STRING
- MOV EDX, ..@1.LEN
- INT 0x80
- ; print "Hello again"
- [SECTION .DATA]
- ..@2.STRING: DB "Hello again!", 10
- ..@2.LEN: EQU $-..@2.STRING
- [SECTION .TEXT]
- MOV EAX, 4
- MOV EBX, 1
- MOV ECX, ..@2.STRING
- MOV EDX, ..@2.LEN
- INT 0x80
- ; exit 0
- MOV EAX, 1
- MOV EBX, 0
- INT 0x80
可以看到,在 ‘print “Hello world!”’ 宏中, %%STRING 被展开为 ..@1.STRING ,而在 ‘print “Hello again!”’ 宏中, %%STRING 被展开为 ..@2.STRING 。