物理地址和虚拟地址(二)

Windows 2000 使用基于分页机制的虚拟内存。每个进程有4GB的虚拟地址空间。基于分页机制,这4GB地址空间的一些部分被映射了物理内存,一些部分映射硬盘上的交换文件,一些部分什么也没有映射。程序中使用的都是4GB地址空间中的虚拟地址。而访问物理内存,需要使用物理地址。

 

下面我们看看什么是物理地址,什么是虚拟地址。

 

物理地址 (physical address): 放在寻址总线上的地址。放在寻址总线上,如果是读,电路根据这个地址每位的值就将相应地址的物理内存中的数据放到数据总线中传输。如果是写,电路根据这个地址每位的值就将相应地址的物理内存中放入数据总线上的内容。物理内存是以字节(8位)为单位编址的。

 

虚拟地址 (virtual address): 4G虚拟地址空间中的地址,程序中使用的都是虚拟地址

 

如果CPU寄存器中的分页标志位被设置,那么执行内存操作的机器指令时,CPU会自动根据页目录和页表中的信息,把虚拟地址转换成物理地址,完成该指令。比如 mov eax,004227b8h ,这是把地址004227b8h处的值赋给寄存器的汇编代码,004227b8这个地址就是虚拟址。CPU在执行这行代码时,发现寄存器中的分页标志位已经被设定,就自动完成虚拟地址到物理地址的转换,使用物理地址取出值,完成指令。对于Intel CPU 来说,分页标志位是寄存器CR0的第31位,为1表示使用分页,为0表示不使用分页。对于初始化之后的 Win2k 我们观察 CR0 ,发现第31位为1。表明Win2k是使用分页的。

 

使用了分页机制之后,4G的地址空间被分成了固定大小的页,每一页或者被映射到物理内存,或者被映射到硬盘上的交换文件中,或者没有映射任何东西。对于一般程序来说,4G的地址空间,只有一小部分映射了物理内存,大片大片的部分是没有映射任何东西。物理内存也被分页,来映射地址空间。对于32bit的Win2k,页的大小是4K字节。CPU用来把虚拟地址转换成物理地址的信息存放在叫做页目录和页表的结构里。

 

 物理内存分页,一个物理页的大小为4K字节,第0个物理页从物理地址 0x00000000 处开始。由于页的大小为4KB,就是0x1000字节,所以第1页从物理地址 0x00001000 处开始。第2页从物理地址 0x00002000 处开始。可以看到由于页的大小是4KB,所以只需要32bit的地址中高20bit来寻址物理页。

 

 页表,一个页表的大小为4K字节,放在一个物理页中。由1024个4字节的页表项组成。页表项的大小为4个字节(32bit),所以一个页表中有1024个页表项。页表中的每一项的内容(每项4个字节,32bit)高20bit用来放一个物理页的物理地址,低12bit放着一些标志。

 

页目录,一个页目录大小为4K字节,放在一个物理页中。由1024个4字节的页目录项组成。页目录项的大小为4个字节(32bit),所以一个页目录中有1024个页目录项。页目录中的每一项的内容(每项4个字节)高20bit用来放一个页表(页表放在一个物理页中)的物理地址,低12bit放着一些标志。

 

对于x86系统,页目录的物理地址放在CPU的CR3寄存器中。

 

CPU把虚拟地址转换成物理地址:

 一个虚拟地址,大小4个字节(32bit),包含着找到物理地址的信息,分为3个部分:第22位到第31位这10位(最高10位)是页目录中的索引,第12位到第21位这10位是页表中的索引,第0位到第11位这12位(低12位)是页内偏移。对于一个要转换成物理地址的虚拟地址,CPU首先根据CR3中的值,找到页目录所在的物理页。然后根据虚拟地址的第22位到第31位这10位(最高的10bit)的值作为索引,找到相应的页目录项(PDE,page directory entry),页目录项中有这个虚拟地址所对应页表的物理地址。有了页表的物理地址,根据虚拟地址的第12位到第21位这10位的值作为索引,找到该页表中相应的页表项(PTE,page table entry),页表项中就有这个虚拟地址所对应物理页的物理地址。最后用虚拟地址的最低12位,也就是页内偏移,加上这个物理页的物理地址,就得到了该虚拟地址所对应的物理地址。

 

