(九)《汇编语言(王爽)》 | 实验 5:编写、调试具有多个段的程序

2024-01-05 18:08

本文主要是介绍(九)《汇编语言(王爽)》 | 实验 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,stackmov ss,axmov sp,16		;定义一个空栈mov ax,datamov ds,axpush ds:[0]		;入栈push ds:[2]		;入栈pop ds:[2]		;出栈后元素放到数据段data中,偏移地址为2pop ds:[0]		;出栈后元素放到数据段data中,偏移地址为0mov ax,4c00h	;程序返回int 21h
code ends
end start	;start标号处作为程序入口
  • CPU 执行程序,程序返回前,data 段中的数据为 没有变化

end start 指明程序将标号 start 作为程序入口,代码段部分分为以下功能:定义空栈,设置寄存器 DS 的值,两次入栈,两次出栈。由于两次入栈出栈的地址逆序,且都是针对 DS 操作,所以以寄存器 DS 为标识的代码段部分数据没有改变。

  • CPU 执行程序,程序返回前,cs= 076C、ss= 0769、ds= 075A

程序加载后,系统为代码段、数据段和栈段自动分配内存空间,空间的段地址分别由代码段寄存器 CS栈段寄存器 SS数据段寄存器 DS 表示。直接使用 Debug 加载程序后,使用 r 指令查看各段寄存器的内容即可。

请添加图片描述

  • 设程序加载后,code 段的段地址为 X,则 data 段的段地址为 X-2,stack 段的段地址为 X-1。使用 d 指令查看数据段的内容:

请添加图片描述
在 076A:0~076A:F 这段内存存放了数据段,共 16 个字节;栈段的段地址为 076B,如果偏移为零,则可以写作 076A:10,而上图中恰好有连续 10 个字节的零,与定义的栈段内容相符,为证实这一结论,使用 u 指令查看 076A:20 的汇编指令内容:

请添加图片描述
这部分恰好是定义的代码段部分,所以 076A:10~076A:1F 部分是定义的栈段,后面 6 字节系统自动做了填充,以 16 字节对齐。由此得知,系统以各段的定义顺序为各段分配空间,并且空间以 16 字节对齐

