英文原版地址:NASM Assembly Language Tutorials - asmtutor.com
背景知识
在第一讲成功地学习了执行系统调用之后,我们现在需要学习最重要的系统调用之一 sys_exit。
注意在我们的'Hello World!'程序运行后我们收到了一个段错误。计算机程序可以被当做被加载到内存并分割进不同的段的一长条说明书。通用内存池在所有程序之间共享,用于保存变量,指令,其他程序或其他任何东西。所有段都有一个地址,以便以后在段中找到存储的信息。
执行一个被加载到内存的程序,我们使用全局标识符 _start 来告诉操作系统我们的程序在内存中能被找到和执行的位置。内存随着程序的逻辑决定的下一个访问地址被依次访问。内核跳转到内存对应的地址并执行它。
告诉操作系统开始执行和结束执行准确位置是非常重要的。在第一讲,我们没有告诉内核在哪停止执行。所以,在我们调用sys_write后,程序继续按顺序执行内存中的下一个地址,什么都可能发生。我们不知道内核试图去执行什么,但是它造成了阻塞和留下错误消息而不是终止进程。在所有程序最后调用sys_exit意味着内核准确地知道何时终止进程和将内存返回给通用池,从而避免错误。
写代码
sys_exit 有一个简单的函数定义。在linux系统调用表,它被分配操作码为1以及传递一个简单的参数到EBX中。
为了执行这个函数我们需要这样做
将0存入EBX,传入0到这个函数意思是0错误
将1存入EAX来调用sys_exit
在libc用INT 80h请求中断
使用下边的命令编译,链接然后执行这个程序 hello.asm
- SECTION .data
- msg db 'Hello World!', 0Ah ; 指定msg变量为你的消息字符串
-
- SECTION .text
- global _start
-
- _start:
-
- mov edx, 13 ; 要写入的字节数,字符数加0ah
- mov ecx, msg ; 将消息的内存地址复制ECX寄存器
- mov ebx, 1 ; 写到标准输出
- mov eax, 4 ; 调用 SYS_WRITE (kernel opcode 4)
- int 80h
-
- mov ebx, 0 ; 返回状态值0退出,无错误
- mov eax, 1 ; 调用 SYS_EXIT (kernel opcode 1)
- int 80h
编译命令
nasm -f elf hello.asm
链接命令
ld -m elf_i386 hello.o -o hello
执行命令
./hello