实验2 多个逻辑段的汇编源程序编写与调试

2021/11/8 22:11:08

本文主要是介绍实验2 多个逻辑段的汇编源程序编写与调试,对大家解决编程问题具有一定的参考价值,需要的程序猿们随着小编来一起学习吧!

任务一

任务1-1

对程序task1_1.asm进行汇编、连接,用debug加载、跟踪调试,基于结果,回答问题:

 1 assume ds:data, cs:code, ss:stack
 2 
 3 data segment
 4     db 16 dup(0) ; 预留16个字节单元,初始值均为0
 5 data ends
 6 
 7 stack segment
 8     db 16 dup(0) ;预留16个字节单元,初始值均为0
 9 stack ends
10 code segment
11 start:
12     mov ax, data
13     mov ds, ax
14 
15     mov ax, stack
16     mov ss, ax
17     mov sp, 16 ; 设置栈顶
18 
19     mov ah, 4ch
20     int 21h
21 code ends
22 end start

 

① 在debug中将执行到line17结束、line19之前,记录此时:寄存器(DS) = __0770h__, 寄存器(SS) = __0771h__, 寄存器(CS) = __0772h__

② 假设程序加载后,code段的段地址是X,则,data段的段地址是__X-2h__, stack的段地址是__X-1h__。

  code段之前,分别预留了16个字节单元作为数据段和栈段,实际地址需要在code段的基础上分别减去20h、10h,因此对应的段地址只需要在code段的基础之上减去2h、1h。

 

任务1-2

对程序task1_2.asm进行汇编、连接,用debug加载、跟踪调试,基于结果,回答问题

 1 assume ds:data, cs:code, ss:stack
 2 
 3 data segment
 4     db 4 dup(0) ; 预留4个字节单元,初始值均为0
 5 data ends
 6     
 7 stack segment
 8     db 8 dup(0) ; 预留8个字节单元,初始值均为0
 9 stack ends
10 code segment
11 start:
12     mov ax, data
13     mov ds, ax
14 
15     mov ax, stack
16     mov ss, ax
17     mov sp, 8 ; 设置栈顶
18 
19     mov ah, 4ch
20     int 21h
21 code ends
22 end start

① 在debug中将执行到line17结束、line19之前,记录此时:寄存器(DS) = __0770h__, 寄存器(SS) = __0771h__, 寄存器(CS) = __0772h__

② 假设程序加载后,code段的段地址是X,则,data段的段地址是__X-2h__, stack的段地址是__X-1h__。

因为段地址在计算实际地址是是需要乘以16的,因此,为段分配的每一个地址都是包含了16个字节的空间,所以data、stack对应的段地址与上相同

 

任务1-3

对程序task1_3.asm进行汇编、连接,用debug加载、跟踪调试,基于结果,回答问题

 1 assume ds:data, cs:code, ss:stack
 2 
 3 data segment
 4     db 20 dup(0) ; 预留20个字节单元,初始值均为0
 5 data ends
 6 
 7 stack segment
 8     db 20 dup(0) ; 预留20个字节单元,初始值均为0
 9 stack ends
10 code segment
11 start:
12     mov ax, data
13     mov ds, ax
14 
15     mov ax, stack
16     mov ss, ax
17     mov sp, 20 ; 设置初始栈顶
18 
19     mov ah, 4ch
20     int 21h
21 code ends
22 end start

① 在debug中将执行到line17结束、line19之前,记录此时:寄存器(DS) = __0770h__, 寄存器(SS) = __0772h__, 寄存器(CS) = __0774h__

② 假设程序加载后,code段的段地址是X,则,data段的段地址是__X-4h__, stack的段地址是__X-2h__。

  理由同上,预留段地址空间,大小为16的整数倍字节,因此data、stack段为预留32字节

 

任务1-4

对程序task1_4.asm进行汇编、连接,用debug加载、跟踪调试,基于结果,回答问题

 1 assume ds:data, cs:code, ss:stack
 2 code segment
 3 start:
 4     mov ax, data
 5     mov ds, ax
 6 
 7     mov ax, stack
 8     mov ss, ax
 9     mov sp, 20
