6.1 在代码段中使用数据

程序框架
image.png

实验5 编写、调试具有多个段的程序

(1)将下面的程序编译、连接,用 Debug 加载、跟踪,然后回答问题

  1. assume cs:code,ds:data,ss:stack
  2. data segment
  3. dw 0123h,0456h,0789h,0abch,0defh,0fedh,0cbah,0987h
  4. data ends
  5. stack segment
  6. dw 0,0,0,0,0,0,0,0
  7. stack ends
  8. code segment
  9. start: mov ax,stack
  10. mov ss,ax
  11. mov sp,16
  12. mov ax,data
  13. mov ds,ax
  14. push ds:[0]
  15. push ds:[2]
  16. pop ds:[2]
  17. pop ds:[0]
  18. mov ax,4c00h
  19. int 21h
  20. code ends
  21. end start

image.png
image.png
image.png
image.png
image.png
image.png
image.png
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

image.png
image.png
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

image.png
image.png
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