首页 > linux内核 -内存管理模块概图

linux内核 -内存管理模块概图

1.从进程(task)的角度来看内存管理

这里写图片描述

  1. 每个进程对应一个task_struct;
  2. 每个task_struct 里面包含指向mm_struct 的指针mm,

    mm_struct 里面的主要成员:

    a. 指向vma链表的头指针:mmap

    b. 指向vma红黑树的根节点: mm_rb

    c. 指向进程列表的指针pgb;
  3. vma(vm_area_struct): 进程虚存管理的最基本的管理单元应该是struct vm_area_struct了,它描述的是一段连续的、具有相同访问属性的虚存空间,该虚存空间的大小为物理内存页面的整数倍。通常,进程所使用到的虚存空间不连续,且各部分虚存空间的访问属性也可能不同。所以一个进程的虚存空间需要多个vm_area_struct结构来描述。

    结构体的主要成员:

    a. vma的起始和结束地址;

    b. 指向vma 前后节点的指针

    c. 指向当前vma在红黑树中的位置指针;

    d. 指向当前vma所归属的mm_struct 的指针;

2.从物理页面(page)的角度来看待内存管理

这里写图片描述

  1. mem_map[] 里面包含了所有的物理页面,可以通过索引来访问。
  2. 每一个物理页面用 struct page来表示,page 里面的主要成员介绍:

    a. flags里面包含了当前页面的标志

    另外也包含了其所属的zone的标志。

    b. mapping:表示这个页面指向的地址空间。反响映射(reverse mapping)时使用,比如页面回收时。

    c. _mapcount:表示这个页面被进程映射的个数。

    d. _count:内核中引用该页面的次数,当其为0时,表示这个页面空闲。

  3. struct zone:

    zone 里面的主要成员介绍:

    a. watermark[]: zone的3个水位值:min/low/high, 在kswapd页面回收中会用到;

    b. lowmem_reserve[]: zone中遗留的内存:

    https://blog.csdn.net/kickxxx/article/details/883573

    kernel在分配内存时,可能会涉及到多个zone,分配会尝试从zonelist第一个zone分配,如果失败就会尝试下一个低级的zone(这里的低级仅仅指zone内存的位置,实际上低地址zone是更稀缺的资源)。我们可以想像应用进程通过内存映射申请Highmem 并且加mlock分配,如果此时Highmem zone无法满足分配,则会尝试从Normal进行分配。这就有一个问题,来自Highmem的请求可能会耗尽Normal zone的内存,而且由于mlock又无法回收,最终的结果就是Normal zone无内存提供给kernel的正常分配,而Highmem有大把的可回收内存无法有效利用。

    因此针对这个case,使得Normal zone在碰到来自Highmem的分配请求时,可以通过lowmem_reserve声明:可以使用我的内存,但是必须要保留lowmem_reserve[NORMAL]给我自己使用。

    同样当从Normal失败后,会尝试从zonelist中的DMA申请分配,通过lowmem_reserve[DMA],限制来自HIGHMEM和Normal的分配请求。

    c. zone_pgdat: 指向内存节点

    在UMA系统上,只使用一个NUMA结点来管理整个系统内存

    d. lruvec: LRU的链表集合,用于内存页面回收(page reclaim)

    共5个链表:

    匿名页面的不活跃链表、匿名页面的活跃链表

    文件页面的不活跃链表、文件页面的活跃链表

    不可回收页面链表

struct lruvec {struct list_head lists[NR_LRU_LISTS];struct zone_reclaim_stat reclaim_stat;
#ifdef CONFIG_MEMCGstruct zone *zone;
#endif
};
enum lru_list { LRU_INACTIVE_ANON = LRU_BASE, LRU_ACTIVE_ANON = LRU_BASE + LRU_ACTIVE, LRU_INACTIVE_FILE = LRU_BASE + LRU_FILE, LRU_ACTIVE_FILE = LRU_BASE + LRU_FILE + LRU_ACTIVE,LRU_UNEVICTABLE,NR_LRU_LISTS
};

3.task里面的vma和page怎么关联呢?

下一节详细研究

更多相关:

  • 更多内容,欢迎关注微信公众号:全菜工程师小辉~前言在笔者上一篇博客,详解了NIO,并总结NIO相比BIO的效率要高的三个原因,彻底搞懂NIO效率高的原理。这篇博客将针对第三个原因,进行更详细的讲解。首先澄清,零拷贝与内存直接映射并不是Java中独有的概念,并且这两个技术并不是等价的。零拷贝零拷贝是指避免在用户态(User-space)...

  • 一、预备知识—程序的内存分配  一个由c/C++编译的程序占用的内存分为以下几个部分  1、栈区(stack)— 由编译器自动分配释放 ,存放函数的参数值,局部变量的值等。其操作方式类似于数据结构中的栈,如果还不清楚,那么就把它想成数组,它的内存分配是连续分配的,即,所分配的内存是在一块连续的内存区域内.当我们声明变量时,那么编译器...

  • 我的爱机是一台ThinkPad T420,原装三星DDR 1333 4G内存一根,还剩一根内存位置,最近趁京东6.18促销,准备增加一根物理内存。为了确保兼容性,觉得仍然选购DDR 1333 4G内存,于是购买了金士顿这款,比如DDR3 1600的还贵。 这个安装过程完全参照该内存的网页提示进行 这里简单记录一下,以备...

  • 陪伴我多年的老本ThinkPad T420渐渐垂垂老矣, 我想更新一下可以更新的部分, 比如将2.5寸HDD更换为SSD, 将单条4G内存再增加一根, 凡此种种想法, 可能最后归结为如何获取该笔记本的硬件配置信息, 在windows下面使用鲁大师之类的检测软件, 也许很好搞定,但是在Ubuntu 14.04平台上如果办到呢? 很简单...

  • 一.内存错误出现的场景 这几天在重构ATS插件代码的过程中遇到了烦人的内存泄露问题, 周五周六连续两天通过走查代码的方法,未能看出明显的导致内存错误的代码, 同时也觉得C和C++混合编程得到一个动态库, 在一个.cpp主文件中,即用new又用malloc来动态分配内存, 可能会导致内存错误.后来网上调研和查资料发现, new和mal...