关于虚拟内存管理之前的请看:OS之内存管理 — 虚拟内存管理(一)

帧分配

每个进程对的最小帧数是由操作系统的体系结构决定的,但是最大帧数是由可用物理内存的数量决定的。所以在这之间,对于进程的帧的分配是有很多选择的。

分配算法

  • 平均分配

    在n个进程中分配m个帧的最容易的方法就是,给每个进程一个平均值,即m/n帧(在这里是忽略操作系统所需的帧,即m个帧是空闲帧)。比如有93个帧和5个进程,那么分配给每个进程18个帧,剩余的3个帧可以用作空闲帧缓冲池,这种方法称为平均分配

  • 比例分配

    因为进程大小不同,如果平均分配的话,小进程分配的帧数太多,浪费资源;大进程分配帧太少,不够用。为了解决这个问题,采用比例分配。根据每个进程大小分配可用内存。假设进程pip_ipi​的虚拟内存大小为sis_isi​,并且定义:S=∑si (S为系统空闲内存总量)S=\sum{s_i} (S为系统空闲内存总量)S=∑si​ (S为系统空闲内存总量)

    如果可用帧的总数为m,那么进程pip_ipi​可以分配得到aia_iai​个帧,aia_iai​近似为:

    ai=si/S∗ma_i=s_i / S * mai​=si​/S∗m

对于平均分配和比例分配,每个进程分得的数量可以因多道程度而变化。如果多道程度增加,则每个进程会失去一些帧,以提供给新进程所需的内存;如果多达程度降低,那么原来分配给离开进程的帧会分配给剩余进程。

在这里需要注意:对于平均分配和比例分配,高优先级进程和低优先级进程同样处理。但是希望给予高优先的进程更多的内存以便加速执行,一种解决方法是:所采用的比例分配的策略不是根据进程的相对大小,而是根据进程的优先级或大小和优先级的结合。

全局分配和局部分配

为个进程分配帧的一个重要因素是页面置换。由于多个进程竞争帧,可以将页面置换算法分为两大类:全局置换局部置换

全局置换允许一个进程从所有帧的集合中选择一个置换帧,而不管该帧是否已分配给其他进程,也就是说,一个进程可以从另外一个进程那里获取帧。

局部置换要求每个进程只从它自己分配的帧中选择。

那么就有一种帧的分配方案:允许高优先级的进程从低优先级进程中选择帧用于置换。采用局部置换策略,分配给每个进程的帧数不变;采用全局置换,一个进程可能从分配给其他进程的帧中选择一个用于置换,从而增加了分配给它的帧数。

但是如果采用全局置换的分配方案,那么进程不能控制自己的缺页错误率。因为一个进程的内存页面的集合不仅取决于进程本身的调页行为,还取决于其他进程的调页行为。

通常情况下,局部置换由于不能使用其他进程的较少使用的内存页面,可能会阻碍一个进程,而全局置换通常会有更好的系统吞吐量,所以还是在很多系统中经常使用的。

非均匀内存访问

之前对于虚拟内存的讨论都是在假设所有内存都相同或可以被平等访问的基础上。但是现代计算机一般情况下都是“多核”或者“多芯”。对于具有多个CPU的系统,给定的CPU可以比其他CPU更快的访问内存的某些部分。这些性能上的差异主要是因为CPU和内存在系统中互连造成的,在这里不做深究。

对于特定CPU访问同板内存的延迟,要小于访问其他板内存的延迟,这种具有明显不同内存访问时间的系统称为非均匀内存访问系统(NUMA)。那么在考虑NUMA中帧分配时,要考虑让分配的帧"尽可能的靠近"运行进程的CPU。这通常意味着与CPU一样位于同一系统板。

对于帧分配算法的修改应该让调度程序跟踪每个进程运行的最后一个CPU,如果调度程序尝试将每个进程调度到它先前的CPU上,而且内存管理系统尝试分配的帧靠近正在分配的CPU,那么将导致高速缓存命中率的改进和内存访问时间的减少。

系统抖动

如果进程没有需要支持活动使用页面的帧数,那么它会很快产生缺页错误,这时必须要置换某个页面。但是由于它的所有页面都在使用中,那么必须立即置换需要再次使用的页面。因此它会再次快速产生缺页错误,再一次置换必须立即返回的页面,如此快速进行。这种高度的页面置换活动叫做抖动如果一个进程的调页时间多于执行时间,那么进程就在抖动

解决抖动的方法

通过局部置换算法优先权置换算法,可以限制系统抖动。如果一个进程开始抖动,那么由于采用局部置换,它将不能从另外进程中获取帧。

注意:这种方法只是限制,并不能解决抖动。因为发生了抖动,那么大多数时间内会排队等待调页设备,即使对于不在抖动的进程,也会影响其性能。

工作集模型

为了防止抖动,需为进程提供足够多的所需帧,但是怎么样才能知道进程需要多少帧呢?有一种技术叫做工作集模型

这种方法定义了进程执行的局部性模型:局部性模型指出随着进程执行,它从一个局部移向另一个局部。局部性是最近使用页面的一个集合。一个程序通常由多个不同的可能重叠的局部组成。

