实验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.我知道了将数据段数据依次入栈后,如果出栈的地址顺序和入栈的地址顺序相同,则数据就会逆序存放。

posted @ 2021-11-06 12:59  Romantic2021  阅读(72)  评论(1)    收藏  举报