C語言與匯編語言相互調用原理以及實例
來源:程序員人生 發布時間:2015-04-13 08:54:18 閱讀次數:4424次
下面兩個分別是1個foo.asm(匯編語言文件),bar.c(c語言文件)
首先來了解C語言為何能調用匯編語言,和匯編語言為何能調用C語言。其實不論是C語言還是匯編語言想要履行都是終究編譯鏈接成為2進制文件。
注意是編譯鏈接這個兩個步驟,編譯產生的其實不是可履行的2進制文件,鏈接以后才是可履行的2進制文件。
這里1定要明確編譯和鏈接是兩個步驟,生成的文件格式也是不1樣的。
編譯生成的文件是1定格式的,里面包括函數符號表、參數表...等信息,這些信息主要是提供給鏈接階段使用,函數調用是怎樣調用的?是否是指定利用的函數的符號?所以鏈接階段就是將函數調用的符號變成相對地址(要特別注意這個階段,由于這個進程使得C語言和匯編語言相互調用成為可能)。
gcc -m32 -c -o bar.o bar.c 生成了bar.o(是中間文件,不是可履行文件,帶有符號表,參數表等)
nasm -f elf -o foo.o foo.asm 生成foo.o(是中間文件,不是可履行文件,帶有符號表,參數表等)
既然都生成了相同格式的.o中間文件,那末ld鏈接器便可將兩個中間文件鏈接成為2進制可履行文件了,ld主要做的工作是將bar.o foo.o中的相互援用的函數符號變成在2進制文件foobar中相對地址(相對地址在2進制文件裝入內存運行的時候會被全部轉換為絕對地址)
ld -m elf_i386 -s -o foobar foo.o bar.o (鏈接成為2進制可履行文件了)
進程如圖所示:
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;
}
生活不易,碼農辛苦
如果您覺得本網站對您的學習有所幫助,可以手機掃描二維碼進行捐贈