当前位置: 首页 > 编程日记 > 正文

linux 内存管理slab源码,Linux内核源代码情景分析-内存管理之slab-回收

0818b9ca8b590ca3270a3433284dd417.png

图 1

我们看到空闲slab块占用的若干页面,不会自己释放;我们是通过kmem_cache_reap和kmem_cache_shrink来回收的。他们的区别是:

0818b9ca8b590ca3270a3433284dd417.png

1、我们先看kmem_cache_shrink,代码如下:

int kmem_cache_shrink(kmem_cache_t *cachep)

{

if (!cachep || in_interrupt() || !is_chained_kmem_cache(cachep))

BUG();

return __kmem_cache_shrink(cachep);

}

__kmem_cache_shrink,代码如下:

static int __kmem_cache_shrink(kmem_cache_t *cachep)

{

slab_t *slabp;

int ret;

drain_cpu_caches(cachep);

spin_lock_irq(&cachep->spinlock);

/* If the cache is growing, stop shrinking. */

while (!cachep->growing) {//确定缓存区不在growing

struct list_head *p;

p = cachep->slabs.prev;//链表最后面的是空闲块,本例中就是第4个空闲块

if (p == &cachep->slabs)//如果链表中没有slab块,就直接break

break;

slabp = list_entry(cachep->slabs.prev, slab_t, list);

if (slabp->inuse)//如果不是空闲块,就break

break;

list_del(&slabp->list);//如果是空闲块就删除,这样下一次循环cachep->slabs.prev找的就是新的slab块

if (cachep->firstnotfull == &slabp->list)//如果firstnotfull就是这个空闲块,那么系统中一定没有部分块或者空闲块可供分配了

cachep->firstnotfull = &cachep->slabs;//指向cachep->slabs

spin_unlock_irq(&cachep->spinlock);

kmem_slab_destroy(cachep, slabp);//析构对象,并释放空闲slab块所在的所有页面

spin_lock_irq(&cachep->spinlock);

}

ret = !list_empty(&cachep->slabs);

spin_unlock_irq(&cachep->spinlock);

return ret;

}

kmem_slab_destroy,析构对象,并释放空闲slab块所在的所有页面,代码如下:

static void kmem_slab_destroy (kmem_cache_t *cachep, slab_t *slabp)

{

if (cachep->dtor

......

) {

int i;

for (i = 0; i < cachep->num; i++) {

void* objp = slabp->s_mem+cachep->objsize*i;

......

if (cachep->dtor)

(cachep->dtor)(objp, cachep, 0);//析构所有对象

......

}

}

kmem_freepages(cachep, slabp->s_mem-slabp->colouroff);//释放空闲slab块所占的所有页面

if (OFF_SLAB(cachep))

kmem_cache_free(cachep->slabp_cache, slabp);

}

kmem_freepages,释放空闲slab块所占的所有页面,代码如下:

static inline void kmem_freepages (kmem_cache_t *cachep, void *addr)

{

unsigned long i = (1<gfporder);

struct page *page = virt_to_page(addr);

/* free_pages() does not clear the type bit - we do that.

* The pages have been unlinked from their cache-slab,

* but their 'struct page's might be accessed in

* vm_scan(). Shouldn't be a worry.

*/

while (i--) {

PageClearSlab(page);

page++;

}

free_pages((unsigned long)addr, cachep->gfporder);

}

2、再看kmem_cache_reap,遍历cache_chain链表查找合适的缓存区,而且只释放了所选中的缓存区中最多80%的完全空闲slab块,代码如下:

void kmem_cache_reap (int gfp_mask)

