NASM 简介

软件开发大郭
0 评论
/
19 阅读
/
3848 字
09 2023-02

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 。

    暂无数据