一个页目录有1024项,虚拟地址最高的10bit刚好可以索引1024项(2的10次方等于1024)。一个页表也有1024项,虚拟地址中间部分的10bit,刚好索引1024项。虚拟地址最低的12bit(2的12次方等于4096),作为页内偏移,刚好可以索引4KB,也就是一个物理页中的每个字节。

 

 一个虚拟地址转换成物理地址的计算过程就是,处理器通过CR3找到当前页目录所在物理页,取虚拟地址的高10bit,然后把这10bit右移2bit(因为每个页目录项4个字节长,右移2bit相当于乘4)得到在该页中的地址,取出该地址处PDE(4个字节),就找到了该虚拟地址对应页表所在物理页,取虚拟地址第12位到第21位这10位,然后把这10bit右移2bit(因为每个页表项4个字节长,右移2bit相当于乘4)得到在该页中的地址,取出该地址处的PTE(4个字节),就找到了该虚拟地址对应物理页的地址,最后加上12bit的页内偏移得到了物理地址。

 

32bit的一个指针,可以寻址范围0x00000000-0xFFFFFFFF,4GB大小。也就是说一个32bit的指针可以寻址整个4GB地址空间的每一个字节。一个页表项负责4K的地址空间和物理内存的映射,一个页表1024项,也就是负责1024*4k=4M的地址空间的映射。一个页目录项,对应一个页表。一个页目录有1024项,也就对应着1024个页表,每个页表负责4M地址空间的映射。1024个页表负责1024*4M=4G的地址空间映射。一个进程有一个页目录。所以以页为单位,页目录和页表可以保证4G的地址空间中的每页和物理内存的映射。

 

每个进程都有自己的4G地址空间,从 0x00000000-0xFFFFFFFF 。通过每个进程自己的一套页目录和页表来实现。由于每个进程有自己的页目录和页表,所以每个进程的地址空间映射的物理内存是不一样的。两个进程的同一个虚拟地址处(如果都有物理内存映射)的值一般是不同的,因为他们往往对应不同的物理页。

 

4G地址空间中低2G,0x00000000-0x7FFFFFFF 是用户地址空间,4G地址空间中高2G, 0x80000000-0xFFFFFFFF 是系统地址空间。访问系统地址空间需要程序有ring0的权限。

 

http://blog.csdn.net/tony821224/archive/2008/02/25/2118100.aspx

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: C语言中的物理地址虚拟地址是两个不同的概念。 物理地址是指计算机中实际的内存地址,通常是由硬件提供的,它是一个唯一的标识符,用于访问计算机中的实际内存单元。物理地址是由硬件提供的,因此程序无法直接访问或修改物理地址虚拟地址是指程序中使用的地址,它是由操作系统提供的,在程序执行时被翻译成物理地址虚拟地址空间是指操作系统为每个进程分配的地址空间,每个进程都有自己独立的虚拟地址空间。在程序执行时,所有的内存访问都是针对虚拟地址空间进行的,操作系统负责将虚拟地址翻译成物理地址。 在C语言中,程序员通常只需要使用指针来访问内存,而无需关心物理地址虚拟地址的具体实现。指针可以指向任意类型的数据,包括整数、字符、数组等等。当程序使用指针来访问内存时,实际上是在使用虚拟地址,而操作系统会将其翻译成物理地址,然后将数据从内存读取到CPU中进行处理。 ### 回答2: 在计算机系统中,物理地址虚拟地址是两个重要的概念。 物理地址指的是计算机内存中的实际物理位置,也就是内存条上的存储单元。每个存储单元都有一个唯一的物理地址,用来标识它在内存中的位置。物理地址是由硬件产生和管理的,程序无法直接访问和控制物理地址。操作系统负责将程序的虚拟地址转换为对应的物理地址,然后再进行实际的内存访问。 虚拟地址是指程序中使用的地址,它是相对于程序自身的地址空间而言的,并不直接对应于实际的物理存储位置。虚拟地址是在程序执行过程中产生的,由操作系统中的内存管理单元进行转换成物理地址。使用虚拟地址可以使程序独立于具体的硬件环境,提高了系统的可移植性。 操作系统通过使用页表来进行虚拟地址物理地址的转换。页表记录了虚拟页和物理页之间的映射关系。当程序访问一个虚拟地址时,操作系统会查找页表,找到对应的物理页的地址,并进行实际的内存读写操作。 使用虚拟地址可以实现多个程序在同一台计算机上同时运行,并且彼此之间互不干扰。虚拟地址还可以提供内存保护的功能,防止程序之间的错误读写操作破坏系统的稳定性。 总结来说,物理地址是计算机内存中实际的物理位置,由硬件生成和管理;虚拟地址是程序使用的地址,相对于程序自身的地址空间;操作系统通过页表实现虚拟地址物理地址的转换。虚拟地址的使用提高了系统的可移植性和稳定性。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值