1. 实验任务1
使用任何一款文本编辑器,录入8086汇编程序源码task1.asm。
task1.asm
assume cs:code, ds:data data segment x db 1, 9, 3 len1 equ $ - x ; 符号常量, $指下一个数据项的偏移地址,这个示例中,是3 y dw 1, 9, 3 len2 equ $ - y ; 符号常量, $指下一个数据项的偏移地址,这个示例中,是9 data ends code segment start: mov ax, data mov ds, ax mov si, offset x ; 取符号x对应的偏移地址0 -> si mov cx, len1 ; 从符号x开始的连续字节数据项个数 -> cx mov ah, 2 s1:mov dl, [si] or dl, 30h int 21h mov dl, ' ' int 21h ; 输出空格 inc si loop s1 mov ah, 2 mov dl, 0ah int 21h ; 换行 mov si, offset y ; 取符号y对应的偏移地址3 -> si mov cx, len2/2 ; 从符号y开始的连续字数据项个数 -> cx mov ah, 2 s2:mov dx, [si] or dl, 30h int 21h mov dl, ' ' int 21h ; 输出空格 add si, 2 loop s2 mov ah, 4ch int 21h code ends end start
运行截图:
offset的作用是取得标号的偏移地址,通过offset x可以取得x的起始地址,也就是0.
len1 equ相当于定义一个常量,常量名为equ,对应偏移地址为$-x,即x后面一个数据项的偏移量,本题为3.
回答问题:
① line27, 汇编指令 loop s1 跳转时,是根据位移量跳转的。通过debug反汇编,查看其机器码,分析其跳转的位移量是多少?
(位移量数值以十进制数值回答)从CPU的角度,说明是如何计算得到跳转后标号s1其后指令的偏移地址的。
答:使用u命令反汇编,机器码为E2F2,F2转换成原码后对应的十进制数值为-14,故位移量为-14.
可以发现CPU首先取得目前正在执行的指令LOOP的地址加上本条指令占用的地址,即001B,减去位移量14,
001B-E= 000D,跳转至IP为000D处。
② line44,汇编指令 loop s2 跳转时,是根据位移量跳转的。通过debug反汇编,查看其机器码,分析其跳转的位移量是多少?
(位移量数值以十进制数值回答)从CPU的角度,说明 是如何计算得到跳转后标号s2其后指令的偏移地址的。
答:机器码为E2F0, F0对应的十进制值为-16,故跳转的位移量为-16。CPU跳转过程如下:
CPU首先取得目前正在执行的指令LOOP的地址0037指令,加上本条指令占用的地址2,变为0039,
然后减去标号的地址进行跳转,0039+(-0010)=0029,即跳转至IP为0029处。
③ 附上上述分析时,在debug中进行调试观察的反汇编截图
反汇编截图见上面两小题中。
2.实验任务2
task2.asm源码:
assume cs:code, ds:data data segment dw 200h, 0h, 230h, 0h data ends stack segment db 16 dup(0) stack ends code segment start: mov ax, data mov ds, ax mov word ptr ds:[0], offset s1 mov word ptr ds:[2], offset s2 mov ds:[4], cs mov ax, stack mov ss, ax mov sp, 16 call word ptr ds:[0] s1: pop ax call dword ptr ds:[2] s2: pop bx pop cx mov ah, 4ch int 21h code ends end start① 根据call指令的跳转原理,先从理论上分析,程序执行到退出(line31)之前,寄存器(ax) = ? 寄存器 (bx) = ? 寄存器(cx) = ? 答:ax寄存器存放的是s1的偏移地址,bx寄存器存放的是s2的偏移地址;cx寄存器存放的是cs的段地址。 执行源程序line24 call word ptr ds:[0]时,下一条指令s1:pop ax的IP值入栈,接着跳转执行s1:pop ax, 栈顶的IP值(即s1的偏移地址)赋给ax。 执行源程序line27 call dword ptr ds:[2]时,代码段段地址cs和下一条指令的IP值相继入栈,接着跳转 到执行s2段的代码pop bx以及pop cx,即将栈顶的IP(S2偏移地址)出栈赋给bx,再将段地址cs出栈赋给cx。 ② 对源程序进行汇编、链接,得到可执行程序task2.exe。使用debug调试,观察、验证调试结果与理论 分析结果是否一致。
由上面几张图可知,执行完call word ptr ds:[0]的IP值0021,在执行完s1:pop ax后被赋给了ax寄存器。
执行完call dword ptr ds:[2]的CS=076C和IP=0026入栈,执行s2段后,bx = 0026,cx=076C。
调试结果与理论分析结果一致。
3. 实验任务3 编写8086汇编源程序task3.asm,在屏幕上以十进制形式输出data段中这一组连续的数据,数据和数据 之间以空格间隔。 task3.asm代码
assume cs:code,ds:data data segment x db 99, 72, 85, 63, 89, 97, 55 len equ $- x data ends code segment start: mov ax,data mov ds,ax mov cx,len mov bx,0 s: mov al,[di] mov ah,0 call printNumber call printSpace inc di loop s mov ax,4c00h int 21h printNumber: mov bl,10 div bl mov dl,al or dl,30h mov bh,ah mov ah,2 int 21h mov dl,bh or dl,30h int 21h ret printSpace: mov dl,' ' mov ah,2 int 21h ret code ends end start截图:
4. 实验任务4
task4.asm代码assume ds:data,cs:code data segment str db 'try' len equ $ - str data ends code segment start: mov ax,data mov ds,ax mov cx,len mov si,0 mov bl,2h mov bh,0 call printStr mov cx,len mov si,0 mov bl,4h mov bh,24 call printStr mov ah,4ch int 21h printStr: mov ax,0b800h mov es,ax mov ax,00a0h mul bh mov di,ax s: mov al,[si] mov es:[di],al inc si inc di mov es:[di],bl inc di loop s ret code ends end start截图:
5. 实验任务5
task5.asm代码
assume cs:code,ds:data data segment stu_no db '201983290116' len = $ - stu_no data ends code segment start: mov ax,data mov ds,ax mov bl,16 call backGround mov bh,24 mov bl,23 mov si,0 call font mov ah,4ch int 21h backGround: mov ax,0b800h mov es,ax mov cx,25*80 mov al,' ' mov di,0 s: mov es:[di],al inc di mov es:[di],bl inc di loop s ret font: mov ax,0b800h mov es,ax mov ax,00a0h mul bh mov di,ax mov cx,(80-len)/2 s1: mov al,'-' mov es:[di],al inc di mov es:[di],bl inc di loop s1 mov cx,len s2: mov al,[si] mov es:[di],al inc si inc di mov es:[di],bl inc di loop s2 mov cx,(80-len)/2 s3: mov al,'-' mov es:[di],al inc di mov es:[di],bl inc di loop s3 ret code ends end start
截图: