实验3 转移指令跳转原理及其简单应用编程 201983290153吕轩皓
一、实验目的
1. 理解和掌握转移指令的跳转原理
2. 掌握使用call和ret指令实现子程序的方法,理解和掌握其参数传递方式
3. 理解和掌握80×25彩色字符模式显示原理
4. 综合应用寻址方式和汇编指令完成简单应用编程
二、实验准备
复习教材9-10章:
转移指令的跳转原理
汇编指令jmp, loop, jcxz, call, ret, retf的用法
三、实验内容
1. 实验任务1
使用任何一款文本编辑器,录入8086汇编程序源码task1.asm。
task1.asm
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
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
v>
对源程序进行汇编、链接,得到可执行程序task1.exe,运行后,结合运行结果和注释,及必要的debug
调试:
2. 回答问题
① line27, 汇编指令 loop s1 跳转时,是根据位移量跳转的。通过debug反汇编,查看其机器码,
分析其跳转的位移量是多少?(位移量数值以十进制数值回答)从CPU的角度,说明是如何计算得
到跳转后标号s1其后指令的偏移地址的。
答:跳转位移量是14,loop命令的机器码为E2F2,F2八位二进制形式为11110010,补码为10001110,补码二进制数即为跳转位移值
② line44,汇编指令 loop s2 跳转时,是根据位移量跳转的。通过debug反汇编,查看其机器码,
分析其跳转的位移量是多少?(位移量数值以十进制数值回答)从CPU的角度,说明是如何计算得
到跳转后标号s2其后指令的偏移地址的。
答:跳转位移量是16,loop命令的机器码为E2F0,F0八位二进制形式为11110000,补码为10010000,补码二进制数即为跳转位移值
③ 附上上述分析时,在debug中进行调试观察的反汇编截图
已附.
2. 实验任务2
使用任何一款文本编辑器,录入8086汇编程序源码task2.asm。
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
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
② 对源程序进行汇编、链接,得到可执行程序task2.exe。使用debug调试,观察、验证调试结果与理论
分析结果是否一致。
答: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 cx,len
mov si,0
s: mov ah,0
mov al,[si]
mov bx,offset printnumber
call bx
mov bx,offset printSpace
call bx
inc si
loop s
mov ah, 4ch
int 21h
printnumber:
mov bl,10
div bl
mov bx,ax
mov ah,2
mov dl,bl
or dl,30h
int 21h
mov dl,bh
or dl,30h
int 21h
ret
printSpace:
mov ah,2
mov dl,' '
int 21h
ret
code ends
end start
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 si,0
s: mov ah,0
mov al,[si]
mov bx,offset printnumber
call bx
mov bx,offset printSpace
call bx
inc si
loop s
mov ah, 4ch
int 21h
printnumber:
mov bl,10
div bl
mov bx,ax
mov ah,2
mov dl,bl
or dl,30h
int 21h
mov dl,bh
or dl,30h
int 21h
ret
printSpace:
mov ah,2
mov dl,' '
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)
出口参数:无
在主体代码中,两次调用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 ax,0B800H
mov es,ax
mov si,offset printStr
mov ah,2
mov bx,0
call si
mov si,offset printStr
mov ah,4
mov bx,0F00H
call si
mov ah, 4ch
int 21h
printStr:
mov cx,len
mov si,0
s: mov al,[si]
mov es:[bx+si],ax
inc si
inc bx
loop s
ret
code ends
end start
data segment
str db 'try'
len equ $ - str
data ends
code segment
start:
mov ax, data
mov ds, ax
mov ax,0B800H
mov es,ax
mov si,offset printStr
mov ah,2
mov bx,0
call si
mov si,offset printStr
mov ah,4
mov bx,0F00H
call si
mov ah, 4ch
int 21h
printStr:
mov cx,len
mov si,0
s: mov al,[si]
mov es:[bx+si],ax
inc si
inc bx
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 '201983290153'
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
效果演示:data segment
stu_no db '201983290153'
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