下面兩個分別是一個foo.asm(匯編語言文件),bar.c(c語言文件)
首先來了解C語言為什麼能調用匯編語言,以及匯編語言為什麼能調用C語言。其實不管是C語言還是匯編語言想要執行都是最終編譯鏈接成為二進制文件。
注意是編譯鏈接這個兩個步驟,編譯產生的並不是可執行的二進制文件,鏈接之後才是可執行的二進制文件。
這裡一定要明確編譯和鏈接是兩個步驟,生成的文件格式也是不一樣的。
編譯生成的文件是一定格式的,裡面包括函數符號表、參數表...等信息,這些信息主要是提供給鏈接階段使用,函數調用是怎麼調用的?是不是指定利用的函數的符號?所以鏈接階段就是將函數調用的符號變成相對地址(要特別注意這個階段,因為這個過程使得C語言和匯編語言相互調用成為可能)。
gcc -m32 -c -o bar.o bar.c 生成了bar.o(是中間文件,不是可執行文件,帶有符號表,參數表等)
nasm -f elf -o foo.o foo.asm 生成foo.o(是中間文件,不是可執行文件,帶有符號表,參數表等)
既然都生成了相同格式的.o中間文件,那麼ld鏈接器就可將兩個中間文件鏈接成為二進制可執行文件了,ld主要做的工作是將bar.o foo.o中的相互引用的函數符號變成在二進制文件foobar中相對地址(相對地址在二進制文件裝入內存運行的時候會被全部轉換為絕對地址)
ld -m elf_i386 -s -o foobar foo.o bar.o (鏈接成為二進制可執行文件了)
過程如圖所示:
foo.asm
extern choose;
[section .data]
num1st dq 3
num2nd dq 4
[section .text]
global main
global myprint
main:
push qword [num2nd]
push qword [num1st]
call choose
add esp,8
mov ebx,0
mov eax,1
int 0x80
; pop qword [num1st]
; pop qword [num2nd]
myprint:
mov edx,[esp+8]
mov ecx,[esp+4]
mov ebx,1
mov eax,4
int 0x80
; pop qword [num1st]
; pop qword [num2nd]
ret
bar.c
void myprint(char * msg ,int len); int choose(int a,int b) { if (a>=b){ myprint(the 1st one ,13);} else { myprint(the 2nd one ,13);} return 0; }