用匯編語言詮釋Linux的簡單之美

Linux 彙編語言 GCC C語言 EdX Windows 迪吉多 X86 大熊貓侯佩 2019-06-07
用匯編語言詮釋Linux的簡單之美

系統接口

對於底層調用,話說windows也有syscall,這是必須的。但是Windows的syscall可以直接call嗎?

可以是可以,但是破費周折,搞成SDT之類的複雜概念。下面看看linux是如何做的吧。

section .data
msg db "hello hopy!",0x0a
section .text
global _start
_start:
mov eax,4
mov ebx,1
mov ecx,msg
mov edx,12
int 0x80

mov eax,1
mov ebx,0
int 0x80

直接用int陷入!!!

關鍵是系統調用號要知道,開始本貓找的是usr/include/asm-gen*/unistd.h,

可是都不對。後來找的是usr/include/x86_64_linux_gnu/asm/下的頭文件,

有2個,分別對應x86和x64.、編譯連接指令如下:

nasm -f elf main.asm
ld main.o

運行,段轉儲錯誤,查了一下,本貓用的是x64位的linux,所以要生成

x64位的程序,或者指明是x86的程序,我選擇後者:

ld -m elf_i386 -o main main.o

哦鳥!

用匯編語言詮釋Linux的簡單之美

系統接口 VS C標準庫

上面我們看到了如何僅僅用syscall做一些簡單的事,現在我們看能不能直接調用C標準庫中的函數快速做一些"複雜"的事:

section .data
ft db "now is %d",10
section .text
extern puts
extern exit
extern sleep
extern printf
global main
main:
mov edi,11
again:
dec edi
push edi
push ft
call printf

push 1
call sleep

cmp edi,0
jnz again
push msg
call puts

push 0
call exit

msg:
db "happy xxx day!",0

以上代碼功能很簡單,從10倒數到0,然後打印一行,最後結束.與之前代碼不同的是其中

調用了C標準庫中的函數.編譯和以前一樣:

nasm -e elf main.asm

我們看看怎麼連接:

gcc -m32 -o main main.o

好了!運行正常!

用匯編語言詮釋Linux的簡單之美

值得注意的是:我的OS是ubuntu64,而asm代碼中是32位的,如果開始用

ld -m elf_i386 -lc -o main main.o

的方式,首先會提示找不到c庫,這可以進入/usr/lib,然後使用

sudo ln -sv /lib/i386-linux-gun/libc.so.6 libc.so

創建軟連接解決.

但在運行時提示無法找到可執行文件!該文件明明在的呀?

我們換為用gcc來連接,但要將_start改為main,還要裝載32庫

sudo apt-get install ia32-libs

還會提示找不到h文件,這時再裝載庫

sudo apt-get install g++-multilib

還有2族庫,如有必要也可加載:

sudo apt-get install libc6:i386 libgcc1:i386 gcc-4.6-base:i386 
libstdc++5:i386 libstdc++6:i386
sudo apt-get install libc6-i386

最後要說的是,一些C代碼在用std=c99編譯時會發現提示無法獲取結構大小,

這時改成如下即可:

gcc -D_GNU_SOURCE -std=c99 main.c

回到系統調用

之前我們嘗試使用了C庫的函數完成功能,那麼能不能用syscall方式來搞呢?

用匯編語言詮釋Linux的簡單之美

顯然可以!

section .data
ft db "now is X",10
section .text
global _start
_start:
mov edi,10
again:
dec edi
mov eax,edi
add eax,0x30
mov byte [ft+7],al
mov eax,4
mov ebx,1
mov ecx,ft
mov edx,9
int 0x80
mov eax,162
push 0
push 1
mov ebx,esp
mov ecx,0
int 0x80

cmp edi,0
jnz again

mov eax,4
mov ebx,1
mov ecx,msg
mov edx,15
int 0x80

mov eax,1
mov ebx,0
int 0x80

msg:
db "happy xxx day!",10
nasm -f elf main.asm
ld -m elf_i386 -o main main.o

在代碼中延時使用的是nanosleep,其他和第一篇一致,只不過做了一個bin->ascii的小轉換.

總結

我們通過以上幾個例子可以看到在Linux下調用系統底層接口是多麼的輕而易舉,如果不信,請逆向Windows中相同功能的代碼,你會覺得在Linux下開發簡直太幸福了!!!

用匯編語言詮釋Linux的簡單之美

感謝觀賞 ;)

相關推薦

推薦中...