实验二 多个逻辑段的汇编源程序编写与调试
实验任务1
任务1-1
源程序代码:
1 assume ds:data, cs:code, ss:stack 2 3 data segment 4 db 16 dup(0) 5 data ends 6 7 stack segment 8 db 16 dup(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) = _076A_, 寄存器(SS) =__076B__, 寄存器(CS) =_076C_
② 假设程序加载后,code段的段地址是X,则,data段的段地址是__X-2__, stack的段地址是__X-1__
任务1-2
源程序代码:
1 assume ds:data, cs:code, ss:stack 2 3 data segment 4 db 4 dup(0) 5 data ends 6 7 stack segment 8 db 8 dup(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) = __076A__, 寄存器(SS) =_076B__, 寄存器(CS) = __076C__
② 假设程序加载后,code段的段地址是X,则,data段的段地址是__X-2__, stack的段地址是__X-1__。
任务1-3
源程序代码:
1 assume ds:data, cs:code, ss:stack 2 3 data segment 4 db 20 dup(0) 5 data ends 6 7 stack segment 8 db 20 dup(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) = __076A_, 寄存器(SS) =__076C__, 寄存器(CS) = __076E__
② 假设程序加载后,code段的段地址是X,则,data段的段地址是__X-4__, stack的段地址是__X-2__.
任务1-4
源程序代码:
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 end
截图:
① 在debug中将执行到line9结束、line11之前,记录此时:寄存器(DS) = __076C__, 寄存器(SS) =__076E__, 寄存器(CS) = _076A_
② 假设程序加载后,code段的段地址是X,则,data段的段地址是__X+2__, stack的段地址是_X+4__。
任务1-5
① 对于如下定义的段,程序加载后,实际分配给该段的内存空间大小是 16*ceil(N/16) .
② task1_4.asm仍然可以正确执行,因为end后面说明了原程序的入口,若为空那么程序将从从头开始执行,只有task1_4的一开始就是代码段的内容。
实验任务2
源程序代码:
1 assume cs:code 2 code segment 3 start: 4 mov ax, 0b800h ;16进制字母开头的要加0 5 mov ds, ax 6 7 mov bx, 0f00h 8 mov cx, 80 9 s:mov byte ptr [bx], 03h 10 inc bx 11 mov byte ptr [bx], 04h 12 inc bx 13 loop s 14 15 mov ah, 4ch 16 int 21h 17 code ends 18 end start
实验截图:
实验任务3
源程序代码:
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 mov bx, 0 17 mov cx, 10 18 19 s:mov dx, 0 20 mov ax, data1 21 mov ds, ax 22 add dx, [bx] 23 mov ax, data2 24 mov ds, ax 25 add dx, [bx] 26 27 mov ax, data3 28 mov ds, ax 29 mov [bx], dx 30 inc bx 31 loop s 32 33 mov ah, 4ch 34 int 21h 35 code ends 36 end start
截图:
反汇编得到逻辑段的地址:
初始状态:
执行完成后:
可以发现的确完成了相加的操作。
实验任务4
源程序代码:将data2用作栈段在进行压入就可以实现倒序。
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, data1 14 mov ds, ax 15 mov ax, data2 16 mov ss, ax 17 mov sp, 16 18 mov bx, 0 19 mov cx, 8 20 s:push [bx] 21 add bx, 2 22 loop s 23 24 mov ah, 4ch 25 int 21h 26 code ends 27 end start
截图:
首先反汇编得到逻辑段地址
执行前后;
实验任务5
源程序代码:
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
程序运行结果:
25行之后27行之前:
因此,源程序的功能就是打印大写彩色的NUIST
第15行设置循环次数为5,即NUIST字符个数,第16、17行分别设置数据段和显存地址偏移量的初始值,第18、19行将数据段中的内容复制到寄存器ax中,并全部字符转换成大写,第20行将寄存器ax中的值送到显存地址空间,第21、22行将数据段后五个字节分别送到显存地址空间中与前五个字符相邻的单元中,用来设置每个字符的颜色。
其中第19行,dfH=1101 1111(B),与ax低地址数据相与,相当于将从右往左第五位都设置成0,如果原先是1,就相当于减小了25=32,如果原先是0,则不变;因此可以将字符都转换成大写。
改变第4行:
这里的数值是用来设置显示字符的颜色。
实验任务6
程序源代码:
1 assume cs:code, ds:data 2 3 data segment 4 db 'Pink Floyd ' 5 db 'JOAN Baez ' 6 db 'NEIL Young ' 7 db 'Joan Lennon ' 8 data ends 9 10 code segment 11 start: 12 mov ax, data 13 mov ds, ax 14 mov bx, 0 15 s:or byte ptr [bx], 20h 16 or byte ptr [bx+1], 20h 17 or byte ptr [bx+2], 20h 18 or byte ptr [bx+3], 20h 19 add bx, 16 20 loop s 21 22 mov ah, 4ch 23 int 21h 24 code ends 25 end start
截图:
反汇编结果:
程序执行后:
实验任务7
源程序代码:
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 bx, 0 20 mov si, 0 21 mov cx, 5 22 23 s0:mov ax, ds:[bx] 24 mov es:[si], ax 25 mov ax, ds:[bx+2] 26 mov es:[si+2], ax 27 add bx, 4 28 add si,10h 29 loop s0 30 31 mov bx, 20 ;跳过年份的20个字节 32 mov si, 5 33 mov cx, 5 34 s1:mov ax, ds:[bx] 35 mov es:[si], ax 36 mov word ptr es:[si+2], 0 ;收入字段的高址部分设为0 37 add bx, 2 38 add si, 10h 39 loop s1 40 41 mov cx, 5 42 mov si, 10 43 mov bx, 30 44 s2:mov ax, ds:[bx] 45 mov es:[si], ax 46 add bx, 2 47 add si, 10h 48 loop s2 49 50 mov si, 0 51 mov cx, 5 52 s3:mov ax, es:[si+5] ;被除数低16位,放在ax中 53 mov dx, es:[si+7] ;高16位,放在dx中 54 div word ptr es:[si+10] 55 mov es:[si+13], ax ;将商移入位置 56 57 add si, 10h 58 loop s3 59 60 mov ah, 4ch 61 int 21h 62 code ends 63 end start
table初始状态:
执行完毕以后: