实验2 多个逻辑段的汇编源程序编写与调试
实验任务1
此实验任务中,包含4个子任务。逐一实践,结合实践观察、验证,回答问题。
任务1-1
点击查看task1_1.asm
assume ds:data, cs:code, ss:stack
data segment
db 16 dup(0)
data ends
stack segment
db 16 dup(0)
stack ends
code segment
start:
mov ax, data
mov ds, ax
mov ax, stack
mov ss, ax
mov sp, 16
mov ah, 4ch
int 21h
code ends
end start
- 问题一
在debug中将执行到line17结束、line19之前,记录此时:寄存器(DS)是__770__, 寄存器(SS)是__771__寄存器(CS)是__772__。 - 问题二
假设程序加载后,code段的段地址是X,则,data段的段地址是__X-2__stack的段地址是__X-1__。
任务1-2
点击查看task1_2.asm
assume ds:data, cs:code, ss:stack
data segment
db 4 dup(0)
data ends
stack segment
db 8 dup(0)
stack ends
code segment
start:
mov ax, data
mov ds, ax
mov ax, stack
mov ss, ax
mov sp, 8
mov ah, 4ch
int 21h
code ends
end start
- 问题一
在debug中将执行到line17结束、line19之前,记录此时:寄存器(DS)是__770__, 寄存器(SS)是__771__寄存器(CS)是__772__。 - 问题二
假设程序加载后,code段的段地址是X,则,data段的段地址是__X-2__stack的段地址是__X-1__。
任务1-3
点击查看task1_3.asm
assume ds:data, cs:code, ss:stack
data segment
db 20 dup(0)
data ends
stack segment
db 20 dup(0)
stack ends
code segment
start:
mov ax, data
mov ds, ax
mov ax, stack
mov ss, ax
mov sp, 20
mov ah, 4ch
int 21h
code ends
end start
- 问题一
在debug中将执行到line17结束、line19之前,记录此时:寄存器(DS)是__770__, 寄存器(SS)是__772__寄存器(CS)是__774__。 - 问题二
假设程序加载后,code段的段地址是X,则,data段的段地址是__X-4__stack的段地址是__X-2__。
任务1-4
点击查看task1_4.asm
assume ds:data, cs:code, ss:stack
code segment
start:
mov ax, data
mov ds, ax
mov ax, stack
mov ss, ax
mov sp, 20
mov ah, 4ch
int 21h
code ends
data segment
db 20 dup(0)
data ends
stack segment
db 20 dup(0)
stack ends
end start
- 问题一
在debug中将执行到line17结束、line19之前,记录此时:寄存器(DS)是__772__, 寄存器(SS)是__774__寄存器(CS)是__770__。 - 问题二
假设程序加载后,code段的段地址是X,则,data段的段地址是__X+2__stack的段地址是__X+4__。
任务1-5
基于上述四个实验任务的实践、观察,总结并回答:
-
问题一
对于如下定义的段,程序加载后,实际分配给该段的内存空间大小是 ____。xxx segment db N dup(0) xxx ends
实际空间为16*Math.ceil(N/16)个字节
-
问题二
如果将程序task1_1.asm, task1_2.asm, task1_3.asm, task1_4.asm中,伪指令 end start 改成end , 哪一个程序仍然可以正确执行?结合实践观察得到的结论,分析、说明原因。- task1_1.asm
- task1_2.asm
- task1_3.asm
- task1_4.asm
结论:end start指明程序入口在start处,若是没有就会从程序的第一行执行。前三个程序都是都是直接执行空间声明,所以程序无法识别,而最后一个是先执行正确代码,再声明空间,程序能够正确识别,也就是说第四个程序的执行顺序并没有因为start的缺失而发生变化。
- task1_1.asm
实验任务2
编写一个汇编源程序,实现向内存单元b800:0f00 ~ b800:0f9f连续160字节,依次重复填充十六进制数据03 04。
要求
1在实验1的实验任务3中,做过达到同样效果的实验。但当时是通过debug的f命令实现填充的。这一次,要求编程实现。
2 编程实现时,注意进制问题、字节顺序问题。编写后,运行程序,如果与实验1的实验任务3结果不一致,说明程序编写有错误。
编写代码如下:
点击查看代码
DATAS SEGMENT
;此处输入数据段代码
DATAS ENDS
STACKS SEGMENT
;此处输入堆栈段代码
STACKS ENDS
CODES SEGMENT
ASSUME CS:CODES,DS:DATAS,SS:STACKS
START:
mov ax,0b800h
mov ds,ax
mov bx,0f00h
mov cx,80
s:mov byte ptr ds:[bx],0003h
inc bx
mov word ptr ds:[bx],0004h
inc bx
loop s
mov ah, 4ch
int 21h
CODES ENDS
END START
实验结果:
可以发现确实修改成功了。但有些属于只读存储器的内容是不能更改的。
实验任务3
已知8086汇编源程序task3.asm代码片段如下。
点击查看task3.asm
assume cs:code
data1 segment
db 50, 48, 50, 50, 0, 48, 49, 0, 48, 49 ; ten numbers
data1 ends
data2 segment
db 0, 0, 0, 0, 47, 0, 0, 47, 0, 0 ; ten numbers
data2 ends
data3 segment
db 16 dup(0)
data3 ends
code segment
start:
; ×××
code ends
end start
要求
1编程实现把逻辑段data1和逻辑段data2的数据依次相加,结果保存到逻辑段data3中。
2 在debug中加载、反汇编、调试。在数据项依次相加前,和相加后,分别查看三个逻辑段data1,data2, data3对应的内存空间,确认逐一相加后,结果的确保存在了逻辑段data3中。
点击查看补全后代码
assume cs:code
data1 segment
db 50, 48, 50, 50, 0, 48, 49, 0, 48, 49 ; ten numbers
data1 ends
data2 segment
db 0, 0, 0, 0, 47, 0, 0, 47, 0, 0 ; ten numbers
data2 ends
data3 segment
db 16 dup(0)
data3 ends
code segment
start:
mov ax,data1
mov ds,ax
mov ax,data2
mov es,ax
mov ax,data3
mov ss,ax
mov bx,0
mov cx,10
s:mov ax,0
add ax,ds:[bx]
add ax,es:[bx]
mov ss:[bx],ax
inc bx
loop s
mov ah, 4ch
int 21h
code ends
end start
数据段1:
数据段2:
数据段3:
可以很清楚地看到结果正确且保存在了data3中。
实验任务4
已知8086汇编源程序task4.asm代码片段如下
点击查看task4.asm
assume cs:code
data1 segment
dw 2, 0, 4, 9, 2, 0, 1, 9
data1 ends
data2 segment
dw 8 dup(?)
data2 ends
code segment
start:
; ×××
mov ah, 4ch
int 21h
code ends
end start
要求
1补全程序,实现把逻辑段data1中的8个字数据逆序存储到逻辑段b中。
2 汇编、连接后,在debug中加载程序,运行到line15程序退出前,使用d命令查看数据段data2对应的内存空间,确认是否实现题目要求。
点击查看补全后代码
assume cs:code
data1 segment
dw 2, 0, 4, 9, 2, 0, 1, 9
data1 ends
data2 segment
dw 8 dup(?)
data2 ends
code segment
start:
mov ax,data1
mov ds,ax
mov ax,data2
mov es,ax
mov bx,0
mov si,14
s:mov ax,ds:[bx]
mov es:[si],ax
add bx,2
sub si,2
loop s
mov ah, 4ch
int 21h
code ends
end start
结果:
可以很清楚地看到它们成功逆序排放了。
实验任务5
使用任意文本编辑器,录入汇编源程序task5.asm。
点击查看task5.asm
assume cs:code, ds:data
data segment
db 'Nuist'
db 2, 3, 4, 5, 6
data ends
code segment
start:
mov ax, data
mov ds, ax
mov ax, 0b800H
mov es, ax
mov cx, 5
mov si, 0
mov di, 0f00h
s: mov al, [si]
and al, 0dfh
mov es:[di], al
mov al, [5+si]
mov es:[di+1], al
inc si
add di, 2
loop s
mov ah, 4ch
int 21h
code ends
end start
实验步骤:
- 对程序进行汇编、链接,得到可执行文件,运行并观察结果。
- 使用debug工具对程序进行调试,执行到程序返回前,即line25执行之后、line27执行之前,观察结果。
- 源代码中line19的作用是?
- 修改line4里5个字节单元的值,重新汇编、链接、运行,观察结果。
基于观察,分析、猜测这里的数值作用是什么。db 2,3,4,5,6 --> 改成: db 5 dup(2) 或 db 5 dup(5)
回答:
-
把数据中的小写字母转成大写字母
-
这里的数值是用来调整每个字符的颜色的。
实验任务6
已知8086汇编源程序task6.asm代码片段如下。
点击查看task6.asm
assume cs:code, ds:data
data segment
db 'Pink Floyd ' ; 16字节
db 'JOAN Baez ' ; 16字节
db 'NEIL Young ' ; 16字节
db 'Joan Lennon ' ; 16字节
data ends
code segment
start:
; xxx
mov ah, 4ch
int 21h
code ends
end start
要求
1补全程序,将data段中的每行第一个单词从大写->小写。
2在debug中加载程序,反汇编,执行到line13退出前,用d命令查看data段对应的内存空间,确认每行第一个单词已经由大写->小写。
点击查看补全后代码
assume cs:code, ds:data
data segment
db 'Pink Floyd ' ; 16字节
db 'JOAN Baez ' ; 16字节
db 'NEIL Young ' ; 16字节
db 'Joan Lennon ' ; 16字节
data ends
code segment
start:
mov ax,data
mov ds,ax
mov bx,0
mov cx,4
mov si,0
s:mov al,ds:[0+si]
or al,20H
mov ds:[0+si],al
add si,16
loop s
mov ah, 4ch
int 21h
code ends
end start
结果:
第一行原本:
第一行改后:
第二行原本:
第二行改后:
第三行原本:
第三行改后:
第四行原本:
第四行改后:
可以很清楚地看到实现了所需功能。
实验任务7
问题场景描述:
Power idea公司1975年-1979年的基本情况如下:
程序task7.asm的逻辑段data中(line4-6),已经定义好了这些数据。
点击查看task7.asm
assume cs:code, ds:data, es:table
data segment
db '1975', '1976', '1977', '1978', '1979'
dw 16, 22, 382, 1356, 2390
dw 3, 7, 9, 13, 28
data ends
table segment
db 5 dup( 16 dup(' ') ) ;
table ends
code segment
start:
mov ah, 4ch
int 21h
code ends
end start
最后展示的数据格式要求:
同时人均收入保留整数即可
点击查看补全后代码
assume cs:code, ds:data, es:table
data segment
db '1975', '1976', '1977', '1978', '1979'
dw 16, 22, 382, 1356, 2390
dw 3, 7, 9, 13, 28
data ends
table segment
db 5 dup( 16 dup(' ') )
table ends
code segment
start:
mov ax,data
mov ds,ax
mov ax,table
mov es,ax
;mov ax,ds:[1]
;bx,si,di
mov cx,5
mov bx,0
mov si,0
s1:mov ax,ds:[bx]
mov es:[si],ax
add bx,2
mov ax,ds:[bx]
add si,2
mov es:[si],ax
add bx,2
add si,14
loop s1
mov bx,20
mov cx,5
mov si,5
s2:mov ax,ds:[bx]
mov es:[si],ax
add si,2
mov word ptr es:[si],0
add bx,2
add si,14
loop s2
mov bx,30
mov cx,5
mov si,10
s3:mov ax,ds:[bx]
mov es:[si],ax
add bx,2
add si,16
loop s3
mov bx,20
mov di,30
mov cx,5
mov si,13
s4:mov ax,ds:[bx]
mov dl,ds:[di]
div dl
mov es:[si],al
add bx,2
add di,2
add si,16
loop s4
mov ah, 4ch
int 21h
code ends
end start
结果:
检验:
- 数据结构满足要求。
- 年份、收入、雇员就是复制过来,只是收入多了两个字节的空间。经确认数据无误。
- 人均收入:(十进制)16/3=5,22/7=3,382/9=42,1356/13=104,2390/28=85
转化为十六进制分别为:05,03,2A,68,55 --经确认数据无误。
总结
- 对数据段定义的知识有了更深的理解,同时也能灵活使用多个数据段,也掌握了数据段的大小限制以及数据段地址与cs之间的关系的相关知识。
- 掌握了操作数寻址时常用的寄存器,同时能够使用它们去灵活地寻址。
- 掌握了包括db、dw、dd、dup和ptr在内的伪指令。同时也能够灵活使用加减乘除以及与或在内的普通指令,并能完成简单的工作。
- 更加灵活地使用调试命令对程序进行调试。
- 学习了一些markdown的语法,使页面更有层次,重点突出。