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

本章实验的主题主要讲代码段、数据段、栈段的使用。

(1)

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	;ss:sp stack

	mov ax, data
	mov ds, ax	;ds data

	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=004h, ss=076bh, ds=076ah
  • 设程序加载后,code段的地址为X,则data段的地址X-2,stack段为X-1

(2)

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	;ss:sp stack

	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

这些程序是以后复杂程序的架子,这个架子从刚开始的简陋到最后的丰满,是一步一步建成的,看看计算机的发展史,就明白了一个道理,不要不在乎自己写的helloword,坚持下去就是牛逼的东西。

这个程序本身没什么意思,在程序执行到最后,观察一下相关寄存器的情况,实验结果如下:


然后可以推知:该程序加载后,code段的段地址为X,则data段的段地址X-2,stack的段地址为X-1


(3)

assume cs:code, ds:data, ss:stack

code segment
start:
	mov ax, stack
	mov ss, ax
	mov sp, 16	;ss:sp stack

	mov ax, data
	mov ds, ax

	push ds:[0]
	push ds:[2]
	pop ds:[2]
	pop ds:[0]

	mov ax, 4c00h
	int 21h

	data segment
	dw 0123h, 0456h
	data ends

	stack segment
		dw 0, 0
	stack ends

code ends
end start

这段代码和上一段对比,就是把data段和stack段放到了最后,看下程序执行返回前(就是int 21h前),相关寄存器的情况:


这时,CS, SS, DS三个寄存器的位置关系发生了变化,和上面两种情况可以对比一下,分别从程序的角度和内存的角度体会栈段的使用、变化情况。


(4)

如果将(1)(2)(3)题中的最后一条伪指令“end start”改为end(也就是说不指明函数的入口),则哪个程序仍然可以正确的执行?

第三个程序,因为不指明程序入口时,cs:codesegment默认ip为0,第三个程序正好是程序开始的地方,前两个ip=0开始的地方存的是数据,解析为汇编指令是错误的。这里也就明白什么时候用end,什么时候用end start,不过所有时刻用end start绝对没错。


(5)

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 1,2,3,4,5,6,7,8
c ends

code segment
start:
	mov ax, a
	mov ds, ax
	mov bx, 0


code ends
end start

先看a,b,c三个数据段是怎么分布的,如下图:


每个段都不到16个字节,但是三个段绝对不是一个接着一个排列,而是另起一行重新开始,也就是说当一个数据段凑不够正行时,那下一段的数据,栈,代码也都是另起一行。

下面按照实验要求的代码如下:

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 8 dup(0)
c ends

code segment
start:
	mov ax, a
	mov ds, ax
	mov bx, 0		;bx, first position of a piece of data

	mov cx, 8
	mov si, 0		;si, loop varibles
	
s:	mov al, 0		;al, temp varible
	add al, [bx+si]
	add al, [bx+si+10h]
	mov [bx+si+10h+10h], al
	inc si
	loop s

	mov ax, 4c00h
	int 21h

code ends
end start

实验结果如下:



(6)

代码如下:

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 8 dup(0)
b ends

code segment
start:
	mov ax, a
	mov ds, ax
	mov bx, 0	;ds:bx, source data

	mov ax, b
	mov es, ax	;es:bx, destination data
			
	mov si, 0	;loop varible
	mov cx, 8
s:	mov ax, ds:[bx+si]
	mov es:[bx+si], ax
	add si, 2	;word type
	loop s



	mov ax, 4c00h
	int 21h
code ends
end start

实验结果:


说明:细细想来,人每天的活动都是按套路做事的,写程序也莫过于与此,不是每一步都要创新,那样效率太慢,也太艰辛了,所以这样导出了学习的一个本质,先学个粗略的模型,先干着,就像学驾照一样,先学几招能把车稳当安全的开走即可,至于漂移也只是更难的套路;回到程序中,ds,es,si这些设置都是套路。



















  • 1
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
好的,我会尽力回答您的问题。 在汇编语言中,程序可以被分为多个,每个都有自己的代码、数据和堆栈。编写调试具有多个程序需要注意以下几点: 1. 定义 在汇编语言中,通过使用ASSUME语句来指定寄存器和逻辑之间的对应关系。例如,下面的代码定义了两个逻辑CODEDATA。 ``` ASSUME CS:CODE, DS:DATA CODE SEGMENT ; 代码 CODE ENDS DATA SEGMENT ; 数据 DATA ENDS ``` 2. 定义变量和常量 在数据中,可以定义变量和常量。例如,下面的代码定义了一个字节型变量和一个字型常量。 ``` DATA SEGMENT MYBYTE BYTE ? MYWORD WORD 1234H DATA ENDS ``` 3. 定义堆栈 在汇编语言中,堆栈是由程序自己管理的。可以使用SS和SP寄存器来访问堆栈。在程序中必须定义堆栈的大小,以便程序运行时可以正确地使用堆栈。例如,下面的代码定义了一个堆栈和堆栈大小。 ``` STACK SEGMENT STACK DW 100H DUP(?) STACK ENDS ``` 4. 编写代码 在代码中,编写实际的程序代码。例如,下面的代码使用MOV指令将一个常量加载到AX寄存器中,并使用INT指令调用中断。 ``` CODE SEGMENT MOV AX, MYWORD INT 21H CODE ENDS ``` 5. 调试程序调试程序时,可以使用调试器来跟踪程序的执行过程。可以使用INT 3指令在程序中插入一个断点,当程序执行到这个断点时,程序会停止执行。例如,下面的代码在程序中插入了一个断点。 ``` CODE SEGMENT MOV AX, MYWORD INT 3 INT 21H CODE ENDS ``` 以上是编写调试具有多个程序所需要注意的几点。在实践中,需要根据具体的程序需求进行相应的调整和修改。总之,多练习、多尝试,才能更好地掌握汇编语言的编程技巧。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值