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

实验任务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

 

    调试截图如下

    

 

     问题回答:

    ① 在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

  截图如下

    

     问题回答

    ① 在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
    基于上述四个实验任务的实践、观察,总结并回答:
    ① 对于如下定义的段,程序加载后,实际分配给该段的内存空间大小是 __16*ceil(N/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_1.asm,task1_2.asm,task1_3.asm不可正确运行
     task1_4.asm可以正确运行.start和end start表示了程序开始与结束的地方,框出了执行代码段.
     若将end start换成end则无法知晓可执行代码段的位置,默认从代码开头开始执行,1,2,3均是以
     数据段开始,所以不能执行
 
2. 实验任务2
  编写一个汇编源程序,实现向内存单元b800:0f00 ~ b800:0f9f连续160字节,依次重复填充十六进制数
  据03 04。
  代码如下
assume cs:code

code segment
start: 
     mov ax, 0b800h
     mov ds, ax
     mov bx, 0f00h
     mov cx, 160
 s:
     mov [bx], 0403h
     add bx, 2
     loop s
 
     mov ah, 4ch
     int 21h
code ends

end start
  结果如下
  

 

   

3. 实验任务3
已知8086汇编源程序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:
; ×××
mov ah, 4ch
int 21h
code ends
end start
  要求:
  ① 编程实现把逻辑段data1和逻辑段data2的数据依次相加,结果保存到逻辑段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 cx,10
    mov bx,0
    
s1: mov ax, es:[bx]
    add [bx],ax
    inc bx
    inc bx
    loop s1

    mov ax,data3
    mov es,ax
    mov bx,0
    mov cx,10

s2: mov ax,[bx]
    mov es:[bx],ax
    inc bx
    loop s2

    mov ah,4ch
    int 21h

code ends
end start

 

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

 

   运行结束后查看该空间的数值

   

 

   发现data3中的数值已经被相加和修改

 

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

code segment
start:
    mov ax,data1
    mov ds,ax
    mov ax,data2
    mov ss,ax
    mov sp,16

    mov bx,0
    mov cx,8

s: push [bx]
    add bx,2
    loop s


    mov ah, 4ch
    int 21h
code ends
end start

 

  在debug中加载、反汇编、调试截图
  要求给出,在程序退出前,使用d命令查看数据段data2对应的内存空间的截图。  

  

 

 

 

5. 实验任务5
运行结果截图

 

 

使用debug工具对程序进行调试,使用g命令一次性执行到程序返回前(即ine25执行之后、line27
执行之前)的截图

 

   源代码中line19的作用是?

      大小写转换.与0dfh做与运算会将原数值的第三位置0,即减去32,也就实现了小写字母转变为大写字母.

     第十五行的作用是设置循环次数为5,第十六行的作用是设置循环初始位置和偏移地址

     第十七行的作用是将显存地址放在寄存器中,以便后续打印使用.循环体内,第十八行和第十九行的作用是将

     数据段中的数据复制到al中并转换为大写字母.20,21,22行作用是将字符送到屏幕的指定位置并显示

  源代码中data段line4的字节数据的用途是? 
      

 

    修改代码后文字颜色改变,应该是用于修改字符颜色的数据

 

6. 实验任务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 bx,0  ;偏移量
    mov cx,4  ;循环次数为4

s1: mov dx,cx ;存储外循环次数
    mov cx,4  ;设置内循环次数
    mov si,0  ;设置偏移亮
    s2: mov ax,[bx+si]
        or  ax,00100000B  ;大写字母的ASCII码减去32,小写字母不变
        mov [bx+si],ax    ;写回原来的空间
        inc si            ;自加
        loop s2
    add bx,16             ;处理下一个db
    mov cx,dx             ;恢复外循环次数记录
    loop s1

   mov ah, 4ch
   int 21h
code ends
end start

 

在debug中加载、反汇编、调试截图
反汇编代码如下所示
查看程序未执行时的相关内存如下
要求给出,在程序退出前,使用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   ;将数据段地址交给ds
    mov ax,table
    mov es,ax   ;将表格地址交给es

    mov bx,0    ;设置关于data数据区的偏移位置
    mov di,0    ;设置输入位置为table第一行的第一个
    mov cx,5    ;设置循环次数

s1: mov ax,[bx]
    mov es:[di],ax ;写入table的相应位置
    mov ax,[bx+2]
    mov es:[di+2],ax
    add bx,4    ;指向下一个年份
    add di,16   ;进入table的下一行
    loop s1

    mov di,5    ;指向table第一行的第五个位置
    mov cx,5    ;设置循环次数

s2: mov ax,[bx]
    mov es:[bx],ax
    add bx,2
    add di,16
    loop s2

    mov di,10
    mov cx,5

s3: mov ax,[bx]
    mov es:[bx],ax
    add bx,2
    add di,16
    loop s3

    mov di,5
    mov cx,5

s4: mov ax,es:[di]  ;被除数"收入"放在ax中
    mov bp,es:[di+5] ;除数"雇员数"放在bp中
    div bp
    mov es:[di+8],ax
    add di,16
    loop s4

    mov ah, 4ch
    int 21h
code ends
end start

 

调试截图
查看table段原始数据信息截图

 

 

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

 

 

posted @ 2021-11-11 19:42  Nekasu  阅读(81)  评论(1)    收藏  举报