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
对源程序进行汇编、链接,得到可执行程序task1.exe,运行后,结合运行结果和注释,及必要的debug
调试:
1. 理解运算符offset、伪指令equ、预定义符号$的灵活使用。
通过line5、line8,以及数据项的数据属性(字节、字、双字,等),可以方便计算出连续数据项
的个数,而无需人工计数。
2. 回答问题
① line27, 汇编指令 loop s1 跳转时,是根据位移量跳转的。通过debug反汇编,查看其机器码,
分析其跳转的位移量是多少?(位移量数值以十进制数值回答)从CPU的角度,说明是如何计算得
到跳转后标号s1其后指令的偏移地址的。
答:跳转位移量是-14个字节。
答:cpu将指令E2F2进入指令缓冲器,指令告诉了cpu要向前移动14个字节。
② line44,汇编指令 loop s2 跳转时,是根据位移量跳转的。通过debug反汇编,查看其机器码,
分析其跳转的位移量是多少?(位移量数值以十进制数值回答)从CPU的角度,说明是如何计算得
到跳转后标号s2其后指令的偏移地址的。
答:跳转位移量是-16个字节。
答:cpu将指令E2F0进入指令缓冲器,指令告诉了cpu要向前移动16个字节。
③ 附上上述分析时,在debug中进行调试观察的反汇编截图
Task1.asm
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
① 根据call指令的跳转原理,先从理论上分析,程序执行到退出(line31)之前,寄存器(ax) = 0021寄存器
(bx) = 0026寄存器(cx) = 076C
② 对源程序进行汇编、链接,得到可执行程序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, 实现题目要求。
正确编写后,预期测试结果如下:
3. 实验任务3
源码
assume ds:data, cs:code
data segment
x db 99, 72, 85, 63, 89, 97, 55
len equ $- x
data ends
code segmentstart:
mov ax, data
mov ds, ax
mov si, offset x
mov cx, len
mov byte ptr ds:[10], 10
s1: mov ah, 0
mov al, ds:[si]
div byte ptr ds:[10]
call printNumber
call printSpace
inc si
loop s1
mov ax, 4c00h
int 21h
printNumber:mov bx, ax
or bl, 30h
or bh, 30h
mov ah, 2
mov dl, bl
int 21h
mov dl, bh
int 21h
ret
printSpace:mov ah, 2
mov dl, ' '
int 21h
ret
code ends
end start
截图
4. 实验任务4
源码
assume ds:data, cs:code
data segment
str db 'try'
len equ $ - str
data ends
code segmentstart:
mov ax, data
mov ds, ax
mov cx, len
mov si, offset str
mov ax, 0b800h
mov es, ax
mov di, 0
mov ah, 2
s: call printStr
inc si
add di, 2
loop s
mov di, 3840
mov si, offset str
mov cx, len
mov ah, 4
s1: call printStr
inc si
add di, 2
loop s1
mov ax, 4c00h
int 21h
printStr:mov al, [si]
mov es:[di], al
mov es:[di+1], ah
ret
code ends
end start
截图
5. 实验任务5
源码assume ds:data, cs:code
data segment
stu_no db '201983290455'
len = $ - stu_no
data ends
code segmentstart:
mov ax, data
mov ds, ax
mov cx, 4000
mov si, offset stu_no
mov ax, 0b800h
mov es, ax
mov di, 0
mov ah,17h
s: mov al, 0
mov es:[di], al
mov es:[di+1], ah
inc si
add di, 2
loop s
mov di, 3840
mov si, offset stu_no
mov cx, 74
mov ah, 17hs1: call printgang
add di, 2
loop s1
mov di, 3908
mov si, offset stu_no
mov cx, len
mov ah, 17hs2: call printStu_no
inc si
add di, 2
loop s2
mov di, 3932
mov si, offset stu_no
mov cx, 74
mov ah, 17hs3: call printgang
add di, 2
loop s3
mov ax, 4c00h
int 21h
printStu_no:mov al, [si]
mov es:[di], al
mov es:[di+1], ah
ret
printgang:mov al, 45
mov es:[di], al
mov es:[di + 1], ah
ret
code ends
end start
截图