深入SQL SERVER 2000的内存管理机制(二)
2024-08-31 00:48:57
供稿:网友
深入sql server 2000的内存管理机制(二)
http://msdn.microsoft.com/data/default.aspx?pull=/library/en-us/dnsqldev/html/sqldev_01262004.asp
可访问大地址的应用 (large-address-aware executables)
在windows增加支持/3gb参数以前,一个应用程序是无法访问一个带有高位设置的指针.一个32位的指针只有前31位地址空间可以被用户模式的应用程序访问.这剩余的一位不用.因此有一些聪明的开发者因为其他的目的不愿意在处理内存地址空间时浪费这一位.(举例来说:可以用来标志一个指针引用其它应用程序分配的数据类型).这样就/3gb参数就遇到一个难题,因为这种类型的程序不能方便的区分一个合法的指针引用的内存空间在2g的分界线以上与一个内存地址空间在2g以下,但它的高位已经被用做它用的指针.基本上如果一台机器用/3gb的参数启动,这种应用程序是无法运行的.为解决这种状况,微软在win32的pe文件characteristics字段增加了一个新的标志位来表示一个程序是否运行在可访问大地址的(large-address-aware executables)模式.当该标志位被起用( image_file_large_address_aware ),该可执行文件头部characteristics字段的32位被置位.通过这个可执行文件头部的标志位,应用程序可以暗示windows 它可以处理带有高位设置的指针,这样就不会带来任何异常结果.当该标志被置位并且支持这种方式启动的windows也是通过/3gb的参数启动,这时操作系统会提供一直扩展的私有用户模式的地址空间给应用程序.你可以通过工具,比如: dumpbin 和 imagecfg (转储可执行文件的头部)来检查一个可执行文件的该标志位.visual c++ 通过 /largeaddressaware 连接器开关来表示image_file_large_address_aware,sql servr已经将该标志位激活,所以你可以在支持/3gb参数的windows 版本中使用该参数,这样操作系统回自动扩展sql server的用户内存空间.
注释:windows在可执行文件启动时检查image_file_large_address_aware 标志位,但忽略dll文件.dll代码必须自动处理高位被置位的指针。
物理内存扩展
intel 处理器自从pentium pro开始,以后的处理器都支持一种物理内存扩展(pae)的内存映射模式。pae模式提供了可以访问64gb的物理内存空间。在pae模式中,内存管理单元(memory management unit (mmu))仍然执行:页目录入口page directory entries(pdes) 和页表入口page table entries (ptes),但是在此之上有一个新的级别:页目录指针表。同时,在pae模式pdes 和ptes是64位的(不仅仅是标准的32位),这样系统可以映射比标准转换更多的内存空间,因为pdes 和ptes模式的寻址宽度是标准的2倍。这并不仅仅是增加了页目录指针表。页目录指针表被用来管理这些大容量表和索引。一些特殊版本的windows内核需要运行在pae模式。这中内核集成在windows 2000以及以后的版本中,在单处理器的机器中体现在ntkrnlpa.exe文件中,在多处理器的机器中体现在ntkrnlpamp.exe文件中。你可以向增加/3gb和/userva参数一样在boot.ini文件中加/pae参数激活pae模式。
地址窗口扩展
windows的地址窗口扩展(awe)机制可以允许应用程序访问超过4gb的物理内存。一个32位的指针是一个整数,只能保存0x00000000到0xffffffff的值,就是说可以引用4gb 以内线性的内存地址空间,awe允许一个应用程序绕过这些限制,通过操作系统访问所有的内存空间。
从概念上说,awe并不是一个新的事物。在计算机发展之初,操作系统和应用程序已经使用相似的机制回避指针的限制。例如:我们倒退到dos年代,32位扩充功能被经常用来允许一个16位的程序访问他自已以外的内存地址空间。一些特殊目的管理者和api经常使用扩充内存和扩展内存。你可能记得这样一个很久以前产品quarterdeck qemm-386经常用来做这样的事情。典型的机制是允许一个指针可以访问超过本身限制的空间,(比如:地址太大无法在自己的指针中)通过在可以访问的地址空间分配一个窗口或区域来和本身无法访问的内存地址之间传递指针。awe的工作原理:你可以在可以访问的地址空间提供一块区域(窗口)作为分段传输区,来传送在用户内存空间无法访问的内存地址。
为了使用awe,一个应用程序需要:
1. 分配的物理内存地址可以通过allocateuserphysicalpages api函数访问。这个函数需要调用者有lock pages in memory的权限。
2. 在可以访问的内存空间建立一块区域。通过virtualalloc api函数可以作为映射一个物理内存的映射窗口。
3. 通过mapuserphysicalpages 或mapuserphysicalpagesscatter win32 api 函数完成物理内存和虚拟内存的映射。
awe已经存在于所有的windows 2000和以后的操作系统中,甚至可以用于物理内存低于2gb的操作系统中,最典型的应用是在2gb或以上物理内存的机器上,因为这是一个32位处理器访问3gb以下内存空间的唯一方法。如果你在一个低于3gb物理内存的sql server系统中激活awe支持,系统将忽略这个选项同时转换为虚拟内存管理器代替。awe内存有一个有趣的特征就是从不和磁盘交换数据。你也许注意到特有的awe api程序引用可以访问的内存空间是作为物理内存访问。这点确切的说就是:awe内存是不和系统的虚拟页面文件交互物理内存空间。
虚拟内存窗口被用来缓存awe读写访问物理内存的请求。因此,当你配置这个窗口是page_readwrite唯一可以保护的特征就是转嫁给了virtualalloc api函数。不要惊奇,这也意味你不能用virtualprotect api函数来保护这块内存区域的修改和访问。
注释:还没有专门的工具用来调查应用程序awe内存使用(任务管理器,性能监视器和监视系统 等等),显示每一个程序awe内存的使用数量。这样就没有每个程序使用awe内存数量的轨迹,同时这些内存也没包括在每个的程序的工作内存集中。
/3gb 和awe比较
增加用户程序地址空间的能力几乎有50%的应用程序是通过内存调整,这是在windows内存管理机制当中非常快捷和受欢迎的手段。而且windows awe内存机制也是非常灵活和稳定的。就像我前面所说的,当你增加1g的用户内存空间,这些内存是通过减少核心内存空间获得的(从2g减到1g)。因为核心代码的运行对整个内存空间来说是很狭窄的一块即使用于2g的空间,收缩这些内存空间意味着内部核心架构也会收缩。其中最重要的是windows用来管理物理内存的表,当你收缩核心内存空间到1g,你就限制了这个表的大小,这样只能管理最大16gb的物理内存。例如:如果你的应用程序运行在有64gb物理内存的server上并且你在启动时加了/3gb的参数。你只能访问整个内存的25%的空间—其余的48gb的内存空间无论时操作系统还是应用程序都无法访问。awe可以允许你访问比加/3gb参数更高的内存空间。显然,你通过/3gb的参数只是增加了1gb的用户内存空间,这些增加的内存空间只是对那些大地址自动获得的应用程序有效,但是只有1gb。和/3gb参数对比,awe可以使整个的物理内存对操作系统有效和对使用awe win32 api的应用程序有效。因此,awe使用和操作起来更加复杂,也更加灵活和可扩展。
着并不是说/3gb比awe更好是不存在的-不过这确实存在,比如:如果你需要分配更多的内存空间,但不能在awe中分配(线程堆栈,内存锁,过程计划),你也许发现/3gb回更好一些。