10 
11     mov ah, 4ch
12     int 21h
13 code ends
14 
15 data segment
16     db 20 dup(0)
17 data ends
18 
19 stack segment
20     db 20 dup(0)
21 stack ends
22 
23 end start

① 在debug中将执行到line9结束、line11之前,记录此时:寄存器(DS) = __0772h__, 寄存器(SS) = __0774h__, 寄存器(CS) = __0770h__

② 假设程序加载后,code段的段地址是X,则,data段的段地址是__X+2h__, stack的段地址是 __X+4h__。

data、stack段预留空间在code段后,因此在code段地址的基础上加上对应的空间大小

 

任务1-5

基于上述四个实验任务的实践、观察,总结并回答

① 对于如下定义的段,程序加载后,实际分配给该段的内存空间大小是 __(N/16+1)*16_bytes__

1 xxx segment
2     db N dup(0)
3 xxx ends

② 如果将程序task1_1.asm, task1_2.asm, task1_3.asm, task1_4.asm中,伪指令 end start 改成 end , 哪一个程序仍然可以正确执行。结合实践观察得到的结论,分析、说明原因。

end start指明程序入口在start处,若是没有就会从程序的第一行开始执行。前三个程序都是都是先预留的空间,所以程序无法识别,因此无法执行程序。而最后一个程序是先写代码,再声明空间,程序能够正确识别,也就是说第四个程序的执行顺序并没有因为start的缺失而发生变化。

 

任务二

编写一个汇编源程序,实现向内存单元b800:0f00 ~ b800:0f9f连续160字节,依次重复填充十六进制数 据03 04

Tips: 1. 在实验1的实验任务3中,做过达到同样效果的实验。但当时是通过debug的f命令实现填充的。这一 次,要求编程实现。

在debug中,使用f命令,向内存单元批量填写数据。 -f b800:0f00 0f9f 03 04 把内存单元区间b800:0f00 ~ b800:0f9f连续160个字节,依次重复填充十六进制数据03 04。

使用f命令填充

 编程实现

 1 assume cs:code
 2 code segment
 3 start:
 4     mov ax,0b8f0h
 5     mov ds,ax
 6     mov bx,0h
 7     mov ax,0403h
 8     mov cx,50h        ;每次输入一个字数据,循环80次
 9     
10 s:  mov [bx],ax
11     add bx,2
12     loop s
13     
14     mov ah,4ch
15     int 21h
16 code ends
17 end start    

 后面部分由于权限原因,无法进行数据修改

 

 

任务三

已知8086汇编源程序task3.asm代码片段如下

 1 assume cs:code
 2 data1 segment
 3     db 50, 48, 50, 50, 0, 48, 49, 0, 48, 49 ; ten numbers
 4 data1 ends
 5 
 6 data2 segment
 7     db 0, 0, 0, 0, 47, 0, 0, 47, 0, 0 ; ten numbers
 8 data2 ends
 9 
10 data3 segment
11     db 16 dup(0)
12 data3 ends
13 
14 code segment
15 start:
16     ; ×××
17     mov ah, 4ch
18     nt 21h
19 code ends
20 end start

① 编程实现把逻辑段data1和逻辑段data2的数据依次相加,结果保存到逻辑段data3中。

 1 assume cs:code
 2 
 3 data1 segment
 4     db 50, 48, 50, 50, 0, 48, 49, 0, 48, 49 ; ten numbers
 5 data1 ends
 6 
 7 data2 segment
 8     db 0, 0, 0, 0, 47, 0, 0, 47, 0, 0 ; ten numbers
 9 data2 ends
10 
11 data3 segment
12     db 16 dup(0)
13 data3 ends
14 
15 code segment
16 start:
17     mov ax,cs
18     sub ax,3h        ;data1段地址
19     mov ds,ax
20     mov ax,data3
21     mov es,ax
22     mov bx,0
23     mov cx,8h
24     
25 s:    mov ax,[bx]
26     add es:[bx],ax
27     mov ax,[bx+10h]
28     add es:[bx],ax
29     add bx,2
30     loop s
31     
32     mov ah, 4ch
33     int 21h
34 code ends
35 end start

