调用惯例 释放参数的方式不一样: 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 子函数 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 |
|