工作集模型就是基于局部性假设的。这个模型采用参数Δ\DeltaΔ,定义工作集窗口,其思想就是检查最近Δ\DeltaΔ个页面引用。这最近Δ\DeltaΔ个页面引用的页面集合称为工作集。如果一个页面处于活动使用状态,那么它处在工作集中。如果它不再使用,那么它在最后一次引用的Δ\DeltaΔ时间单位后,会从工作集中删除。所以,工作集是程序局部的近似。

工作集的精度取决于Δ\DeltaΔ的选择,如果Δ\DeltaΔ太小,那么他不能包含整个局部;如果Δ\DeltaΔ太大,那么他可能包含多个局部,在极端情况下,如果Δ\DeltaΔ为无穷大,那么工作集为进程执行所需所有页面的集合。所以最重要的工作集属性就是它的大小。

如果系统内的每个工作集通过计算为WSSiWSS_iWSSi​,那么会得到:

D=ΣWSSiD=\Sigma{WSS_i}D=ΣWSSi​

这里的D就是帧的总需求量,每个进程都使用其工作集内的页面,所以进程i需要WSSiWSS_iWSSi​帧,如果总需求量大于可用帧的总数(D>m),将会发生抖动,因此有些进程得不到足够的帧数。

一旦选中了Δ\DeltaΔ,工作机的使用就是由操作系统监视每个进程的工作集,并为它分配大于其工作集的帧数。如果还有足够的额外帧,那么可启用另一进程。如果工作集大小的总和增加,以至于超过可用帧的总数,那么操作系统会选择一个进程来挂起。该进程的页面被写出(交换),并且其帧可分配给其他进程。挂起的进程以后可以重启。

缺页错误频率

工作集模型的控制是比较笨拙的,因为系统需要监控每个进程的工作集模型,然后做出相应的策略。采用**缺页错误频率的策略(PFF)**是更加直接的方法。

缺页错误频率策略设置所需缺页错误率的上下限,如果实际缺页错误率超过上限,则可为进程在分配一帧;如果实际缺页错误率低于下限,则可从进程中删除一帧。通过直接测量和控制缺页错误率以防止抖动。

与工作集策略相同,也可能不得不换出一个进程,如果缺页错误率增加并且没有空闲帧可用,你那么必须选择某个进程并将其交换到后备存储,然后再将释放的帧分配给具有高缺页错误率的进程

分配内核内存

用于内核内存的空闲内存池通常不同于用于普通用户模式进程的列表,主要有两个原因:

  1. 内核需要为不同大小的数据结构请求内存,其中有的小于一页,因此内核应该保守的使用内存,并努力最小化碎片的浪费
  2. 用户模式进程分配的页面不必位于连续物理内存,但是有的硬件设备和物理内存直接交互,即无法享有虚拟内存接口的便利。

为了管理用于内核进程的空闲内存,有两种策略:伙伴系统和slab分配

伙伴系统

伙伴系统从物理连续的大小固定的端上进行分配,从这个段上分配内存,采用2的幂分配器来满足请求分配单元的大小为2的幂,请求单元的大小如果不适当,那么就调整到下一个更大的2的幂。

假设内存段的大小最初为256kb,内核请求21kb的内存,最初这个段分为两个伙伴,称为ALA_LAL​和ARA_RAR​,每个的大小都是128kb;这两个伙伴之一进一步分成两个64kb的伙伴,即BLB_LBL​和BRB_RBR​。然而从21kb开始的下一个打的幂是32kb,因此BLB_LBL​和BRB_RBR​再次划分为两个32kb的伙伴CLC_LCL​和CRC_RCR​。因此其中一个32kb的段可用于满足21kb请求。

伙伴系统的优点是:通过称为合并的技术,可以将相邻伙伴快速组合已形成更大的分段。

缺点:由于调整到下一个2的幂,很可能造成分配段内的碎片

slab分配

每个slab由一个或多个物理连续的页面组成。每个cache由一个或多个slab组成。每个内核数据结构都有一个cache。每个cache含有内核数据结构的对象实例(称为object),三者关系如下:



slab分配算法采用cache来存储内核对象。在创建cache时,若干起初标记为free的对象被分配到cache。cache内的对象数量取决于相关slab的大小。例如12kb slab(由三个连续的4kb页面组成)可以存储6个2kb的对象。最初cache内的所有对象都标记为空闲,当需要内核数据结构的新对象时,分配器可以从cache内分配任何空闲对象以便满足请求。

在Linux中,slab可以处于三种可能状态之一:

  • 满的(full):slab的所有对象标记为使用
  • 空的(empty):slab上的所有对象标记为空闲
  • 部分(partial):slab上的对象有的标记为使用,有的标记为空闲

slab分配器有两个优点:

  • 没有因碎片而引起内存浪费
  • 可以快速的满足内存请求
参见于:《操作系统概念

