实验3 转移指令跳转原理及其简单应用编程
1.实验任务1
使用任何一款文本编辑器,录入8086汇编程序源码
assume cs:code, ds:data data segment x db 1, 9, 3 len1 equ $ - x y dw 1, 9, 3 len2 equ $ - y data ends code segment start: mov ax, data mov ds, ax mov si, offset x mov cx, len1 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 mov cx, len2/2 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
对源程序进行汇编、链接,得到可执行程序task1.exe,运行后,结合运行结果和注释,及必要的debug
1. 回答问题
① line27, 汇编指令 loop s1 跳转时,是根据位移量跳转的。通过debug反汇编,查看其机器码,分析其跳转的位移量是多少?
(位移量数值以十进制数值回答)从CPU的角度,说明是如何计算得到跳转后标号s1其后指令的偏移地址的。
由图可知,loop s1的机器码为E2F2(补码表示),其位移量为标号处的地址-loop指令后的第一个字节的地址,即000D-001B=-000E,用十进制表示为-14
CPU读取指令码E2F2并将其存入指令缓冲器,IP=IP+读取指令长度,IP=001B,指向下一条指令,CPU执行指令E2F2,IP=000D,CS:IP指向s1.
② line44,汇编指令 loop s2 跳转时,是根据位移量跳转的。通过debug反汇编,查看其机器码,分析其跳转的位移量是多少?
(位移量数值以十进制数值回答)从CPU的角度,说明是如何计算得到跳转后标号s2其后指令的偏移地址的。
由图可知,loop s2指令的机器码为E2F0(补码表示),其位移量为标号处的地址-loop指令后的第一个字节的地址,即0029-0039=-0010,用十进制表示为-16
CPU读取指令码E2F0并将其存入指令缓冲器,IP=IP+读取指令长度,IP=0039,指向下一条指令,CPU执行指令E2F0,IP=0029,CS:IP指向s2.
2.实验任务2
使用任何一款文本编辑器,录入8086汇编程序源码
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) =0021 寄存器(bx) = 0026 寄存器(cx) = 076C
mov word ptr ds:[0], offset s1 的意思是 把s1的偏移地址赋给ds:[0],同理ds:[2]是s2的偏移地址,ds:[4]是cs的偏移地址,而call命令是将当前的IP
压栈后,转到标号处执行指令,因而得出上述结果。
② 对源程序进行汇编、链接,得到可执行程序task2.exe。使用debug调试,观察、验证调试结果与理论分析结果是否一致。
与理论分析结果一致
3.实验任务3
针对8086CPU,已知逻辑段定义如下:
data segment x db 99, 72, 85, 63, 89, 97, 55 len equ $- x data ends
编写8086汇编源程序task3.asm,在屏幕上以十进制形式输出data段中这一组连续的数据,数据和数据之间以空格间隔。
要求:
编写子程序printNumber
功能:以十进制形式输出一个两位数
入口参数:寄存器ax(待输出的数据 --> ax)
出口参数:无
编写子程序printSpace
功能:打印一个空格
入口参数:无
出口参数:无
在主体代码中,综合应用寻址方式和循环,调用printNumber和printSpace, 实现题目要求。
代码
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 si, offset x mov cx, len s: mov ah, 0 mov al, [si] call printNumber call printSpace inc si loop s mov ah, 4ch int 21h printNumber: mov bx, 10 div bl mov bx, ax add bx, 3030h mov ah, 2 mov dl, bl int 21h mov ah, 2 mov dl, bh int 21h ret printSpace: mov ah, 2 mov dl, 20h int 21h ret code ends end start
4.实验任务4
针对8086CPU,已知逻辑段定义如下:
data segment str db 'try' len equ $ - str data ends
编写8086汇编源程序task4.asm,在屏幕上以指定颜色、指定行,在屏幕上输出字符串。
要求:
编写子程序printStr
功能:在指定行、以指定颜色,在屏幕上显示字符串
入口参数
字符串首字符地址 --> ds:si(其中,字符串所在段的段地址—> ds, 字符串起始地址的偏移地址—> si)
字符串长度 --> cx
字符串颜色 --> bl
指定行 --> bh (取值:0 ~24)
出口参数:无
; 功能:输出单个字符 mov ah, 2 mov dl, ×× ; ××是待输出的字符,或其ASCⅡ码值 int 21h 12345 data segment str db 'try' len equ $ - str data ends 1234
在主体代码中,两次调用printStr,使得在屏幕最上方以黑底绿字显示字符串,在屏幕最下方以黑底红色显示字符串
代码
assume cs:code, ds:data data segment str db 'try' len equ $ - str data ends code segment start: mov ax, data mov ds, ax mov si, offset str mov cx, len mov bl, 0ah mov bh, 0 call printStr mov cx, len mov si, offset str mov bl, 0ch mov bh, 24 call printStr mov ah, 4ch int 21h printStr: mov al, 00a0h mul bh mov di, ax mov ax, 0b800h mov es, ax mov al, bl s: mov ah, ds:[si] mov es:[di], ah; mov es:[di+1], bl; add di, 2 inc si loop s ret code ends end start
实验结果
5.实验任务5
针对8086CPU,针对8086CPU,已知逻辑段定义如下:
data segment stu_no db '20498329042' len = $ - stu_no data ends
在80×25彩色字符模式下,在屏幕最后一行正中间显示学号。要求输出窗口蓝底,学号和两侧折线,以白色前景色显示。
注*:
1. 80×25彩色字符模式显示缓冲区结构,参见教材「实验9 根据材料编程」里的说明。
2. 编写程序实现时,将data段的学号换成自己的学号。
代码
assume cs:code, ds:data data segment stu_no db '201983290243' len = $ - stu_no data ends code segment start: mov ax, data mov ds, ax mov ax,0B800H mov es,ax mov cx,0780H mov ah,10H mov al,' ' mov bx,0 s: mov es:[bx],ax add bx,2 loop s mov cx,80 mov ah,17H mov al,'-' s1: mov es:[bx],ax add bx,2 loop s1 mov cx,len mov bx,0F44H mov si,0 s2: mov al,[si] mov es:[bx],ax inc si add bx,2 loop s2 mov ah, 4ch int 21h code ends end start
实验结果