检测点6.1(1)依次用内存0:0~15单元中的内容改写程序中的数据,补全程序:assumecs:codesgcodesegmentdw0123h,0456h,0789h,0ab
检测点6.1
(1)依次用内存0:0~15单元中的内容改写程序中的数据,补全程序:
assume cs:codesg
code segment
dw 0123h,0456h,0789h,0abch,0defh,0fedh,0cbah,0987h
start:
mov ax,0
mov ds,ax
mov bx,0
mov cx,8
s: mov ax,[bx]
mov cs:[bx],ax ;确定目标区域段地址和偏移地址
add bx,2
loop s
mov ax,4c00h
int 21h
codesg ends
end start
(2)程序实现依次用0:0~15单元的内容改写程序中数据,数据传送用栈来进行。栈空间设置在程序内,补全程序:
assume cs:codesg
code segment
dw 0123h,0456h,0789h,0abch,0defh,0fedh,0cbah,0987h
dw 0,0,0,0,0,0,0,0,0,0 ;10个字空间作为栈
start:
mov ax,cs ;设置栈的段地址
mov ss,ax
mov sp,1ah ;cs:0~cs:f为原始数据空间,cs:10~cs:19为栈空间,初始为空,栈顶指向下一单元
mov ax,0
mov ds,ax
mov bx,0
mov cx,8
s: push [bx] ;先把0:0处的字单元内容入栈
pop cs:[bx] ;再把栈顶内容出栈放入程序数据段中
add bx,2
loop s
mov ax,4c00h
int 21h
codesg ends
end start
实验5 编写、调试具有多个段的程序
<程序加载后,ds:0~ff为PSP区域,(ds+10H):0为整个程序的入口,如程序依序设置有data\stack\code3个数据段区域,其中设data和stack段各自为16个字节,那么程序加载后(还未运行前):(ds+10h)则为data的入口段地址;(ds+10h+1)为stack的入口段地址;(ds+10h+2)为code的入口段地址;>
1、编译连接下面程序,用debug加载、跟踪,回答问题。
assume cs:codesg,ds:data,ss:stack
data segment
dw 0123h,0456h,0789h,0abch,0defh,0fedh,0cbah,0987h
data ends
stack segment
dw 0,0,0,0,0,0,0,0
stack ends
code segment
start:
mov ax,stack
mov ss,ax
mov sp,16 ;将逻辑上设置的栈段真正设为栈段区域
mov ax,data ;设置数据区域段地址
mov ds,ax
push ds:[0]
push ds:[2]
pop ds:[2]
pop ds:[0]
mov ax,4c00h
int 21h
code ends
end start
程序先把数据段前两个依序入栈,再出栈返回到各数据原先位置。
(1)CPU执行程序,程序返回前,data段中的数据为多少?
执行程序后,data段中的数据不变,为原始数据。
(2)CPU执行程序,程序返回前,cs=13feh,ss=13fdh,ds=13fch.
(3)程序加载后,code段地址设为X,则data段地址为(x-2),stack段的段地址为(X-1).
注:这里提问的是data段和stack段的段地址分别是多少,程序加载时候,还没有运行,此时的data段只是逻辑上的数据区,并还没有设置ds指向;stack也如此,只是设想里的栈区。只有程序运行后才成为真正的ds指向数据区和栈段。通过(2)也可以验证(3)。
2、编译连接下面程序,用debug加载、跟踪,回答问题。
assume cs:code,ds:data,ss:stack
data segment
dw 0123h,0456h
data ends
stack segment
dw 0,0
stack ends
code segment
start:
mov ax,stack
mov ss,ax
mov sp,16
mov ax,data
mov ds,ax
push ds:[0]
push ds:[2]
pop ds:[2]
pop ds:[0]
mov ax,4c00h
int 21h
code ends
end start
本题的重点在于:数据段和栈段在程序加载后实际占据的空间都是以16个字节为单位的。程序中只给出了前两个字数据,其余空间都用0填充。
(1)CPU执行程序,程序返回前,data段中的数据为多少?
执行程序后,data段有16个字节空间,前两个字数据不变,其余为0。
(2)CPU执行程序,程序返回前,cs=13feh,ss=13fdh,ds=13fch.
(3)程序加载后,code段地址设为X,则data段地址为(x-2),stack段的段地址为(X-1).
(4)对于如下定义的段:
name segment
......
name ends
如果段中数据位N个字节,程序加载后,该段实际占据空间为[N/16]Byte。
3、编译连接下面程序,用debug加载、跟踪,回答问题。
assume cs:code,ds:data,ss:stack
code segment
start:
mov ax,stack
mov ss,ax
mov sp,16
mov ax,data
mov ds,ax
push ds:[0]
push ds:[2]
pop ds:[2]
pop ds:[0]
mov ax,4c00h
int 21h
code ends
data segment
dw 0123h,0456h
data ends
stack segment
dw 0,0
stack ends
end start
本题考查重点是三个段的顺序调换后是否有变化?在加载后各数据段的入口段地址必然是改变了。
(1)CPU执行程序,程序返回前,data段中的数据为多少?
运行程序至程序返回前,通过:d ds:0~f查看data段数据,观察未变。
(2)CPU执行程序,程序返回前,cs=13fch,ss=1400h,ds=13ffh.
(3)程序加载后,code段地址设为X,则data段地址为(x+3),stack段的段地址为(X+4).
4、如果不指名start入口,并且使用end替换end start,程序仍然可以执行。因为如果不指名入口,程序则从加载进内存的第一个单元起开始执行,但因为程序中有部分是作为数据使用的,如果不指明入口,CPU会把这些数值数据当成汇编指令执行,因此有必要通过start来指明入口。
5、编写code段中的代码,将a段和b段数据依次相加,结果存入c段
assume cs:code
a segment
db 1,2,3,4,5,6,7,8
a ends
b segment
db 1,2,3,4,5,6,7,8
b ends
c segment
db 0,0,0,0,0,0,0,0
c ends
code segment
start:
mov ax,a
mov ds,ax
mov ax,b
mov ss,ax
mov ax,c
mov es,ax
mov bx,0
mov cx,8
s:
mov al,ds:[bx]
add al,ss:[bx]
mov es:[bx],al
inc bx
loop s
mov ax,4c00h
int 21h
code ends
end start
6、编写code段中代码,用push指令将a段中前8个字型数据逆序存储到b段中。
assume cs:code
a segment
dw 1,2,3,4,5,6,7,8,9,0ah,0bh,0ch,0dh,0eh,0fh,0ffh
a ends
b segment
dw 0,0,0,0,0,0,0,0
b ends
code segment
start:
mov ax,a
mov ds,ax
mov ax,b
mov ss,ax
mov sp,10h
mov bx,0
mov cx,8
s:
push ds:[bx]
add bx,2
loop s
mov ax,4c00h
int 21h
code ends
end start
注:5、6题要巩固练习通过用debug命令检查内存来看程序是否执行准确。