虚拟地址

生活百科 2023-01-17 17:57生活百科www.aizhengw.cn

虚拟地址

虚拟地址是Windows程式时运行在386保护模式下,这样程式访问存储器所使用的逻辑地址称为虚拟地址,与实地址模式下的分段地址类似,虚拟地址也可以写为“段偏移量”的形式,这里的段是指段选择器。

基本介绍

  • 中文名虚拟地址
  • 外文名virtual address
  • 特点CPU启动保护模式
  • 简称VA

基本简述

Windows 2000 使用基于分页机制的虚拟记忆体。每个进程有4GB的虚拟地址空间。基于分页机制,这4GB地址空间的一些部分被映射了物理记忆体,一些部分映射硬碟上的交换档案,一些部分什幺也没有映射。程式中使用的都是4GB地址空间中的虚拟地址。而访问物理记忆体,需要使用物理地址。

相关比较

下面我们看看什幺是物理地址,什幺是虚拟地址。
物理地址 (physical address): 放在定址汇流排上的地址。放在定址汇流排上,如果是读,电路根据这个地址每位的值就将相应地址的物理记忆体中的数据放到数据汇流排中传输。如果是写,电路根据这个地址每位的值就在相应地址的物理记忆体中放入数据汇流排上的内容。物理记忆体是以位元组(8位)为单位编址的。
虚拟地址 (virtual address): CPU启动保护模式后,程式运行在虚拟地址空间中。注意,并不是所有的“程式”都是运行在虚拟地址中。CPU在启动的时候是运行在实模式的,Bootloader以及核心在初始化页表之前并不使用虚拟地址,而是直接使用物理地址的。
如果CPU暂存器中的分页标誌位被设定,那幺执行记忆体操作的机器指令时,CPU(準确来说,是MMU,即Memory Management Unit,记忆体管理单元)会自动根据页目录和页表中的信息,把虚拟地址转换成物理地址,完成该指令。
比如 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放着一些标誌。
一个页目录有1024项,虚拟地址最高的10bit刚好可以索引1024项(2的10次方等于1024)。一个页表也有1024项,虚拟地址中间部分的10bit,刚好索引1024项。虚拟地址最低的12bit(2的12次方等于4096),作为页内偏移,刚好可以索引4KB,也就是一个物理页中的每个位元组。
对于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项,也就是负责10244k=4M的地址空间的映射。一个页目录项,对应一个页表。一个页目录有1024项,也就对应着1024个页表,每个页表负责4M地址空间的映射。1024个页表负责10244M=4G的地址空间映射。一个进程有一个页目录。所以以页为单位,页目录和页表可以保证4G的地址空间中的每页和物理记忆体的映射。
每个进程都有自己的4G地址空间,从 0x00000000-0xFFFFFFFF 。通过每个进程自己的一套页目录和页表来实现。由于每个进程有自己的页目录和页表,所以每个进程的地址空间映射的物理记忆体是不一样的。两个进程的同一个虚拟地址处(如果都有物理记忆体映射)的值一般是不同的,因为他们往往对应不同的物理页。
在windows下4G地址空间中低2G,0x00000000-0x7FFFFFFF 是用户地址空间,4G地址空间中高2G,
0x80000000-0xFFFFFFFF 是系统地址空间。访问系统地址空间需要程式有ring0的许可权。而Linux对4G空间的划分不同与windows。linux将最高的1G 位元组(从虚拟地址0xC0000000 到0xFFFFFFFF),供核心使用,称为“核心空间”。而将较低的3G 位元组(从虚拟地址0x00000000 到0xBFFFFFFF),供各个进程使用,称为“用户空间”。
在LINUX系统下,0xC00000000-0XFFFFFFFF为系统空间,为所有的系统进程所共享,0X00000000-0XBFFFFFFF为用户空间。

Copyright@2015-2025 www.aizhengw.cn 癌症网版板所有