汇编语言之寄存器(内存访问)2

简介: 汇编语言之寄存器(内存访问)2

7、cpu提供的栈机制

现今的CPU中都有栈的设计。8086CPU提供的入栈和出栈指令,最基本的两个是push(入栈)和pop(出栈)。比如:


Push ax 表示将寄存器ax中的数据送入栈中,pop ax 表示从栈顶取出数据送入ax。8086CPU的入栈和出栈操作都是以字为


单位进行的。


下面两张图描述了push和pop指令的执行过程。

2.png上面两张图指令的执行过程,写成代码如下:


Mov ax, 123H


Push ax


Mov bx, 2266H


Push bx


Mov cx, 1122H


Push cx


Pop ax


Pop bx


Pop cx


注意:字型数据用两个内存单元存放,高地址单元存放高8位,低地址单元存放低8位。


看了上面两张图后,现在提出两个问题。问题1:我们将10000H~1000FH这段内存当作栈来使用,CPU是如何知道这段空


间是栈?关于这个问题将在3.10节解答。


问题2:push ax等入栈指令执行时,要将寄存器中的数据放入当前栈顶单元的上方,成为新的栈顶元素;pop ax等指令执


行时,要从栈顶单元取出数据送入寄存器中。显然,push、pop在执行的时候,CPU必须要知道哪个单元是栈顶单元,可是,如


何知道?


答:8086CPU中,有两个寄存器,堆栈段寄存器SS,堆栈指针寄存器SP,栈顶的段地址存放在SS中,偏移地址存放在


SP中,任意时刻,SS:SP指向栈顶元素,push和pop指令在执行时,CPU从SS和SP中得到栈顶单元的地址。


现在,我们可以完整地描述push和pop指令的功能了,例如:push ax。push ax的执行,由以下两步完成:


1. SP=SP-2,SS:SP指向当前栈顶前面的单元,以当前栈顶前面的单元为新的栈顶。


2. 将ax中的数据送入SS:SP指向的内存单元处,SS:SP此时指向新栈顶。


下图描述了push ax的执行过程。


3.png

3.png

将10000H~1000FH这段空间当作栈段,SS=1000H,栈空间大小为16字节,栈最底部的字单元地址为1000:000E,任


意时刻,SS:SP指向栈顶,当栈中只有一个元素的时候,SS=1000H,SP=000EH,栈为空,就相当于栈中唯一的元素出栈后,


SP=SP+2,原来为000EH,加2后SP=0010H,所以,当栈为空的时候,SS=1000H,SP=0010H。


换一个角度看,任意时刻,SS:SP指向栈顶元素,当栈为空的时候栈中没有元素,也就不存在栈顶元素,所以SS:SP只能指


向栈的最底部单元下面的单元,该单元的偏移地址为栈最底部的字单元的偏移地址+2,栈最底部字单元的地址为1000:000E,


所以,栈空时,SP=0010H。


接下来,我们描述pop指令的功能,例如:pop ax。Pop ax的执行过程和push ax刚好相反,由以下两步完成:


1. 将SS:SP指向的内存单元处的数据送入ax中。


2. SP=SP+2,SS:SP指向当前栈顶下面的单元,以当前栈顶下面的单元为新的栈顶。


下图描述了pop ax的执行过程。


4.png

4.png

注意:上图中,出栈后,SS:SP指向新的栈顶1000EH,pop操作前的栈顶元素,1000CH处的数据2266H依然存在,但


是,它已不在栈中,当再次执行push等入栈指令后,SS:SP移至1000CH,并在里面写入新的数据,将它覆盖。


8、push指令 pop指令

Push和pop指令的格式有如下两种形式:


第一种形式:push 寄存器    ;将一个寄存器中的数据入栈。


           Pop 寄存器    ;出栈,用一个寄存器接收出栈的数据。


这一种形式,它们可以在栈和寄存器之间传送数据。


注意:上面的寄存器可以是段寄存器,比如,可以是:push ds、pop ds。


第二种形式:push 内存单元    ;将一个内存单元中的字型数据入栈(注意,栈操作都是以字为单位)。


           Pop 内存单元    ;出栈,用一个内存字单元接收出栈的数据。


比如:mov ax, 1000H


     Mov ds, ax    ;将内存单元的段地址放在DS中。


     Push [2a38]    ;将内存单元[2a38]中的字型数据入栈。


     Pop [2a3a]    ;出栈,出栈的数据送入内存单元[2a3a]。


这一种形式,它们可以在栈和内存单元之间传送数据。


指令执行时,CPU要知道内存单元的地址,可以在push、pop指令中给出内存单元的偏移地址,段地址在指令执行时,CPU


从DS中取得。


Push和pop实质上是一种内存传送指令,与mov指令不同的是,push和pop指令访问的栈空间的地址不是在指令中给出


