通过堆栈传递8位和16位参数:
在保护模式下传递堆栈参数时,最好使用32位的操作数,虽然可以在栈中压入16位的操作数, 但是这样做会使ESP无法对齐在双字地址边界上,有此可能导致页故障,程序的性能也可能降低。 因此在传递8位或16位堆栈参数时,利用指令movzx把它扩展成32位模式后再压入。例如:
var1 db x
var2 dw y
...
movzx eax,var1
push eax
movzx eax,var2
push eax
call ...
传递长整数参数:
在使用堆栈向过程传递长整数(由多个字节构成)参数时, 可以先把高位字节压入, 然后再把低位字节 压入,这样实际上是以小尾顺序(低地址在低地址高地址在高地址)把场整数入栈了。例如:
longvar dq 1234567890abcdefh
...
push dword ptr longvar+4
push dword ptr longvar
call ...
保存和恢复寄存器:
子例程通常在修改寄存器之前保存其原来的值, 以便在过程返回之前进行恢复。 理想的情况下,要保存的寄存器应在EBP设为ESP的值后,为局部变量保留空间之前压入堆栈, 这有助于避免改变堆栈参数的相对偏移。例如:
subproc proc
push ebp ;保存基址指针
mov ebp,esp ;堆栈基址
push ecx ;压入ecx和edx
push edx
mov eax,[ebp+8] ;获取堆栈参数
... ;过程操作
pop edx ;恢复已保存的寄存器
pop ecx ;ecx和edx
pop ebp ;恢复基址指针
ret ;返回
subproc endp
EBP在初始化以后, 在整个过程中其值保持不变,ecx和edx的入栈不影响栈中的参数相对于EBP的偏移。这是因为,堆栈是向下增长的, 见图:
-----------------
| 堆栈参数 | [EBP+8]
-----------------
| 返回地址 | [EBP+4]
-----------------
| EBP | EBP
-----------------
| ECX |
-----------------
| EDX | <--ESP,SS:ESP时刻指向栈顶
-----------------
USES对堆栈的影响:
uses操作符后跟要在过程开始时保存并在过程结束时恢复的寄存器列表。对于uses后的每个寄存器, masm自动生成合适的push和pop指令。
注意: 使用显式堆栈参数的过程应避免使用uses操作符。这是因为由于masm在过程的开始插入了push指令, 因此改变了堆栈参数对于EBP的偏移, 进而导致发生错误哦。
- [somniumchase] 我一运行就说没有数字 01/01 11:44
- [游客] 为什么啊 08/07 15:36
- [游客] 如果想快一些 就改下面这里 dx值改成1H delay: push ax 04/19 02:53
- [lshhjx] 注释在程序中很重要,楼主不知道吗? 12/08 13:40
- [biaggi] 看不明白,在下還須學習 11/06 08:11
- [游客] 我运行的时候直接显示Unkown filename跳出了- -请问怎么改 06/16 21:44
- [游客] 勿庸置疑,注释是好习惯。与人方便自己方便。 04/12 10:33
- [游客] 老实说,看着真心累呀! 04/07 18:37
- [游客] 很无语,初学者就多看书,不要动不动要别人注释,基础打好了,再自己注释,这样比别人帮你注释好得多 12/17 19:43
- [dgkepu] 初学者:不懂,希望有多点注释带着学习学习! 12/07 20:52
- [游客] windows 7是一个64Bit操作系统,它不兼容DOS,无法识别16Bit系统。重装系统wind 02/28 21:05
- [游客] windows 7是一个64Bit操作系统,它不兼容DOS,无法识别16Bit系统。重装系统wind 02/28 21:05
- [466987333] 你好,高手,我想请教一个问题。 我用的是win7操作系统,32位的,里面没有找masm目录,是不是 12/12 17:30
- [lanfioncc] 那个太高级了。。。我还有点看不懂。。不过谢谢!!! 11/27 11:23
- [yc2010] 实验16中的 table: dw sub1,sub2,sub3,sub4 可不可以改成呢? 09/11 09:08
- [yc2010] mov bl,ah mov bh,0 add bx,bx ----------->这里为 09/07 21:03
- [yc2010] 为什么要add bx,bx呢? 09/07 20:55
- [yc2010] 那是不是像table[bx],ds[bx]....等(内存单元)都是表示一个字节呢? 09/06 21:10
- [masmaster] 杨季文的《80X86汇编语言程序设计教程》 09/01 12:52
- [游客] to masmaster shl左移4位,那al传进来的4,5,6位背景色不就没了. 为什 09/01 11:00