段寄存器和8种地址寻址方式

摘要:
通常,默认数据段寄存器为DS,但有一个例外,即在执行字符串操作时,其目标地址的段寄存器被指定为ES。在某些字符串操作中,另一个例外是目标操作数的段寄存器指定为ES,FS和GS在访问数据时也可以用作段寄存器,但它们必须以段超过前缀的方式直接写入指令中。

段寄存器是因为对内存的分段管理而设置的。

16位CPU有四个段寄存器,其程序可同时访问四个不同含义的段,引用方面有如下规定:
1. 取命令:段寄存器CS指向存放程序的内存段,IP是用来存放下条待执行的指令在该段的偏移量,把它们合在一起可在该内存段内取到下次要执行的指令。
2. 取堆栈:段寄存器SS指向用于堆栈的内存段,SP是用来指向该堆栈的栈顶,把它们合在一起可访问栈顶单元。另外,当偏移量用到了指针寄存器BP,则其缺省的段寄存器也是SS,并且用BP可访问整个堆栈,不仅仅是只访问栈顶。
3. 取数据:段寄存器DS指向数据段,ES指向附加段,在存取操作数时,二者之一和一个偏移量合并就可得到存储单元的物理地址。该偏移量可以是具体数值、符号地址和指针寄存器的值等之一,具体情况将由指令的寻址方式来决定。通常,缺省的数据段寄存器是DS,只有一个例外,即:在进行串操作时,其目的地址的段寄存器规定为ES。
4. 其它情况,段寄存器除了其默认引用的寄存器外,还可以强行改变为其它段寄存器。

32位CPU内有6个段寄存器,程序在某一时刻可访问6个不同的段。其段寄存器的值在不同的方式下具有不同的含义:
1. 代码段寄存器:32位微机在取指令时,系统自动引用CS和EIP来取出下条指令。
2. 堆栈段寄存器:32位微机在访问堆栈段时,总是引用堆栈段寄存器SS。堆栈指针可用32位的ESP和16位的SP。
3. 数据段寄存器:DS是主要的数据段寄存器。通常情况下,它是除访问堆栈以外数据时的默认段寄存器。在某些串操作中,其目的操作数的段寄存器被指定为ES是另一个例外。
段寄存器CS、SS、ES、FS和GS也都可以作为访问数据时的段寄存器(原来是DS),但它们必须用段超越前缀的方式在指令中直接写出。用这种方式会增加指令的长度,指令的执行时间也有所延长。
一般来说,程序频繁访问的数据段用DS来指向,不太经常访问的数据段可用ES、FS和GS等来指向。


--------------------------------------操作数的寻址方式------------------------------------------
1. 立即寻址方式(操作数作为指令的一部分而直接写在指令中,这种操作数称为立即数)
MOV AH, 80H
ADD AX, 1234H

2. 寄存器寻址方式(指令所要的操作数已存储在某寄存器中,或把目标操作数存入寄存器)
ADD VARD, EAX (源操作数是寄存器寻址方式)
ADD BH, 78h (目的操作数是寄存器寻址方式)
MOV EAX, EBX (源和目的操作数都是寄存器寻址方式)

3. 直接寻址方式(指令所要的操作数存放在内存中,在指令中直接给出该操作数的有效地址)
MOV BX, [1234H],在执行时,(DS)=2000H,内存单元21234H的值为5213H。
执行该指令要分三部分:
1). 由于1234H是一个直接地址,它紧跟在指令的操作码之后,随取指令而被读出;
2). 访问数据段的段寄存器是DS,所以,用DS的值(左移4位)和偏移量1234H相加,得存储单元的物理地址:21234H;
3). 取单元21234H的值5213H,并按“高高低低”的原则存入寄存器BX中。
所以,在执行该指令后,BX的值就为5213H。
MOV ES:[1000H], AX (认为DS,如果要指定访问其它段内的数据,可在指令中用段前缀的方式显式地书写出来)

4. 寄存器间接寻址方式(用SI、DI和BX等之一来指定,则其缺省的段寄存器为DS;用BP来指定,则其缺省的段寄存器为SS)
MOV BX,[DI],在执行时,(DS)=1000H,(DI)=2345H,存储单元12345H的内容是4354H。
执行结果 PA=(DS)*16+DI=1000H*16+2345H=12345H。该指令的执行效果是:把从物理地址为12345H开始的一个字的值传送给BX。

5. 寄存器相对寻址方式(同上,但增加一个偏移量)
MOV BX, [SI+100H],在执行它时,(DS)=1000H,(SI)=2345H,内存单元12445H的内容为2715H
EA=(SI)+100H=2345H+100H=2445H
PA=(DS)*16+EA=1000H*16+2445H=12445H
该指令的执行效果是:把从物理地址为12445H开始的一个字的值传送给BX。

6. 基址加变址寻址方式(效地址是一个基址寄存器(BX、BP)和一个变址寄存器(SI、DI)的内容之和)
MOV BX, [BX+SI],在执行时,(DS)=1000H,(BX)=2100H,(SI)=0011H,内存单元12111H的内容为1234H
EA=(BX)+(SI)=2100H+0011H=2111H
PA=(DS)*16+EA=1000H*16+2111H=12111H
该指令的执行效果是:把从物理地址为12111H开始的一个字的值传送给BX。

7. 相对基址加变址寻址方式(同上,但增加一个偏移量)
MOV AX, [BX+SI+200H],在执行时,(DS)=1000H,(BX)=2100H,(SI)=0010H,内存单元12310H的内容为1234H
EA=(BX)+(SI)+200H=2100H+0010H+200H=2310H
PA=(DS)*16+EA=1000H*16+2310H=12310H
该指令的执行效果是:把从物理地址为12310H开始的一个字的值传送给AX。
下面四种书写方式等价:
MOV AX, [BX+SI+1000H]
MOV AX, 1000H[BX+SI]
MOV AX, 1000H[BX][SI]
MOV AX, 1000H[SI][BX]

