1. 预备知识
- 汇编语言程序从写出到最终执行的简要过程分为三步:编写汇编源程序,对源程序进行编译连接,执行可执行文件中的程序。
- 汇编语言程序中,包含汇编指令和伪指令两种,伪指令包括 assume、xx segment 和 xx ends、end。assume 指令将代码段和代码寄存器联系起来,xx segment 和 xx ends 标识了汇编指令的开始和结束,end 指令标识了整个程序的结束。特别地,在汇编指令部分使用 mov ax,4c00H 和 int 21h 作为程序返回指令。
- masm *.asm 可编译源程序,并选择是否生成中间文件,也可以后接分号跳过选择过程;link * 可连接目标文件,并选择是否生成中间文件,也可以后接分号直接跳过选择过程。其中,星号表示文件名。
- 可输入文件名 * 或使用 debug *.exe 以调试模式运行可执行文件。
- 汇编指令使用 t 执行,但 int 21h 语句使用指令 p 完成,正常运行完成后,界面显示 Program terminated normally。
2. 实验任务
(1)根据下面程序生成可执行文件,并指出相关寄存器的内容:
assume cs:codesg
codesg segment
mov ax,2000H
mov ss,ax
mov sp,0
add sp,4
pop ax
pop bx
push ax
push bx
pop ax
pop bx
mov ax,4c00H
int 21h
codesg ends
end
使用 Debug 来跟踪程序执行过程中寄存器的内容:
指令 | AX | BX | SS | SP |
---|---|---|---|---|
mov ax,2000 | 2000 | - | - | - |
mov ss,ax | 2000 | - | 2000 | - |
mov sp,0 | 2000 | - | 2000 | 0000 |
add sp,4 | 2000 | - | 2000 | 0004 |
pop ax | 0000 | - | 2000 | 0006 |
pop bx | 0000 | 0000 | 2000 | 0008 |
push ax | 0000 | 0000 | 2000 | 0006 |
push bx | 0000 | 0000 | 2000 | 0004 |
pop ax | 0000 | 0000 | 2000 | 0006 |
pop bx | 0000 | 0000 | 2000 | 0008 |
mov ax,4c20 | 4C20 | 0000 | 2000 | 0008 |
int 21h | 4C20 | 0000 | 2000 | 0008 |
说明:
- 第 3 行完成栈的初始化,空栈时:SS=2000、SP=0000,入栈时 SP 值减小、出栈时 SP 值增大。
- 第 4 行,栈顶指向 2000:0004 的内存单元,即 2000:3 4。
- 第 5、6 行后,AX=0000、BX=0000,并且每出栈一个元素,SP 值加 2。
- 第 7~10 行完成入栈出栈操作,由图可知这一内存单元的值全部为零。
- 第 11、12 行设置程序返回。
- 在整个过程中,栈顶 SS:SP 指向的值始终为零。
(2)查看 PSP 的内容。
程序加载后,寄存器 DS 中存放着程序所在内存区的段地址,这个内存区的偏移地址为零,所以程序所在内存区的起始地址为 ds:0;这个内存区得前 256 个字节存放得是 PSP,是 DOS 用来和程序进行通信的相关数据。从 256 字节处往后存放的是程序,起始地址为 ds×16:0。
使用 debug 运行上述程序,由图可知 DS 寄存器的值是 075A,然后使用 d 指令查看以 075A:0 为起始地址的前 256 个字节的内容:
由图可知,PSP(Program Segment Prefix,程序段前缀) 中包含了可执行文件名。