{

slab_t *slabp;

kmem_cache_t *searchp;

kmem_cache_t *best_cachep;

unsigned int best_pages;

unsigned int best_len;

unsigned int scan;

if (gfp_mask & __GFP_WAIT)

down(&cache_chain_sem);

else

if (down_trylock(&cache_chain_sem))

return;

scan = REAP_SCANLEN;

best_len = 0;

best_pages = 0;

best_cachep = NULL;

searchp = clock_searchp;//上一次考察的缓存区

do {

unsigned int pages;

struct list_head* p;

unsigned int full_free;

/* It's safe to test this without holding the cache-lock. */

if (searchp->flags & SLAB_NO_REAP)

goto next;

spin_lock_irq(&searchp->spinlock);

if (searchp->growing)//缓存区不能处于增长状态

goto next_unlock;

if (searchp->dflags & DFLGS_GROWN) {

searchp->dflags &= ~DFLGS_GROWN;

goto next_unlock;

}

......

full_free = 0;

p = searchp->slabs.prev;//从空闲块开始

while (p != &searchp->slabs) {

slabp = list_entry(p, slab_t, list);

if (slabp->inuse)

break;

full_free++;//有一个空闲块,full_free就加1

p = p->prev;//往前继续搜索

}

/*

* Try to avoid slabs with constructors and/or

* more than one page per slab (as it can be difficult

* to get high orders from gfp()).

*/

pages = full_free * (1<gfporder);//若干空闲块所占的页面数

if (searchp->ctor)

pages = (pages*4+1)/5;

if (searchp->gfporder)

pages = (pages*4+1)/5;//页面数的百分之80

if (pages > best_pages) {//找到空闲块所占页面数最多的

best_cachep = searchp;

best_len = full_free;//空闲块的个数

best_pages = pages;//若干空闲块所占的页面数的百分之80

if (full_free >= REAP_PERFECT) {

clock_searchp = list_entry(searchp->next.next,

kmem_cache_t,next);

goto perfect;

}

}

next_unlock:

spin_unlock_irq(&searchp->spinlock);

next:

searchp = list_entry(searchp->next.next,kmem_cache_t,next);

} while (--scan && searchp != clock_searchp);//遍历cache_chain链表查找合适的缓存区

clock_searchp = searchp;//这次考察的,也就是下次考察的开始

if (!best_cachep)//没有找到

/* couldn't find anything to reap */

goto out;

spin_lock_irq(&best_cachep->spinlock);

perfect:

/* free only 80% of the free slabs */

best_len = (best_len*4 + 1)/5;//空闲块个数的百分之80

for (scan = 0; scan < best_len; scan++) {//依次释放空闲块

struct list_head *p;

if (best_cachep->growing)//不能处于增长状态

break;

p = best_cachep->slabs.prev;//因为刚才空闲slab块已经删除,所以又指向了新的空闲slab块

if (p == &best_cachep->slabs)//说明所有slab块都已经被遍历过了,break

break;

slabp = list_entry(p,slab_t,list);

if (slabp->inuse)//不是空闲块,break

break;

list_del(&slabp->list);//删除空闲slab

if (best_cachep->firstnotfull == &slabp->list)//如果firstnotfull就是这个空闲块,那么系统中一定没有部分块或者空闲块可供分配了

best_cachep->firstnotfull = &best_cachep->slabs;//指向cachep->slabs

STATS_INC_REAPED(best_cachep);

/* Safe to drop the lock. The slab is no longer linked to the

* cache.

*/

spin_unlock_irq(&best_cachep->spinlock);

kmem_slab_destroy(best_cachep, slabp);//同上

spin_lock_irq(&best_cachep->spinlock);

}

spin_unlock_irq(&best_cachep->spinlock);

out:

up(&cache_chain_sem);

return;

}

相关文章:

Vlookup的兄弟lookup讲解

Vlookup是查找函数&#xff0c;lookup也是&#xff0c;但它主要是充当模糊查找。最常见的例子就是算个税等级和成绩区间。我们创建源数据如图结果要求的是300&#xff0c;500&#xff0c;50对应的积分情况。因为数据量大&#xff0c;这里只取小部分。运用lookup函数&#xff0c…

6.python探测Web服务质量方法之pycurl模块

才开始学习的时候有点忽略了这个模块&#xff0c;觉得既然Python3提供了requests库&#xff0c;为什么多此一举学习这个模块。后来才发现pycurl在探测Web服务器的时候的强大。 pycurl是一个用c语言写的libcurl Python实现&#xff0c;支持的操作协议有FTP&#xff0c;HTTP&…

Rocksdb DeleteRange实现原理

文章目录1. 基本介绍2. 两种接口使用及简单性能对比3. DeleteRange 的基本实现3.1 写流程的实现3.2 读流程的实现 -- skyline算法以下涉及到的代码都是基于rocksdb 6.4.0版本进行描述的 1. 基本介绍 DeleteRange接口的设计是为了代替传统的删除一个区间[start,end) 内的key-va…

题目1460:Oil Deposit

题目描述&#xff1a;The GeoSurvComp geologic survey company is responsible for detecting underground oil deposits. GeoSurvComp works with one large rectangular region of land at a time, and creates a grid that divides the land into numerous square plots. I…

linux做预警机制,预警通告:Linux内核中TCP SACK机制远程DoS

漏洞描述2019年6月18日&#xff0c;RedHat官网发布报告&#xff1a;安全研究人员在Linux内核处理TCPSACK数据包模块中发现了三个漏洞&#xff0c;CVE编号为CVE-2019-11477、CVE-2019-11478和CVE-2019-11479&#xff0c;其中CVE-2019-11477漏洞能够降低系统运行效率&#xff0c;…

C# 使用xsd文件验证XML 格式是否正确

//创建xmlDocument XmlDocument doc new XmlDocument(); //创建声明段 如<?xml version"1.0" encoding"utf-8" ?> doc.AppendChild(doc.CreateXmlDeclaration("1.0", "utf-8", null)); //创建一个根节点 KYTResults Xm…

