实验2 多个逻辑段的汇编源程序编写与调试
实验任务1
· 任务1-1
对程序task1_1.asm进行汇编、连接,用debug加载、跟踪调试,基于结果,回答问题。
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
对程序task1_1.asm进行汇编、连接,用debug加载后,使用debug的t命令将程序执行到line17结束,line19之前,结果如下:
回答问题:
① 由上图可知寄存器(DS)=076A,寄存器(SS)=076B,寄存器(CS)=076C
② 假设程序加载后,code段的段地址是X,则,data段的段地址是X-2,stack的段地址是X-1
· 任务1-2
对程序task1_2.asm进行汇编、连接,用debug加载、跟踪调试,基于结果,回答问题。
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
对程序task1_2.asm进行汇编、连接,用debug加载后,使用debug的t命令将程序执行到line17结束,line19之前,结果如下:
回答问题:
① 由上图可知寄存器(DS)=076A,寄存器(SS)=076B,寄存器(CS)=076C
② 假设程序加载后,code段的段地址是X,则,data段的段地址是X-2,stack的段地址是X-1
· 任务1-3
对程序task1_3.asm进行汇编、连接,用debug加载、跟踪调试,基于结果,回答问题。
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
对程序task1_3.asm进行汇编、连接,用debug加载后,使用debug的t命令将程序执行到line17结束,line19之前,结果如下:
① 由上图可知寄存器(DS)=076A,寄存器(SS)=076C,寄存器(CS)=076E
② 假设程序加载后,code段的段地址是X,则,data段的段地址是X-4,stack的段地址是X-2
· 任务1-4
对程序task1_4.asm进行汇编、连接,用debug加载、跟踪调试,基于结果,回答问题。
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
对程序task1_4.asm进行汇编、连接,用debug加载后,使用debug的t命令将程序执行到line9结束,line11之前,结果如下:
① 由上图可知寄存器(DS)=076C,寄存器(SS)=076E,寄存器(CS)=076A
② 假设程序加载后,code段的段地址是X,则,data段的段地址是X+2,stack的段地址是X+4
· 任务1-5
基于上述四个实验任务的实践、观察,总结并回答:
① 对于如下定义的段,程序加载后,实际分配给该段的内存空间大小是(N/16+1)*16
xxx segment db N dup(0) xxx ends
② 如果将程序task1_1.asm, task1_2.asm, task1_3.asm, task1_4.asm中,伪指令end start改成 end , 哪一个程序仍然可以正确执行?结合实践观察得到的结论,分析、说明原因。
修改后,每一个程序都可以加载链接,但是只有task1_4能正确执行,因为start表示代码段的开始。task1_4将代码段卸载了前面而其他的则将代码段写在了后面数据段在前, 如果没有end start,数据段部分就会被当作代码段执行。所以得出代码段不在最前,则需要指定程序入口。
实验任务2
编写一个汇编源程序,实现向内存单元b800:0f00 ~ b800:0f9f连续160字节,依次重复填充十六进制数据03 04。
(1)汇编源代码:
assume cs:code code segment start: mov ax, 0b800h mov ds, ax mov bx, 0f00h mov cx, 50h s: mov ds:[bx], 0403h add bx, 2 loop s mov ah, 4ch int 21h code ends end start
(2)运行结果截图 :
实验任务3
① 编程实现把逻辑段data1和逻辑段data2的数据依次相加,结果保存到逻辑段data3中。
② 在debug中加载、反汇编、调试。在数据项依次相加前,和相加后,分别查看三个逻辑段data1, data2, data3对应的内存空间,确认逐一相加后,结果的确保存在了逻辑段data3中。
task3.asm
assume ds:data1, cs:code data1 segment db 50, 48, 50, 50, 0, 48, 49, 0, 48, 49 data1 ends data2 segment db 0, 0, 0, 0, 47, 0, 0, 47, 0, 0 data2 ends data3 segment db 16 dup(0) data3 ends code segment start: mov ax, data1 mov ds, ax mov bx, 0 mov cx, 10h
s: mov ax, ds:[bx]
add ax, ds:[bx+10h]
mov ds:[bx+20h], ax
inc bx loop s mov ah, 4ch int 21h code ends end start
相加前的截图:
相加后的截图:
反汇编截图:
实验任务4
① 补全程序,实现把逻辑段data1中的8个字数据逆序存储到逻辑段b中。
② 汇编、连接后,在debug中加载程序,运行到line15程序退出前,使用d命令查看数据段data2对应的 内存空间,确认是否实现题目要求。
task4.asm
assume cs:code, ss:stack data1 segment dw 2, 0, 4, 9, 2, 0, 1, 9 data1 ends data2 segment dw 8 dup(0) data2 ends stack segment dw 8 dup(0) stack ends code segment start: mov ax, data1 mov ds, ax mov sp, 9 mov bx, 0 mov cx, 8 s1: push ds:[bx] add bx, 2 loop s1 mov ax, data2 mov ds, ax mov bx, 0 mov cx, 8 s2: pop ds:[bx] add bx, 2 loop s2 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
(1)运行结果截图
(2)使用g命令执行到ine25执行之后、line27 执行之前
(3)源码中line19的作用是:将小写字母转换为大写字母
(4)修改line4里5个字节单元的值,重新汇编、链接、运行,观察结果。
db 2,3,4,5,6
--> 改成:
db 5 dup(2) 或 db 5 dup(5)
得到结果:
由此猜想结论:源码中data段line4的字节数据的用途是改变字符的颜色
实验任务6
task6.asm
assume cs:code, ds:data data segment db 'Pink Floyd ' db 'JOAN Baez ' db 'NEIL Young ' db 'Joan Lennon ' data ends code segment start: mov ax, data mov ds, ax mov ax, data mov es, ax mov bx, 0 mov cx, 4 s1: mov si, cx mov cx, 4 s2: mov al, es:[bx] or al, 20h mov es:[bx], al inc bx loop s2 mov cx, si mov bx, 0 mov ax, es inc ax mov es, ax loop s1 mov ah, 4ch int 21h code ends end start
反汇编截图:
运行结果截图:
实验任务7
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 ax, data mov ds, ax mov ax, table mov es, ax mov cx, 5 mov bx, 0 mov si, 0 s0: push cx mov cx, 4 s1: mov al, [bx] mov es:[si], al inc si inc bx loop s1 mov byte ptr es:[si], ' ' add si, 12 pop cx loop s0 mov cx, 5 mov bx, 20 mov si, 5 s2: mov ax, [bx] mov dl, 10 div dl add ah, 48 mov es:[si+3], ah mov ah, 0 div dl add ah, 48 mov es:[si+2], ah mov ah, 0 div dl add ah, 48 mov es:[si+1], ah mov ah, 0 div dl add ah, 48 mov es:[si], ah mov ah, 0 mov byte ptr es:[si+4], ' ' add si, 16 add bx, 2 loop s2 mov cx, 5 mov bx, 30 mov si, 10 s3: mov ax, [bx] mov dl, 10 div dl add ah, 48 mov es:[si+1], ah mov ah, 0 div dl add ah, 48 mov es:[si], ah mov ah, 0 mov byte ptr es:[si+2], ' ' add si, 16 add bx, 2 loop s3 mov cx, 5 mov bx, 20 mov si, 13 s4: mov ax, [bx] mov dl, [bx+10] div dl mov ah, 0 mov dl, 10 div dl add ah, 48 mov es:[si+1], ah mov ah, 0 div dl add ah, 48 mov es:[si], ah mov ah, 0 mov byte ptr es:[si+2], ' ' add si, 16 add bx, 2 loop s4 mov ah, 4ch int 21h code ends end start
查看原始数据截图:
确认信息是否按要求写入到指定内存截图:
实验结论
· 通过这次实验对汇编语言编程有了更深一步的认识,汇编语言的编写完全仿照机器的思维来实现
· 这次实验中我也了解了之前并不是很在意start、end start的作用:start表明了程序从哪里开始执行,代码段位置不同时可能会发生不一样的变化
· 实践了课上说过的使用and和or命令可以巧妙地将二进制的某一位置0或置1从而实现大小写转换等功能
· div命令有两种使用方式,一种是16位被除数除以8位除数,ax中al存放商,ah存放余数;另一种是32位被除数除以16位除数,ax存放商,dx存放余数。