② 在debug中加载、反汇编、调试。在数据项依次相加前,和相加后,分别查看三个逻辑段data1, data2, data3对应的内存空间,确认逐一相加后,结果的确保存在了逻辑段data3中。

 

 执行程序后:

 

 

 

 

任务四

已知8086汇编源程序task4.asm代码片段如下

 1 assume cs:code
 2 
 3 data1 segment
 4     dw 2, 0, 4, 9, 2, 0, 1, 9
 5 data1 ends
 6 
 7 data2 segment
 8     dw 8 dup(?)
 9 data2 ends
10 
11 code segment
12 start:
13     ; ×××
14     mov ah, 4ch
15     int 21h
16 code ends
17 end start

① 补全程序,实现把逻辑段data1中的8个字数据逆序存储到逻辑段b中。

 1 assume cs:code
 2 
 3 data1 segment
 4     dw 2, 0, 4, 9, 2, 0, 1, 9
 5 data1 ends
 6 
 7 data2 segment
 8     dw 8 dup(?)
 9 data2 ends
10 
11 code segment
12 start:
13     mov ax,code
14     sub ax,2h        ;data1段地址
15     mov ds,ax
16     mov bx,0h
17     mov si,1eh
18     mov cx,8h
19     
20 s:  mov ax,[bx]
21     mov [si],ax
22     add bx,2
23     sub si,2
24     loop s
25     
26     mov ah, 4ch
27     int 21h
28 code ends
29 end start

② 汇编、连接后,在debug中加载程序,运行到line15程序退出前,使用d命令查看数据段data2对应的 内存空间,确认是否实现题目要求。

 

 

 

 

任务五

阅读源程序,从理论上分析源代码的功能,尤其是line15-25,循环实现的功能是什么,逐行理解每条指 令的功能:

 1 assume cs:code, ds:data
 2 data segment
 3     db 'Nuist'
 4     db 2, 3, 4, 5, 6
 5 data ends
 6 
 7 code segment
 8 start:
 9     mov ax, data
10     mov ds, ax
11     
12     mov ax, 0b800H
13     mov es, ax
14     
15     mov cx, 5
16     mov si, 0
17     mov di, 0f00h
18 s:  mov al, [si]
19     and al, 0dfh
20     mov es:[di], al
21     mov al, [5+si]
22     mov es:[di+1], al
23     inc si
24     add di, 2
25     loop s
26     
27     mov ah, 4ch
28     int 21h
29 code ends
30 end start

① 对程序进行汇编、链接,得到可执行文件,运行并观察结果。

②使用debug工具对程序进行调试,执行到程序返回前,即line25执行之后、line27执行之前,观察结果。

③源代码中line19的作用是?

将[si]中的数据和0dfh进行与运算,然后赋值给es:[di]

 

④修改line4里5个字节单元的值,重新汇编、链接、运行,观察结果。

修改为 db 5 dup(2),得到结果颜色发生变化,line4作用为调整对应5个字符的颜色

 

 

 

 

 

任务六

已知8086汇编源程序task6.asm代码片段如下

 1 assume cs:code, ds:data
 2 
 3 data segment
 4     db 'Pink Floyd ' ; 16字节
 5     db 'JOAN Baez ' ; 16字节
 6     db 'NEIL Young ' ; 16字节
 7     db 'Joan Lennon ' ; 16字节
 8 data ends
 9 
10 code segment
11 start:
12     ; ×××
13     mov ah, 4ch
14     int 21h
15 code ends
16 end start

① 补全程序,将data段中的每行第一个单词从大写->小写。

 1 assume cs:code, ds:data
 2 
 3 data segment
 4     db 'Pink Floyd      ' ; 16字节
 5     db 'JOAN Baez       ' ; 16字节
 6     db 'NEIL Young      ' ; 16字节
 7     db 'Joan Lennon     ' ; 16字节
 8 data ends
 9 
