调用惯例
释放参数的方式不一样:
A、由被调用的子函数销毁传进子函数的参数,__stdcall(Also known as __pascal)
B、由父函数释放参数内存 __cdecl ;
c语言中为了适应可变数目参数的函数调用而需要这样的实现;缺点,释放参数的指令重复出现分散到所有调用子函数的的地方
CASE 1:
由被调用的子函数销毁传进子函数的参数
调用处:
push eAX ; pass some register result
push byte[eBP+20] ; pass some memory variable
;eBP+NUM :表示本函数中被传入的参数的位置;
;eBP-NUM :表示 本函数中被传入的参数局部变量的位置
push 3 ; pass some constant
call calc ; the returned result is now in eAX
子函数
calc:
push eBP ; save old frame pointer
mov eBP,eSP ; get new frame pointer sub eSP,localsize ; reserve place for locals .
. ; perform calculations, leave result in eAX
.
mov eSP,eBP ; free space for locals
pop eBP ; restore old frame pointer
ret paramsize ; free parameter space and return
;paramsize为参数占用的空间大小,ret返回后将SP = SP + paramsize
当存在使用ret paramsize时,这种模式是在子函数里面释放参数的内存;很多情况,
往往是是直接使用ret没有参数,该情况下,由父函数释放参数内存
CASE 2:
由父函数释放参数内存
例如:
父函数:
push eAX ; pass some register result
push dword ptr [eBP+20] ; pass some memory variable (FASM/TASM syntax)
call calc ; the returned result is now in eAX
add esp, 8h ;由调用的函数清空参数;因为参数push的时候用了8个空间
子函数:
.....
.....
ret