实验2 多个逻辑段的汇编源程序编写与调试
四、实验结论
1.实验任务1
●任务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
task1_1调试到line17结束、line19之前截图:

问题回答 :
① 在debug中将执行到line17结束、line19之前,记录此时:寄存器(DS) = _076A_, 寄存器(SS) = _076B_, 寄存器(CS) = _076C_。
② 假设程序加载后,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
task1_2调试到line17结束、line19之前观察寄存器DS, CS, SS值的截图:

问题回答 :
① 在debug中将执行到line17结束、line19之前,记录此时:寄存器(DS) = _076A_, 寄存器(SS) = _076B_, 寄存器(CS) = _076C_
② 假设程序加载后,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
task1_3调试到line17结束、line19之前观察寄存器DS, CS, SS值的截图:

问题回答 :
① 在debug中将执行到line17结束、line19之前,记录此时:寄存器(DS) = _076A_, 寄存器(SS) = _076C_, 寄存器(CS) = _076E_
② 假设程序加载后,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
task1_4调试到line17结束、line19之前观察寄存器DS, CS, SS值的截图:

问题回答 :
① 在debug中将执行到line9结束、line11之前,记录此时:寄存器(DS) = _076C_, 寄存器(SS) = _076E_, 寄存器(CS) = _076A_
② 假设程序加载后,code段的段地址是X,则,data段的段地址是_X+2_, stack的段地址 是_X+4_。
●任务1-5
基于上述四个实验任务的实践、观察,总结并回答:
① 对于如下定义的段,程序加载后,实际分配给该段的内存空间大小是 _ ⌈(N/16)⌉*16字节_。

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

对修改后的task1_2.exe进行反汇编:

对修改后的task1_3.exe进行反汇编:

对修改后的task1_4.exe进行反汇编:

可以看出,将伪指令 end start 改成 end,只有task1_4.asm经过执行并反汇编后得到的汇编代码是正确的,即只有task1_4.asm能够正常执行。因为end start指明了程序从start开始执行,如果只有end则程序默认从第一行开始执行。而task1_1.asm, task1_2.asm, task1_3.asm的代码中都把数据段和栈段的空间声明放在了代码段之前,程序误把这些空间声明当作代码段执行,因此不能正确执行。而task1_4.asm把代码段放在了空间声明之前,因此可以正确执行。
2.实验任务2
汇编源代码:
assume cs:code code segment start: mov ax,0b800h mov ds,ax mov bx,0f00h mov cx,80 s: mov ds:[bx],0403h add bx,2 loop s mov ah,4ch int 21h code ends end start
运行结果截图:

3.实验任务3
补充完整后的汇编源代码:
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
在数据项依次相加之前,查看逻辑段data1, data2, data3对应的内存空间数据原始值的 debug命令和截图:
首先通过反汇编得知三个数据段的地址分别是076A,076B,076C,用d命令查看即可:




依次相加之后,查看逻辑段data1, data2, data3对应的内存空间数据原始值的debug命令和截图:



可见数据已经成功相加。
4.实验任务4
补充完整后的汇编源代码:
assume cs:code data1 segment dw 2, 0, 4, 9, 2, 0, 1, 9 data1 ends data2 segment dw 8 dup(?) data2 ends stack segment dw 16 dup(?) code segment start: mov ax,data1 mov ds,ax mov ax,data2 mov es,ax mov ax,stack mov ss,ax mov sp,16 mov bx,0 mov cx,8 s1: push [bx] add bx,2 loop s1 mov bx,0 mov cx,8 s2: pop es:[bx] add bx,2 loop s2 mov ah, 4ch int 21h code ends end start
在程序退出前,使用d命令查看数据段data2对应的内存空间的截图:
首先通过反汇编得知data1,data2,stack的地址分别是076A,076B,076C:

用d命令查看data1的内容:

用d命令查看data2的内容:

可见数据已经逆序存放。
5.实验任务5
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工具对程序进行调试,使用g命令一次性执行到程序返回前(即ine25执行之后、line27 执行之前)的截图:

源代码中line19的作用是:将小写字母转换为大写字母
源代码中data段line4的字节数据的用途是:将字母设置为不同的颜色
修改line4里5个字节单元的值,重新汇编、链接、运行,观察结果,可看出NUIST五个字母全部变为了绿色:

因此可知字节数据用来调整颜色
6.实验任务6
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: mov ax,data mov ds,ax mov bx,0 mov cx,4 s: mov al,[bx] or al,20h mov [bx],al add bx,16 loop s mov ah, 4ch int 21h code ends end start
在程序退出前,使用d命令查看数据段data对应的内存空间的截图:

可见已经成功将每行第一个字母变为小写。
7.实验任务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 bx,0 mov si,0 mov cx,5 s1: mov ax,[bx] mov es:[si],ax add bx,2 add si,2 mov ax,[bx] mov es:[si],ax add bx,2 add si,14 loop s1 mov si,5 mov cx,5 s2: mov ax,[bx] mov es:[si],ax add bx,2 add si,2 mov word ptr es:[si],0 add si,14 loop s2 mov si,10 mov cx,5 s3: mov ax,[bx] mov es:[si],ax add bx,2 add si,16 loop s3 mov bx,20 mov di,30 mov si,13 mov cx,5 s4: mov ax,[bx] mov dl,[di] div dl mov es:[si],al add si,1 mov byte ptr es:[si],0 add bx,2 add di,2 add si,15 loop s4 mov ah, 4ch int 21h code ends end start
查看table段原始数据信息截图:
首先根据反汇编得知table段地址为076D:

用d命令查看即可:

在debug中运行到程序退出之前,使用d命令查看table段对应的内存空间的截图,确认信息是 否按要求结构化地写入到指定内存:

表格最后一列的商数据,根据计算分别是16/3=5,22/7=3,382/9=42,1356/13=104,2390/28=85,转换为十六进制分别是05,03,2A,68,55,对应于图中的倒数第三列,可见结果正确。
五、实验总结
1.我知道了程序给段空间分配的内存都是16字节的整数倍,只会多不会少。若一个段定义的字节数是N ,则实际分配的字节数为⌈(N/16)⌉*16。
2.我知道了end start的作用是指明程序从start处开始运行。如果没有end start,则程序从第一行开始运行。
3.我知道了将数据段数据依次入栈后,如果出栈的地址顺序和入栈的地址顺序相同,则数据就会逆序存放。
浙公网安备 33010602011771号