- 相關(guān)推薦
C函數(shù)的調(diào)用過程
在C代碼中通過asm或__asm__嵌入一些匯編代碼,如進(jìn)行系統(tǒng)調(diào)用,使用寄存器以提高性能能,需要對函數(shù)調(diào)用過程中的堆棧幀(Stack Frame)、CPU寄存器、GCC inlie assembly等了如指掌。現(xiàn)在看看函數(shù)調(diào)用過程吧。
C函數(shù)的調(diào)用過程
1. Linux 進(jìn)程虛擬地址空間
以32位操作系統(tǒng)為例,下面是Linux進(jìn)程地址空間布局:
32位虛擬地址空間的高1GB的空間是留給操作系統(tǒng)內(nèi)核的,棧由高地址到低地址向下增長,堆由低地址到高地址向上增長。
C中如 malloc 等分配的內(nèi)存在堆中分配。初始化了的靜態(tài)變量和全局變量放在Data段中。未初始化的全局變量和局部靜態(tài)變量放在Bss段中,更準(zhǔn)確的說是在Bss段為它們預(yù)留了空間。非靜態(tài)局部變量是在函數(shù)調(diào)用過程中暫存在棧上的。
2. 函數(shù)的堆棧幀
棧在程序運(yùn)行中具有舉足輕重的地位。最重要的,棧保存了一個函數(shù)調(diào)用所需要的維護(hù)信息,被稱為堆棧幀(Stack Frame),一個函數(shù)(被調(diào)函數(shù))的堆棧幀一般包括下面幾個方面的內(nèi)容:
。1) 函數(shù)參數(shù),默認(rèn)調(diào)用慣例情況下從右向左的順序依次把參數(shù)壓入棧中。由函數(shù)調(diào)用方執(zhí)行。
。2) 函數(shù)的返回地址,即調(diào)用方調(diào)用此函數(shù)(如call func1)的下一條指令的地址。函數(shù)調(diào)用方(call指令)執(zhí)行。
。3) 保存調(diào)用方函數(shù)的'EBP寄存器,即將調(diào)用方函數(shù)的EBP壓入堆棧,并令EBP指向此棧中的地址:pushl %ebp; movl %esp, %ebp。由被調(diào)函數(shù)執(zhí)行。
(4) 上下文:保存在函數(shù)調(diào)用過程中需要保持不變的寄存器(函數(shù)調(diào)用方的),如ebx,esi,edi等。由被調(diào)函數(shù)執(zhí)行。
。5) 臨時變量,如非靜態(tài)局部變量。
下面是一個函數(shù)的堆棧幀結(jié)構(gòu)圖:
壓入函數(shù)參數(shù)和返回地址的過程是由函數(shù)調(diào)用方在調(diào)用函數(shù)之前將其壓入棧中,每個函數(shù)執(zhí)行后首先要執(zhí)行的就是把函數(shù)調(diào)用方的EBP寄存器壓入棧中,之后是在棧上開辟一些空間存放局部變量,最后把要保存的寄存器壓入棧中。
【C函數(shù)的調(diào)用過程】相關(guān)文章:
C語言函數(shù)的運(yùn)用及調(diào)用10-09
java調(diào)用c函數(shù)的實(shí)例09-16
C++如何調(diào)用matlab函數(shù)06-29
C語言函數(shù)調(diào)用與參數(shù)傳遞08-05