6.1 在代码段中使用数据
实验5 编写、调试具有多个段的程序
(1)将下面的程序编译、连接,用 Debug 加载、跟踪,然后回答问题
assume cs:code,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
CPU 执行程序,程序返回前,data 段中的数据为多少?(不变)
CPU 执行程序,程序返回前,cs = ss = ds = 。(cs=076E, ss=076D, ds=076C)
设程序加载后, code 段的段地址为 X, 则 data 段的段地址为 , stack 段的段地址为 __。(data=X-2, stack=X-1)
(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
CPU 执行程序,程序返回前,data 段中的数据为多少?(不变)
CPU 执行程序,程序返回前,cs = ss = ds = 。(cs=076E, ss=076D, ds=076C)
设程序加载后, code 段的段地址为 X, 则 data 段的段地址为 , stack 段的段地址为 __。(data=X-2, stack=X-1)
对于如下定义的段:
name segment
...
name ends
如果段中的数据占 N 个字节,则程序加载后,该段实际占有的空间为 __。((N+15)/16)*16
讨论出处 因为每个段都是以16字节来对齐的,但是最大不能超过64KB 。这也就是说,如果你的段数据在16字节内,一样会被当做一个字节段来算,就是16字节。 要是大于16字节呢,那么如果多出就算是1个字节,因为已经超过一个16字节,多出来的1个字节 也得有一个字节段的容量来存储它,所以这时得占两个字节段,就是32字节,以此类推! (n/16 + 1 )16 详细解释下:n/16 : / 这个符号是取整,就是看有几个段。小于或等于16的 除以16后,小于16取整后等于0,等于16取整后无余数还是等于0,都是等于 0,(0+1)16 = 16字节,一个段的大小。大于16,除以后取整有余数,取整部分为多少个字节段,剩余的部分再需要一个16字节段来存储它。所以最终得到此公式:(n/16+1)16 N分为被16整除和不被16整除。 当N被16整除时: 占有的空间为(N/16)16 当N不被16整除时: 占有的空间为(N/16+1)16,N/16得出的是可以整除的部分,还有一个余数,余数肯定小于16,加上一个16。 两种情况总结成一个通用的公式:((N+15)/16)16
(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
CPU 执行程序,程序返回前,data 段中的数据为多少?(不变)
CPU 执行程序,程序返回前,cs = ss = ds = 。(cs=076C, ss=0770, ds=076F)
设程序加载后, code 段的段地址为 X, 则 data 段的段地址为 , stack 段的段地址为 __。(data=X+3, stack=X+4)
通过-d cs:30
就可以访问到date段的数据,说明代码段占用了48个字节
(4)如果将(1) (2) (3)中的最后一条伪指令 end start 改为 end (也就是说,不指明程序的入口),则哪个程序仍然可以正确执行?请说明原因。**end start**
改为 **end **
后,**cs:ip**
指向程序的首地址,只有 (3) 中的程序加载后 cs:ip 指向的是代码段
(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 ;ds,a
mov ax,b
mov es,ax
mov bx,0
mov cx,8
s:mov al,[bx]
add al,es:[bx]
mov ds:[bx+32],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 cx,8
mov bx,0
s: mov ax,[bx]
push ax
add bx,2
loop s
code ends
end start