10 code segment
11 start:
12     mov ax,data
13     mov ds,ax
14     mov bx,0
15     mov cx,4h
16 
17 s:  mov ax,[bx]
18     or ax,20h        ;大写变小写
19     mov [bx],ax
20     mov ax,[bx+10h]
21     or ax,20h
22     mov [bx+10h],ax
23     mov ax,[bx+20h]
24     or ax,20h
25     mov [bx+20h],ax
26     mov ax,[bx+30h]
27     or ax,20h
28     mov [bx+30h],ax
29     
30     inc bx
31     loop s
32     
33     mov ah, 4ch
34     int 21h
35 code ends
36 end start

② 在debug中加载程序,反汇编,执行到line13退出前,用d命令查看data段对应的内存空间,确认每 行第一个单词已经由大写->小写

 

 

 

任务七

问题场景描述:

Power idea公司1975年-1979年的基本情况如下:

程序task7.asm的逻辑段data中(line4-6),已经定义好了这些数据:

 1 assume cs:code, ds:data, es:table
 2 
 3 data segment
 4     db '1975', '1976', '1977', '1978', '1979'
 5     dw 16, 22, 382, 1356, 2390
 6     dw 3, 7, 9, 13, 28
 7 data ends
 8 
 9 table segment
10     db 5 dup( 16 dup(' ') ) ;
11 table ends
12 
13 code segment
14 start:
15     mov ax,data
16 
17     mov ah, 4ch
18     int 21h
19 code ends
20 end start

如图:

① 补全程序,实现题目要求,把年份、收入、雇员人数、人均收入,以结构化方式写入table段中。 表格中,每行数据,在逻辑段table中占16个字节,各项数据占据字节大小分配如下。期中,数据之间用 空格间隔。

此处被除数为16位:0010h、0016h、017Eh等,除数可以看作8位:0003h、0007h、001Ch

 1 assume cs:code, ds:data, es:table
 2 
 3 data segment
 4     db '1975', '1976', '1977', '1978', '1979'
 5     dw 16, 22, 382, 1356, 2390
 6     dw 3, 7, 9, 13, 28
 7 data ends
 8 
 9 table segment
10     db 5 dup( 16 dup(' ') ) ;
11 table ends
12 
13 code segment
14 start:
15     mov ax,data
16     mov ds,ax
17     mov ax,table
18     mov es,ax
19     mov si,14h    ;收入首地址
20     mov di,1eh    ;雇员人数首地址
21     mov bx,0    ;es
22     mov bp,0    ;ds
23     mov cx,5
24     
25 s1:    mov dx,ds:[bp]
26     mov es:[bx],dx
27     add bx,2
28     add bp,2
29     mov dx,ds:[bp]
30     mov es:[bx],dx
31     add bx,3
32     add bp,2
33     mov dx,[si]
34     mov es:[bx],dx        ;收入
35     mov ax,[si]        ;被除数
36     add bx,5h
37     add si,2
38     mov dx,[di]        
39     mov es:[bx],dx        ;雇员人数
40     div byte ptr [di]    ;除数
41     add bx,3
42     add di,2
43     mov ah,0
44     mov es:[bx],ax        ;ax中为商
45     add bx,3
46     loop s1
47     
48     mov ah, 4ch
49     int 21h
50 code ends
51 end start

 ② 汇编、连接后,在debug中加载、调试程序。灵活、合理使用u命令、g命令、d命令,显示刚开始逻 辑段table的数据信息,以及,结构化存入数据后,数据段table的数据信息,确认实现题目要求。

 经过数据对比和计算,可以发现数据格式和内容全部正确,人均收入分别为05h、03h、2Ah、68h、55h

 

 

总结

  1. 经过本次实验,对于段前缀的使用,以及可用于内存单元寻址的寄存器bx、si、di、dp的使用有了进一步的掌握和理解,可以更加灵活地进行寻址,同时运用多个段来进行复制、计算等操作
  2. 掌握了db、dw、dd、dup、ptr伪指令的使用,能更加灵活的为数据分配空间,或在计算时更灵活的处理每个字节的数据
  3. 能够利用add、sub、and、div等命令来对寄存器内的操作数进行计算来

 



这篇关于实验2 多个逻辑段的汇编源程序编写与调试的文章就介绍到这儿,希望我们推荐的文章对大家有所帮助,也希望大家多多支持为之网!


扫一扫关注最新编程教程