(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,stackmov ss,axmov sp,16		;定义一个栈mov ax,datamov ds,axpush ds:[0]		;入栈push ds:[2]		;入栈pop ds:[2]		;出栈后元素放到数据段data中,偏移地址为2pop ds[0]		;出栈后元素放到数据段data中,偏移地址为0mov ax,4c20h	;程序返回int 21h
code ends
end start
  • CPU 执行程序,程序返回前, data 段中的数据为 前 4 个字节存放定义的数据、后面为填充数据

请添加图片描述

  • CPU 执行程序,程序返回前,cs= 076C、ss= 076B、ds=076A
  • 设程序加载后,code 段的段地址为 X,则 data 段的段地址为 X-2,stack 段的段地址为 X-1
  • 对于如下定义的段:
name segment...
name ends

如果段中的数据占 N 个字节,则程序加载后,该段实际占有的空间为 N/16*16+(N%16!=0)*16

前面由实验得知,各段的大小在实际存储时有 16 字节的对齐。所以,当 N 能整除 16 时,该段实际占用空间为 N;当 N 不能整除 16 时,占有的空间向上 16 字节对齐。

(3)编写下述程序并运行,然后用 Debug 加载、跟踪。

assume cs:code,ds:data,ss:stack
code segment	;代码段
start:	mov ax,stackmov ss,axmov sp,16		;定义一个栈mov ax,datamov ds,axpush ds:[0]		;入栈push ds:[2]		;入栈pop ds:[2]		;出栈后元素放到数据段data中,偏移地址为2pop ds:[0]		;出栈后元素放到数据段data中,偏移地址为0mov ax,4c00h	;程序返回int 21h
code ends
data segment	;数据段dw 0123H,0456H
data ends
stack segment	;栈段dw 0,0
stack ends
end start
  • CPU 执行程序,程序返回前, data 段中的数据为

这段程序与上段程序不同的是,数据段、代码段和栈段的定义顺序不同。根据寄存器的段地址可知,各段的存放顺序与定义顺序相同,即先代码段、然后数据段、最后栈段。根据下一题的答案,076A0~076CF 部分存放代码段,076D:0 ~076D:F 部分存放数据段,076E:0~076E:F 部分存放栈段。

请添加图片描述

  • CPU 执行程序,程序返回前,cs= 076A、ss= 076E、ds= 076D

请添加图片描述

  • 设程序加载后,code 段的段地址为 X,则 data 段的段地址为 X+3,stack 段的段地址为 X+4

(4)如果将上面三题中最后一条伪指令 end start 改为 end,则哪个程序仍可以正确执行。

end start 指明程序入口为 start 标号处,没有指明程序入口时程序将从加载的第一条指令开始执行,所以只有(3)能够正确运行。这里也可以看出,代码和数据在内存中的存放形式都是一致的,关键是编写代码的人去控制 CPU 执行指令实为指令的代码、使用实为数据的代码。所以,在编写程序时最好指定程序入口。

(5)编写 code 段中的代码,将 a 段和 b 段中的数据依次相加,将结果存到 c 段中。

assume cs:code
a segment		;段adb 1,2,3,4,5,6,7,8	;使用dw定义时以字为单位,使用db定义时以字节为单位
a ends
b segment		;段bdb 1,2,3,4,5,6,7,8
b ends
c segment		;段cdb 0,0,0,0,0,0,0,0
c ends
code segment	;代码段
start:	?		;待完成部分
code ends
end start
  • 题目要求将两个段 a 和 b 的内容依次相加,将相加结果存入另一个段 c 中,每个段的大小都是 8 个字节。即,考虑首先将段 a 的内容复制到 段 c 中,然后再将段 b 和段 c 的内容相加,在代码部分使用两个循环。
  • 首先,我们需要使用段寄存器指向某个段,段寄存器有 CS、DS、SS、ES,CS 用于代码段的管理、SS 用于栈段的管理,这里使用 DS 和 ES 两个段寄存器。待完成部分:
	mov ax,amov ds,ax	;使用段寄存器DS指向段amov ax,cmov es,ax	;使用段寄存器ES指向段cmov cx,8	;定义循环次数,一共有8个字节的数据mov bx,0	;定义偏移量
s1:	mov al,ds:[bx]	;每次处理1个字节mov es:[bx],alinc bx			;每次偏移1个字节loop s1			;第一个循环完成将段a的内容复制到段cmov ax,b		;移动mov ds,ax		;使用段寄存器DS指向段bmov cx,8		;定义循环次数,一共有8个字节的数据mov bx,0		;定义偏移量
s2:	mov al,ds:[bx]	;每次处理1个字节add es:[bx],al	;相加inc bx			;每次偏移1个字节loop s2			;第二个循环完成段b和段c的内容相加mov ax,4c00h	;程序返回int 21h

先使用指令 g 跳转到 076D:0010 处查看段 a 和 段 c 的内容:

请添加图片描述
执行完第一次循环后查看段 c 的内容:

请添加图片描述
查看段 b 的内容和执行完第二次循环后查看段 c 的内容:

请添加图片描述

(6)编写 code 段中的代码,用 push 指令将 a 段中的 word 数据,逆序存储到 b 段中。

assume cs:code
a segmentdw 1,2,3,4,5,6,7,8
a ends
b segmentdw 0,0,0,0,0,0,0,0
b ends
code segment
start:	?
code ends
end start
  • 题目要求逆序存放,利用栈先进后出的特点即可。由于入栈时栈顶寄存器 SP 的值减小而出栈时 SP 的值增大。所以,直接将段 a 中的数据入栈,同时将段 b 作为栈段,即可将段 a 中的字数据逆序存放到段 b 中。
	mov ax,amov ds,ax	;使用段寄存器DS指向段amov ax,bmov ss,ax	;使用段寄存器SS指向段bmov sp,16	;初始化空栈,SP指向栈顶的下一个位置mov cx,8	;循环次数mov bx,0	;偏移量
s:	push ds:[bx]add bx,2	;每次偏移一个字loop smov ax,4c00h;程序返回int 21h

先使用指令 g 跳转到 076C:0013 处查看循环前段 a 和 段 c 的内容(暂不清楚栈为什么栈段部分不全为零):

请添加图片描述

执行循环后段 c 的内容:

请添加图片描述

这篇关于(九)《汇编语言(王爽)》 | 实验 5:编写、调试具有多个段的程序的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/573708

相关文章

SpringBoot实现微信小程序支付功能

《SpringBoot实现微信小程序支付功能》小程序支付功能已成为众多应用的核心需求之一,本文主要介绍了SpringBoot实现微信小程序支付功能,文中通过示例代码介绍的非常详细,对大家的学习或者工作... 目录一、引言二、准备工作(一)微信支付商户平台配置(二)Spring Boot项目搭建(三)配置文件

使用Python自建轻量级的HTTP调试工具

《使用Python自建轻量级的HTTP调试工具》这篇文章主要为大家详细介绍了如何使用Python自建一个轻量级的HTTP调试工具,文中的示例代码讲解详细,感兴趣的小伙伴可以参考一下... 目录一、为什么需要自建工具二、核心功能设计三、技术选型四、分步实现五、进阶优化技巧六、使用示例七、性能对比八、扩展方向建

Java编译生成多个.class文件的原理和作用

《Java编译生成多个.class文件的原理和作用》作为一名经验丰富的开发者,在Java项目中执行编译后,可能会发现一个.java源文件有时会产生多个.class文件,从技术实现层面详细剖析这一现象... 目录一、内部类机制与.class文件生成成员内部类(常规内部类)局部内部类(方法内部类)匿名内部类二、

基于Flask框架添加多个AI模型的API并进行交互

《基于Flask框架添加多个AI模型的API并进行交互》:本文主要介绍如何基于Flask框架开发AI模型API管理系统,允许用户添加、删除不同AI模型的API密钥,感兴趣的可以了解下... 目录1. 概述2. 后端代码说明2.1 依赖库导入2.2 应用初始化2.3 API 存储字典2.4 路由函数2.5 应

Python实现合并与拆分多个PDF文档中的指定页

《Python实现合并与拆分多个PDF文档中的指定页》这篇文章主要为大家详细介绍了如何使用Python实现将多个PDF文档中的指定页合并生成新的PDF以及拆分PDF,感兴趣的小伙伴可以参考一下... 安装所需要的库pip install PyPDF2 -i https://pypi.tuna.tsingh

如何用java对接微信小程序下单后的发货接口

《如何用java对接微信小程序下单后的发货接口》:本文主要介绍在微信小程序后台实现发货通知的步骤,包括获取Access_token、使用RestTemplate调用发货接口、处理AccessTok... 目录配置参数 调用代码获取Access_token调用发货的接口类注意点总结配置参数 首先需要获取Ac

基于Python开发PDF转Doc格式小程序

《基于Python开发PDF转Doc格式小程序》这篇文章主要为大家详细介绍了如何基于Python开发PDF转Doc格式小程序,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 用python实现PDF转Doc格式小程序以下是一个使用Python实现PDF转DOC格式的GUI程序,采用T

基于.NET编写工具类解决JSON乱码问题

《基于.NET编写工具类解决JSON乱码问题》在开发过程中,我们经常会遇到JSON数据处理的问题,尤其是在数据传输和解析过程中,很容易出现编码错误导致的乱码问题,下面我们就来编写一个.NET工具类来解... 目录问题背景核心原理工具类实现使用示例总结在开发过程中,我们经常会遇到jsON数据处理的问题,尤其是

前端bug调试的方法技巧及常见错误

《前端bug调试的方法技巧及常见错误》:本文主要介绍编程中常见的报错和Bug,以及调试的重要性,调试的基本流程是通过缩小范围来定位问题,并给出了推测法、删除代码法、console调试和debugg... 目录调试基本流程调试方法排查bug的两大技巧如何看控制台报错前端常见错误取值调用报错资源引入错误解析错误

将java程序打包成可执行文件的实现方式

《将java程序打包成可执行文件的实现方式》本文介绍了将Java程序打包成可执行文件的三种方法:手动打包(将编译后的代码及JRE运行环境一起打包),使用第三方打包工具(如Launch4j)和JDK自带... 目录1.问题提出2.如何将Java程序打包成可执行文件2.1将编译后的代码及jre运行环境一起打包2