的,而是由SS:SP指出的。同时,push和pop指令还要改变SP中数值。Mov指令只需一步操作,就是传送,而执行push、


Pop指令需要两步操作,执行push时,先改变SP,后向SS:SP处传送;执行pop时,先读取SS:SP处的数据,后改变SP。


9 、 栈段

前面讲过(参见2.7节),对于8086PC机,在编程时,可以根据需要,将一组内存单元定义为一个段。我们可以将长度为N


(N≤64KB)的一组地址连续、起始地址为16的倍数的内存单元,当作栈空间来用,从而定义了一个栈段。


比如,我们将10010H~1001FH这段内存空间当作栈来用,以栈的方式进行访问,这段空间就可以认为是一个栈段,大小为


16个字节。


如何使得如push、pop等栈操作指令访问我们定义的栈段呢?那就是要将SS:SP指向我们定义的栈段。


现在我们来回答3.8节中的第一个问题。答:只要这段内存单元被SS:SP指向,那么,CPU就会把这段空间当作栈来使用。


相关文章
|
2月前
|
传感器 人工智能 物联网
C 语言在计算机科学中尤其在硬件交互方面占据重要地位。本文探讨了 C 语言与硬件交互的主要方法,包括直接访问硬件寄存器、中断处理、I/O 端口操作、内存映射 I/O 和设备驱动程序开发
C 语言在计算机科学中尤其在硬件交互方面占据重要地位。本文探讨了 C 语言与硬件交互的主要方法,包括直接访问硬件寄存器、中断处理、I/O 端口操作、内存映射 I/O 和设备驱动程序开发,以及面临的挑战和未来趋势,旨在帮助读者深入了解并掌握这些关键技术。
74 6
|
1月前
|
存储 缓存 数据安全/隐私保护
DMA(Direct Memory Access):直接内存访问
DMA(Direct Memory Access)是一种允许外设直接与内存进行数据传输的技术,无需 CPU 干预。它通过减轻 CPU 负担、提高数据传输效率来提升系统性能。DMA 的工作模式包括直接模式和 FIFO 模式,数据传输方式有单字传送和块传送,寻址模式有增量寻址和非增量寻址。通过缓存一致性协议、同步机制、数据校验和合理的内存管理,DMA 确保了数据在内存中的一致性和完整性。
81 0
|
2月前
|
存储 编译器 Linux
【c++】类和对象(上)(类的定义格式、访问限定符、类域、类的实例化、对象的内存大小、this指针)
本文介绍了C++中的类和对象,包括类的概念、定义格式、访问限定符、类域、对象的创建及内存大小、以及this指针。通过示例代码详细解释了类的定义、成员函数和成员变量的作用,以及如何使用访问限定符控制成员的访问权限。此外,还讨论了对象的内存分配规则和this指针的使用场景,帮助读者深入理解面向对象编程的核心概念。
201 4
|
3月前
|
缓存 算法 数据处理
如何选择合适的内存访问模式
【10月更文挑战第20天】如何选择合适的内存访问模式
74 1
|
3月前
|
存储 容器
内存越界访问(Out-of-Bounds Access)
【10月更文挑战第12天】
375 2
|
3月前
|
Rust 编译器
|
2月前
|
缓存 Prometheus 监控
Elasticsearch集群JVM调优设置合适的堆内存大小
Elasticsearch集群JVM调优设置合适的堆内存大小
515 1
|
1月前
|
存储 监控 算法
深入探索Java虚拟机(JVM)的内存管理机制
本文旨在为读者提供对Java虚拟机(JVM)内存管理机制的深入理解。通过详细解析JVM的内存结构、垃圾回收算法以及性能优化策略,本文不仅揭示了Java程序高效运行背后的原理,还为开发者提供了优化应用程序性能的实用技巧。不同于常规摘要仅概述文章大意,本文摘要将简要介绍JVM内存管理的关键点,为读者提供一个清晰的学习路线图。
|
2月前
|
Java
JVM内存参数
-Xmx[]:堆空间最大内存 -Xms[]:堆空间最小内存,一般设置成跟堆空间最大内存一样的 -Xmn[]:新生代的最大内存 -xx[use 垃圾回收器名称]:指定垃圾回收器 -xss:设置单个线程栈大小 一般设堆空间为最大可用物理地址的百分之80
|
2月前
|
Java
JVM运行时数据区(内存结构)
1)虚拟机栈:每次调用方法都会在虚拟机栈中产生一个栈帧,每个栈帧中都有方法的参数、局部变量、方法出口等信息,方法执行完毕后释放栈帧 (2)本地方法栈:为native修饰的本地方法提供的空间,在HotSpot中与虚拟机合二为一 (3)程序计数器:保存指令执行的地址,方便线程切回后能继续执行代码
29 3