[蓝桥杯]PREV-23.历届试题_数字游戏

问题描述栋栋正在和同学们玩一个数字游戏。游戏的规则是这样的&#xff1a;栋栋和同学们一共n个人围坐在一圈。栋栋首先说出数字1。接下来&#xff0c;坐在栋栋左手边的同学要说下一个数字2。再下面的一个同学要从上一个同学说的数字往下数两个数说出来&#xff0c;也就是说4。…

Mac 上使用 Clion 阅读C++源码的一些操作

一直在尝试一些写代码方便&#xff0c;阅读代码也很方便的工具&#xff0c;因为使用的是Mac&#xff0c;所以阅读源码上面sourceInsight就没办法用了。 从vscode – sublime – clion 想要可配置性强一点&#xff0c;软件轻一点&#xff0c;也能提供足够的便捷操作&#xff0c…

c语言 字母 八进制表示'/1011',C语言C语言第一课:C语言概述为什么学习C语言怎样学习C语言.DOC...

[摘要]C语言 第一课&#xff1a; C语言概述 为什么学习C语言 怎样学习C语言 参考资料 ----------------------------------------------------------- 入门经典《C语言程序设计》 谭浩强 清华 《汇编语言》 王爽 《The C programming language》 机械工业 《C Primer Plus》 60…

Discuz! X2.5 添加自定义数据调用模块(简单方法)

转&#xff1a;http://521-wf.com/archives/46.html Discuz! X2.5 添加自定义数据调用模块&#xff08;简单方法&#xff09; Discuz!X系列的diy功能还是相当不错的&#xff0c;在对其进行二次开发的过程中&#xff0c;或许需要加入新的数据调用模块&#xff0c;这样可以使你开…

Cassandra数据模型设计最佳实践

2019独角兽企业重金招聘Python工程师标准>>> 本文是Cassandra数据模型设计第一篇&#xff08;全两篇&#xff09;&#xff0c;该系列文章包含了eBay使用Cassandra数据模型设计的一些实践。其中一些最佳实践我们是通过社区学到的&#xff0c;有些对我们来说也是新知识…

矩阵相关概念的物理意义

参考链接&#xff1a; 矩阵乘法的本质是什么&#xff1f; 条件数 病态矩阵与条件数&#xff08;&& 与特征值和SVD的关系&#xff09;矩阵的物理意义&#xff1a;https://blog.csdn.net/NightkidLi_911/article/details/38178533https://blog.csdn.net/NightkidLi_911/a…

Linux 下 进程运行时内部函数耗时的统计 工具:pstack,strace,perf trace,systemtap

简单记录一些 在linux下 统计进程内部函数运行耗时的统计工具&#xff0c;主要是用作性能瓶颈分析。当然以下工具除了pstack功能单一之外&#xff0c;其他的工具都非常强大&#xff0c;这里仅仅整理特定场景的特定用法&#xff0c;用作协同分析。 以下工具需要追踪具体的进程&…

c语言作业扩展名通常为什么,C语言的源程序通常的扩展名是( )

C语言的源程序通常的扩展名是( )更多相关问题【C20】A&#xff0e;asB&#xff0e;afterC&#xff0e;untilD&#xff0e;whenAlthough I spoke to her about the matter several times, she took little ______ of what I s“以质取胜”战略包括三个方面内容&#xff0c;分别是…

VS中C#读取app.config数据库配置字符串的三种方法(转)

关于VS2008或VS2005中数据库配置字符串的三种取法 VS2008建立Form程序时,如果添加数据源会在配置文件 app.config中自动写入连接字符串,这个字符串将会在你利用DataSet,SqlDataAparter,SqlConnection等控件时如影随行地提示你让去选择,或者是新建字符串。如果要用代码的方式取得…

获取线程中抛出的异常信息

1 ScheduledExecutorService service Executors.newScheduledThreadPool(10);2 // 从现在开始delay毫秒之后&#xff0c;每隔一天执行一次&#xff0c;转换为毫秒3 // service.scheduleAtFixedRate(this, delay, period, TimeUnit.MILLISECONDS);4 …

浅谈批处理获取管理员运行权限的几种方法

很多用了Win10版本系统的人都会发现&#xff0c;Windows对程序的运行权限是控制得更加严格了&#xff0c;即使你将UAC控制放至最低&#xff0c;如果没有特别赋予外来程序管理员运行权限的话&#xff0c;很多程序都会运行出错&#xff0c;包括很多用于系统维护的批处理程序由于运…

使用 sched_setaffinity 将线程绑到CPU核上运行

linux 提供CPU调度函数&#xff0c;可以将CPU某一个核和指定的线程绑定到一块运行。 这样能够充分利用CPU&#xff0c;且减少了不同CPU核之间的切换&#xff0c;尤其是在IO密集型压力之下能够提供较为友好的性能。 通过sched_setaffinity 设置 CPU 亲和力的掩码&#xff0c;从…