OS之内存管理 --- 虚拟内存管理(二)的更多相关文章

  1. OS之内存管理 --- 虚拟内存管理(一)

    虚拟内存的背景 在基本的内存管理策略中,所有的策略的相同点都是:每个进程在执行之前需要完全处于内存中.那有没有一种方法可以不需要将进程所有页面加载到内存中就可以开始运行进程呢?有没有可能在进程需要某些 ...

  2. 十问 Linux 虚拟内存管理 (glibc) (二)

    版权声明:本文由陈福荣原创文章,转载请注明出处: 文章原文链接:https://www.qcloud.com/community/article/184 来源:腾云阁 https://www.qclo ...

  3. linux内核内存分配(三、虚拟内存管理)

    在分析虚拟内存管理前要先看下linux内核内存的具体分配我開始就是困在这个地方.对内核内存的分类不是非常清晰.我摘录当中的一段: 内核内存地址 ============================ ...

  4. Linux的虚拟内存管理-如何分配和释放内存,以提高服务器在高并发情况下的性能,从而降低了系统的负载

    Linux的虚拟内存管理有几个关键概念: Linux 虚拟地址空间如何分布?malloc和free是如何分配和释放内存?如何查看堆内内存的碎片情况?既然堆内内存brk和sbrk不能直接释放,为什么不全 ...

  5. Oracle内存管理(之二)

    [深入解析--eygle] 学习笔记 1.2.2 UGA和CGA UGA(用户全局区)由用户会话数据.游标状态和索引区组成.在共享server模式下,一个共享服务进程被多个用户进程共享,此时UGA是S ...

  6. 【python测试开发栈】—python内存管理机制(二)—垃圾回收

    在上一篇文章中(python 内存管理机制-引用计数)中,我们介绍了python内存管理机制中的引用计数,python正是通过它来有效的管理内存.今天来介绍python的垃圾回收,其主要策略是引用计数 ...

  7. Linux虚拟内存管理(glibc)

    转:https://blog.csdn.net/tengxy_cloud/article/details/53067396 https://www.cnblogs.com/purpleraintear ...

  8. 一步一图带你深入理解 Linux 虚拟内存管理

    写在本文开始之前.... 从本文开始我们就正式开启了 Linux 内核内存管理子系统源码解析系列,笔者还是会秉承之前系列文章的风格,采用一步一图的方式先是详细介绍相关原理,在保证大家清晰理解原理的基础 ...

  9. SoC嵌入式软件架构设计II:没有MMU的CPU虚拟内存管理的设计和实现方法

    大多数的程序代码是必要的时,它可以被加载到内存中运行.手术后,可直接丢弃或覆盖其它代码. 我们PC然在同一时间大量的应用,地址空间差点儿能够整个线性地址空间(除了部分留给操作系统或者预留它用).能够觉 ...

随机推荐

  1. 2018.11.24 poj1743Musical Theme(二分答案+后缀数组)

    传送门 代码: 二分答案. 然后对于预处理的heightheightheight数组分成几段. 保证每一段中都是连续的几个heightheightheight并且这些heightheightheigh ...

  2. systemC的环境搭建

    window下systemc的环境搭建 安装视频 一.编译SystemC库 1.下载SystemC library source code (systemc-2.3.1版本) 2.解压到工作目录 3. ...

  3. denyhost安装脚本

    #!/bin/bashDENYHOSTS=DenyHosts-2.6.tar.gzDENYHOSTS_VERSION=DenyHosts-2.6DENYHOSTS_URL=http://192.168 ...

  4. HTML5与相关类的扩充

    1.getElementsByclassName()方法 <body> <div class='a1'>klkx1</div> <ul id='ul1'> ...

  5. JavaScript相关基础知识点

    JavaScript简介: JavaScript是脚本语言,是一种轻量级的编程语言,是可插入 HTML 页面的编程代码,插入 HTML 页面后,可由所有的现代浏览器执行. JavaScript使用: ...

  6. POJ3046选蚂蚁创建集合_线性DP

    POJ3046选蚂蚁创建集合 一个人的精力是有限的呢,如果一直做一件事迟早会疲惫,所以自己要把握好,不要一直埋头于一件事,否则效率低下还浪费时间 题目大意:一共有T(1,2...n为其种类)种蚂蚁,A ...

  7. Http TCP/IP协议和socket之间的区别和联系

    总结,TCP/IP是传输层协议,主要解决数据如何在网路中传输,socket是TCP/IP协议的具体实现,是对TCP/IP协议的封装和应用,属于程序员层面,HTTP是应用层协议,应用层协议很多,类似的像 ...

  8. jQuery WeUI V0.4.2 发布

    http://www.oschina.net/news/71590/jquery-weui-v0-4-2 jQuery WeUI V0.4.2 发布了! jQuery WeUI 中使用的是官方WeUI ...

  9. Android-Kotlin-印章类

    上一篇博客介绍了,Android-Kotlin-枚举enum: 由于枚举 和 印章类 有相似之处,所以两者对比一下: Kotlin的枚举,重点区分的数据本身 Kotlin的印章类,重点区分的是数据类型 ...

  10. Python——面向对象(初级篇)

    1.如何创建类 class 类名: pass 2.如何创建方法 构造方法: __init__ obj = 类名('a1') 普通方法: obj = 类名('xxx') obj.普通方法名() 3.图解 ...