复合页(Compound Page)就是将物理上连续的两个或多个页看成一个

     独立的大页,它能够用来创建hugetlbfs中使用的大页(hugepage)。

     也能够用来创建透明大页(transparent huge page)子系统。可是

     它不能用在页缓存(page cache)中,这是由于页缓存中管理的都是

     单个页。





     分配一个复合页的方式是:使用alloc_pages函数,參数order至少为1,

     且设置__GFP_COMP标记。由于依据复合页的定义,它通常包含2个或多

     个连续的物理内存页,这是由它的实现决定的,因而order參数不可能

     为0。





     通常调用alloc_pages的内存分配方式例如以下:

     p = alloc_pages(GFP_KERNEL, 2);

可是这样的方式和创建一个复合页有什么不同呢?不同点就是在创建复合

     页的时候会创建与这个复合页相关的元数据(metadata)。

表示复合页的元数据都存在于Page结构体中,Page页中的flag标记用来

     识别复合页。

在复合页中,打头的第一个普通页成为“head page”,用

     PG_head标记,而后面的全部页被称为“tail pages”,用PG_tail标记。

     在64位系统中,能够有多余的标记来表示复合页的页头和页尾;可是在

     32位系统中却没有那么多的标记,因此採用了一种复用其它标记的方案,

     即将复合页中的全部页都用PG_compound标记,然后,对于尾页同一时候也

     使用PG_reclaim标记。这是由于PG_reclaim仅仅有在页缓存中会用到,而

复合页根本就不会在页缓存中使用。

能够使用PageCompound函数来检測一个页是否是复合页,另外函数PageHead

     和函数PageTail用来检測一个页是否是页头或者页尾。

在每一个尾页的page

     结构体中都包括一个指向头页的指针 - first_page,能够使用compound_head

     函数获得。





     那么当一个复合页不再被系统使用时,我们怎样知道该复合页包括多少

     个普通页。又怎样知道该复合页的析构函数(destructor)存在哪里呢?

     首先,人们可能会觉得这些信息存在于头页的page结构体中,可是非常不

     幸,在这个结构体中已经没有可用的空间了。

因此,这些信息所有存储

     在第一个尾页的lru字段中,将该复合页的大小(order)首先强制转换

     为指针类型,然后存储在lru.prev中,将析构函数存储在lru.next中。

这里就解释了为什么复合页必须至少是两个页。





     在内核中生命了两个复合页的析构函数。默认情况下会调用free_compound_page

     来将全部的页返回给系统的页框分配器。而hugetlbfs子系统会调用free_huge_page

     来做一些统计并释放。





     使用复合页的最经典的一个样例就是THP(transparent huge page)。

     另外一些驱动使用复合页来方便缓存的管理。





ref

===

1. https://lwn.net/Articles/619514/

复合页( Compound Page )的更多相关文章

  1. 刨根究底字符编码之七——ANSI编码与代码页(Code Page)

    ANSI编码与代码页(Code Page) 一.ANSI编码 1. 如前所述,在全世界所有国家和民族的文字符号统一编码的Unicode编码方案问世之前,各个国家.民族为了用计算机记录并显示自己的字符, ...

  2. 大页(Huge Page)简单介绍

    x86(包括x86-32和x86-64)架构的CPU默认使用4KB大小的内存页面(getconf PAGESIZE),但是它们也支持较大的内存页,如x86-64系统就支持2MB大小的大页(huge p ...

  3. 工作于内存和文件之间的页缓存, Page Cache, the Affair Between Memory and Files

    原文作者:Gustavo Duarte 原文地址:http://duartes.org/gustavo/blog/post/what-your-computer-does-while-you-wait ...

  4. 描述各自页面的 page

    一个小程序页面由四个文件组成(注意:为了方便开发者减少配置项,描述页面的四个文件必须具有相同的路径与文件名).分别是: 页面 Page(JS文件) Page(Object) 函数用来注册一个页面.接受 ...

  5. 教你如何修改FireFox打开新标签页(NewTab Page)的行列数

    FireFox的打开新建标签页(即NewTab Page)默认只能显示3x3个网站缩略图,这9个自定义的网站,非常方便快捷,什么hao123的弱爆了,本人从未用过此类导航网站,曾经用过的也只是abou ...

  6. 实现加载页Loading Page 的几种方法

    网页也可以像原生应用那样加入进度条或者其他的loading效果带来更好的等待体验,这里归纳几种我收集的实现loading page的方法,这几种方法在交互上都有利有弊,适用于不同应用.(PS:以下方法 ...

  7. JSP页面的Page指令指定编码和Meta标签编码

    JSP代码如下: <%@ page language="java" contentType="text/html; charset=UTF-8" page ...

  8. SQL Server 存储(4/8):理解Page Free Space (PFS) 页

    我们已经讨论了GAM与SGAM页,数据页(Data Page) ,现在我们来看下页面自由空间页(Page Free Space (PFS) ). PFS在数据文件里是第2页(页号1,页号从0开始),接 ...

  9. SQL Server :理解Page Free Space (PFS) 页

    原文:SQL Server :理解Page Free Space (PFS) 页 我们已经讨论了GAM与SGAM页,数据页(Data Page) ,现在我们来看下页面自由空间页(Page Free S ...

随机推荐

  1. Vue之x-template(2)

    将html结构写在一对script标签中,设置type=“x-template” <!DOCTYPE html> <html> <head lang="en&q ...

  2. python 获取路径

    获取目录路径和文件路径 import osfor root, dirs, files in os.walk(".", topdown=False): # ‘.’为获取脚本所在路径下 ...

  3. gulp(1) 的使用

    1.安装node.js 2.全局安装gulp.js npm install gulp -g 3.在项目本地根目录再安装(通过黑窗口安装)npm install --save-dev gulp/ 或者 ...

  4. python 3 廖雪峰博客笔记(三) 命令行模式与交互模式

    python 的代码一般保存为 .py结尾的文本文件格式 比如 add.py 里写下如下内容 100 + 200 执行 add.py有两种方式: 1. 命令行方式:将python代码写入脚本中执行 p ...

  5. linux nslookup-查询域名DNS信息的工具

    博主推荐:更多网络测试相关命令关注 网络测试  收藏linux命令大全 nslookup命令是常用域名查询工具,就是查DNS信息用的命令. nslookup4有两种工作模式,即“交互模式”和“非交互模 ...

  6. jquery 修改input输入框的 readOnly属性 && input输入框隐藏

    html的代码 <div class="control-group"> <label class="control-label required&quo ...

  7. Division

    Description   Write a program that finds and displays all pairs of 5-digit numbers that between them ...

  8. UVA 12100 打印队列(STL deque)

    题意: 给定n个优先级打印队列,然后从0开始编号到n-1.出队一个元素,如果他是队列中优先级最高的,打印(耗时一分钟),否则放到队尾(不耗时).给定一个m,求位置m的文件打印的时间. 分析: 用一个p ...

  9. prometheus监控mysql

    创建一个用于mysqld_exporter连接到MySQL的用户并赋予所需的权限 mysql> GRANT REPLICATION CLIENT, PROCESS ON *.* TO '; my ...

  10. 第一个web项目

    1)       创建Java Web Project 2)       创建相应的包 3)       创建类并继承于HttpServlet 4)       重写service()方法 5)    ...