Objective C内存管理之理解autorelease------面试题

Objective C内存管理之理解autorelease Autorelease实际上只是把对release的调用延迟了&#xff0c;对于每一个Autorelease&#xff0c;系统只是把该Object放入了当前的Autorelease pool中&#xff0c;当该pool被释放时&#xff0c;该pool中的所有Object会被调用Release。 &…

c语言子程序return,c语言return返回到哪

c语言return返回到哪c语言return&#xff0c;返回给了上一级&#xff0c;比如一个递归程序&#xff0c;从第三层返回到第二层&#xff1b;又比如一个普通的子程序&#xff0c;那就返回到主程序中去。主程序中return返回给了操作系统。比如下面一个c程序int sum(int a, int b) {…

有关 schema

2019独角兽企业重金招聘Python工程师标准>>> 主要分析2点 &#xff1a;schema含义 以及 多schema下的XA处理 A schema is a collection of database objects (used by a user.). Schema objects are the logical structures that directly refer to the database’…

关于查询ios的app更新的历史版本记录

https://www.qimai.cn 推荐七麦数据 可以查询app的各种版本更新内容 由于历史久远忘记了自己app第一次上架的时间 通过这个可以查询 转载于:https://www.cnblogs.com/ccw-congcong/p/10593917.html

关于 Rocksdb 性能分析 需要知道的一些“小技巧“ -- perf_context的“内功” ,systemtap、perf、 ftrace的颜值

文章目录内部工具包含头文件接口使用核心指标Perf ContextIOStats Context外部工具Systemtap 工具Perf工具Ftrace 工具2020.8.20 23:23 &#xff0c;又到了夜深人静学习时&#xff0c;不断得思考总结总会让繁忙一天的大脑得到舒缓。作为单机存储引擎&#xff0c;Rocksdb总会被嵌…

一维数组求平均值c语言编程软件,c语言编程:用数组名作函数参数,编写一个对一维数组求平均值的函数,并在主函数中调用它...

#includeincludeint main(){void sort1(char*p1);void print(char*p2);static char*name[]{"zhangwww.book1234.com防采集请勿采集本网。#include #include #include float b(float arr[],int n); //<<<不知道你说的第2&#xff0c;4&#xff0c;5语句对应的是什…

2014年10月18日

我姐一个一点追求都没有弄的我气死了.女人管不住自己的臭嘴就让人烦死/ 还能不能嫁出去 蠢 女人说一个男的没追求没出息就是找枪口撞 蠢死转载于:https://www.cnblogs.com/wangduqiang/p/4180892.html

接口响应慢?那是你没用 CompletableFuture 来优化!

大多数程序员在平时工作中,都是增删改查。这里我跟大家讲解如何利用CompletableFuture优化项目代码,使项目性能更佳!

SQL Server 2012入门T-SQL基础篇:(8)Delete语句

基本的语法格式如下:Deleteform表名[where条件语句]此语句将删除表的部分或者全部记录;(1)带WHERE条件子句,将删除符合条件的记录:可以看到已经删除了"EmployeeKey1"的记录;(2)不带条件的delete的语句,将表中删除所有记录;转载于:https://blog.51cto.com/281816327/1…

30张图带你彻底理解红黑树

当在10亿数据进行不到30次比较就能查找到目标时,不禁感叹编程之魅力!人类之伟大呀!—— 学红黑树有感。终于,在学习了几天的红黑树相关的知识后,我想把我所学所想和所感分享给大家。红黑树是一种比较难的数据结构,要完全搞懂非常耗时耗力,红黑树怎么自平衡?什么时候需要左旋或右旋?插入和删除破坏了树的平衡后怎么处理?等等一连串的问题在学习前困扰着我。如果你在学习过程中也会存在我的疑问,那么本文对你会有帮助,本文帮助你全面、彻底地理解红黑树!

Linux内核分析--理解进程调度时机、跟踪分析进程调度和进程切换的过程

学号后三位:426 原创作品转载请注明出处 https://github.com/mengning/linuxkernel/ 1.进程的创建 除了0号进程&#xff08;系统创建的&#xff09;之外&#xff0c;linux系统中都是由其他进程创建的。创建新进程的进程&#xff0c;即调用fork函数的进程为父进程&#xff0c;…

数据结构 -- 散列表

散列表作为一种能够提供高效插入&#xff0c;查找&#xff0c;删除 以及遍历的数据结构&#xff0c;被应用在很多不同的存储组件之中。 就像rocksdb中的hashskiplist&#xff0c;redis的有序集合&#xff0c;java的 LinkedHashMap 等 都是一些非常有特色的核心数据结构&#xf…