8. 32位地址的寻址方式
在用16位寄存器来访问存储单元时,只能使用基地址寄存器(BX和BP)和变址寄存器(SI和DI)来作为地址偏移量的一部分,但在用32位寄存器寻址时,不存在上述限制,所有32位寄存器(EAX、EBX、ECX、EDX、ESI、EDI、EBP和ESP)都可以是地址偏移量的一个组成部分。
当用32位地址偏移量进行寻址时,内存地址的偏移量可分为三部分:一个32位基址寄存器,一个可乘1、2、4或8的32位变址寄存器,一个8位/32位的偏移常量,并且这三部分还可进行任意组合,省去其中之一或之二。

EA = 基址寄存器 + 变址寄存器*比例因子(1或2或4或8)+偏移常量(无/8位/32位)
疑问:这里没有出现段寄存器。难道32位情况下,段寄存器的值是0,或者是什么值都不重要。

32位基址寄存器是:EAX、EBX、ECX、EDX、ESI、EDI、EBP和ESP;
32位变址寄存器是:EAX、EBX、ECX、EDX、ESI、EDI和EBP(除ESP之外)

由于32位寻址方式能使用所有的通用寄存器,所以,和该有效地址相组合的段寄存器也就有新的规定。具体规定如下:
1). 地址中寄存器的书写顺序决定该寄存器是基址寄存器,还是变址寄存器;
如:[EBX+EBP]中的EBX是基址寄存器,EBP是变址寄存器,而[EBP+EBX]中的EBP是基址寄存器,EBX是变址寄存器;
2). 默认段寄存器的选用取决于基址寄存器;
3). 基址寄存器是EBP或ESP时,默认的段寄存器是SS,否则,默认的段寄存器是DS;
4). 在指令中,如果使用段前缀的方式,那么,显式段寄存器优先。

指令的举例 访问内存单元所用的段寄存器
MOV AX, [123456H] ;默认段寄存器DS
MOV EAX, [EBX+EBP] ;默认段寄存器DS
MOV EBX, [EBP+EBX] ;默认段寄存器SS
MOV EBX, [EAX+100H] ;默认段寄存器DS
MOV EDX, ES:[EAX*4+200H] ;显式段寄存器ES
MOV [ESP+EDX*2], AX ;默认段寄存器SS
MOV EBX, GS:[EAX+EDX*2+300H] ;显式段寄存器GS
MOV AX, [ESP] ;默认段寄存器SS

免责声明:文章转载自《段寄存器和8种地址寻址方式》仅用于学习参考。如对内容有疑问,请及时联系本站处理。

上篇基于Locust、Tsung的百万并发秒杀压测案例[转]c#泛型作为返回类型的写法下篇

宿迁高防,2C2G15M,22元/月;香港BGP,2C5G5M,25元/月 雨云优惠码:MjYwNzM=

相关文章

ARM寄存器

ARM寄存器 ARM处理器模式用户模式(User):ARM处理器正常的程序执行状态;快速中断模式(FIQ):用于高速数据传输或通道处理;外部中断模式(IRQ):用于通用的中断处理;管理模式(Supervisor):操作系统使用的保护模式;数据访问终止模式(Abort):当数据或指令预取终止时进入该模式,可用于虚拟存储及存储保护;系统模式(System):运...

Java进程CPU100%的问题

今天review了一下测试环境,后来发现一个java进程CPU 120%多,4核的CPU,很多人都没感觉。但是确实是很大问题。测试环境没什么并发,也没什么数据量怎么会这么高的cpu呢? 找到java进程中耗cpu最高的nid, top –p pid –H 用jstack或者其他方式打印一下线程堆栈,从堆栈记录里找出nid,对应的线程和他的堆栈。找到出问题...

Java虚拟机介绍

Java虚拟机概述 Java虚拟机(JavaVirtualMachine)简称JVMJava虚拟机是一个想象中的机器,在实际的计算机上通过软件模拟来实现。Java虚拟机有自己想象中的硬件,如处理器、堆栈、寄存器等,还具有相应的指令系统。下面我们就来看一下这几部分比较重要的java虚拟机的结构 一、JVM寄存器 所有的CPU均包含用于保存系统状...

MODBUS串行通信协议详细说明

第一章  简介 本文详细地描述了装置在MODBUS 通讯模式下的输入和输出命令、信息和数据,以便第三方使用和开发。 1.1 串行通讯协议的目的 通信协议的作用是使信息和数据在上位机主站和装置之间有效地传递,它包括: (1)   允许主站访问和设定所接装置的全部设置参数; (2) 允许访问装置的所有测量数据。     第二章  装置 MODBUS串行通信协议...

堆栈、堆、方法区介绍

堆栈、堆、方法区介绍 预备知识java数据类型: 基础数据类型:boolean、byte、short、char、int、long、float、double 引用数据类型:类、接口、数组 堆栈、堆、方法区JAVA的JVM的内存可分为3个区:堆(heap)、堆栈(stack)和方法区(method) 堆区: 提供所有类实例和数组对象存储区域 jvm只有一个堆区...

C++ 术语(C++ Primer)

argument(实参):传递给被调用函数的值。block(块):花括号括起来的语句序列。buffer(缓冲区):一段用来存放数据的存储区域。IO 设备常存储输入(或输出)到缓冲区,并独立于程序动作对缓冲区进行读写。输出缓冲区通常必须显式刷新以强制输出缓冲区内容。默认情况下,读 cin 会刷新 cout;当程序正常结束时,cout 也被刷新。built-i...