本文章由vector03原创, 转载请注明出处.

邮箱地址: mmzsmm@163.com, 欢迎来信讨论.

3.4 sys_alloc

sys_alloc是dlmalloc中向系统获取内存的主要接口.
因为涉及到mmap, top-most segment, top chunk的交互,
相对要更复杂. 我们相同先介绍主要分配算法,
再具体分析子函数.

3.4.1 核心算法

基本上sys_alloc分为四个步骤,

1. 首先检查请求大小nb是否超出mmap_threshold的阈值.
假设是, 则放弃由分配器管理,
直接在mmap区开辟,
原因前面说过, 不再赘述.

2. 依据mspace设定及当前top space的使用情况,
向系统申请一块适当的内存.

Dlmalloc依照以下的顺序由主到次开辟,

第一, 假设同意MORECORE,
则优先通过MORECORE开辟连续内存空间.

连续空间开辟又分为例如以下几种情况,

若当前mspace处于诞生阶段,
则直接开辟nb + SYS_ALLOC_PADDING大小的空间.

watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQvdmVjdG9yMDM=/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/gravity/Center">

若当前mspace已存在top,
则部分空间可利用top, 剩余nb – m->topsize + SYS_ALLOC_PADDING大小的空间向系统申请.

若MORECORE返回成功,
但空间不连续, 则会尝试扩展空间esize大小以满足分配需求.
假设空间扩展失败, 则反向MORECORE将之前申请的空间归还给系统.

第二, 假设上一步申请失败,
或不同意MORECORE, 则通过MMAP申请.
注意, 这里的MMAP同sys alloc步骤1的mmap是两码事.
这一步申请的结果是要归入mspace空间的.

第三, 倘若前两步都失败,
且同意非连续(noncontiguous)MORECORE, 则尝试直接在system heap上分配非连续空间.
这有些类似第一步中的扩展空间, 差别是此时已经明白top space不连续,
直接申请目标大小.

3. 依据申请成功的地址与原top space的关系,
对连续空间合并. 假设不能合并,
则新开区段. 申请的地址和大小保存在tbase和tsize暂时变量里.

当中, 区段合并又分为两种.
若tbase与top-most区段末尾相毗邻,
则从后面合并. 这样的情况适用于大部分MORECORE以及小部分MMAP申请到的空间.

若tbase与top-most区段開始相毗邻,
则从前面合并. 这样的情况出现的比較少,
多在MMAP时产生, MORECORE尽管也可能出现该情况,
相对就更少.

watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQvdmVjdG9yMDM=/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/gravity/Center">

4. 最后从已扩展的top space中划分chunk返回给用户.
这样就完毕了sys_alloc的所有流程.

具体代码凝视例如以下,

watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQvdmVjdG9yMDM=/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/gravity/Center">

这个代码基本上就是本小节開始时介绍的流程,
相信看懂了前面的算法说明, 这里自然没有什么难度.
须要说明的仅有两点,

一个就是在Line4064和Line4065出现的推断,
先对nb做padding计算得到asize,
再推断其是否小于等于nb. 这里就是溢出检測,
对于两个无符号整数的加法, 这是比較便捷的检验方法.
类似的代码在dlmalloc中到处都是.

还有一个參考Line4108,
这里相同是溢出检測. 由于MORECORE的入參在dlmalloc中被觉得是有符号的.
而HALF_MAX_SIZE_T是size_t的一半,
以此来推断ssize是否溢出.

3.4.2 mmap_alloc

当nb大于mmap_threshold时,
会调用该函数直接进行mmap分配.
与sys_alloc通过其它途径申请的差别在于, dlmalloc对这类空间倾向于不长期持有,
也不纳入不论什么分箱或区段中. 能够觉得它们是脱离dlmalloc管理的孤立内存区域.

既然是孤立内存,
首尾就不会有毗邻的chunk, 但直接mmap出来的payload地址未必是对齐的,
因此在对齐后会产生内部碎片. dlmalloc就将这些碎片伪装成一个chunk.
这样, 当用户释放这片内存时,
能够依据记录在prev_foot中的size信息找到当初mmap出来的首地址.

上图中, mmap分配的原始地址是mm,
经过对齐后的地址是p. dlmalloc将前面的对齐部分伪装成一个free chunk,
长度记录在p->prev_foot中.
当释放时, 就能够依据payload指针又一次计算出mm的地址.
在结尾, 有长度为MMAP_FOOT_PAD的一段区域,
用来放置fake next chunk. 也就是保存magic以及fencepost.

代码凝视例如以下,

3.4.3 prepend_alloc

在3.4.1小节中介绍了从系统申请的扩展内存会依据其首地址和旧区段之间的位置关系做合并.
倘若append到区段后面,
申请内存是比較简单的, 由于扩展地址会直接补充到top中,
仅仅需分割top就可以.
但假设prepend到前面情况就相对复杂了,
由于从原区段base到top之间的情况不明,
所以必须分情况讨论. 而prepend_alloc函数就是为此而写的.

该函数会在一開始将分配请求从扩展空间中分割出来,
剩余工作就是依据不同情况对remainder做对应处理,

1. 假设旧区段base与top是同一个地址,
直接移动top指针,
将remainder吸收到top中.

2. 假设旧区段base与dv是同一地址,
则扩充dv的范围.

3. 若旧区段開始是普通的free chunk,
则移动oldfirst指针,
将remainder和free chunk合并.

4. 若旧区段開始是inused chunk,
则将remainder插入回分箱.

代码凝视例如以下,

3.4.4 add_segment

对于无法合并的扩展内存区域, dlmalloc最后会将它们作为新的segment插入.

创建新segment依照例如以下步骤进行,

1. 首先,
依据top, 查找到当前top-most区段,
而且定位出在其结尾的隐藏chunk.

2. 将top又一次初始化为新的segment的基址.

3. 将mstate中保存的旧top-most区段信息push到旧区段的隐藏chunk里.
并将新区段信息记录在mstate中.

4. 旧区段末尾写入一连串fenceposts.

5. 若旧区段剩余的top可用,
则将旧top又一次插入分箱系统中.

源代码凝视例如以下,

watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQvdmVjdG9yMDM=/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/gravity/Center">

Line4016是top初始化函数,
该函数基本仅仅是简单的信息记录, 并在末尾伪装隐藏chunk,
代码例如以下,

watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQvdmVjdG9yMDM=/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/gravity/Center">

dlmalloc 2.8.6 源代码具体解释(6)的更多相关文章

  1. dlmalloc 2.8.6 源代码具体解释(5)

    本文章由vector03原创, 转载请注明出处. 邮箱地址: mmzsmm@163.com, 欢迎来信讨论.     3. 分配及实现 本章节介绍dlmalloc的分配算法和实现.由于存在多mspac ...

  2. Spring IOC源代码具体解释之容器初始化

    Spring IOC源代码具体解释之容器初始化 上篇介绍了Spring IOC的大致体系类图,先来看一段简短的代码,使用IOC比較典型的代码 ClassPathResource res = new C ...

  3. Spring IOC源代码具体解释之容器依赖注入

    Spring IOC源代码具体解释之容器依赖注入 上一篇博客中介绍了IOC容器的初始化.通过源代码分析大致了解了IOC容器初始化的一些知识.先简单回想下上篇的内容 加载bean定义文件的过程.这个过程 ...

  4. Spring IOC源代码具体解释之整体结构

    Spring ICO具体解释之整体结构 IOC介绍 IOC, spring的核心.贯穿Spring始终.直观的来说.就是由spring来负责控制对象的生命周期和对象间的关系,将对象之间的关系抽象出来. ...

  5. MQTT---HiveMQ源代码具体解释(四)插件载入

    源博客地址:http://blog.csdn.net/pipinet123 MQTT交流群:221405150 实现功能 将全部放在plugins文件夹下的全部符合plugin编写规范的plugin ...

  6. MQTT---HiveMQ源代码具体解释(一)概览

    源博客地址:http://blog.csdn.net/pipinet123 MQTT交流群:221405150 面向群体 想自己实现MQTT Broker的朋友 对现有开源的MQTT Broker或多 ...

  7. Android MediaScannerJNI源代码具体解释

    1.简单介绍 MediaScannerJNI的在MediaScanner中的地位可參考 Android MediaScanner 总纲 MediaScanner JNI文件名称: android_me ...

  8. MQTT---HiveMQ源代码具体解释(十八)Cluster-kryo与Serializer

    源博客地址:http://blog.csdn.net/pipinet123 MQTT交流群:221405150 既然是Cluster,node之间肯定是须要交互的,那么肯定是须要序列化和反序列化.Hi ...

  9. MQTT---HiveMQ源代码具体解释(七)Netty-SSL/NoSSL

    源博客地址:http://blog.csdn.net/pipinet123 MQTT交流群:221405150 实现功能 依据用户配置的不同的Listener(TcpListener.TlsTcpLi ...

随机推荐

  1. 杭州电ACM1098——Ignatius's puzzle

    这个话题.简单的数学. 对于函数,f(x)=5*x^13+13*x^5+k*a*x,输入k,对于休闲x,一个数字的存在a,使f(x)是65可分. 对于休闲x. 因此,当x = 1时间,f(x) = 1 ...

  2. HTML学习笔记——各种居中对齐

    0.前言     水平居中基本方法--指定块的宽度并设定块的左右外边距为auto,上下外边距可取0,那么该块能够在父元素中水平居中. 样式例如以下: 1:margin:0px auto 2:margi ...

  3. HDU 4940(杭电更多的学校#7 1006) Destroy Transportation system(到处乱混)

    职务地址:pid=4940">HDU 4940 当时这个题一看就看出来了是网络流的最小割.然后就一直在想建图. .然后突然发现,应该要让T集合的数目最少,不然仅仅要有两个,那这两个的每 ...

  4. Android Studio使用心得 - 简单介绍与环境配置

    FBI Warning:欢迎转载,但请标明出处:http://blog.csdn.net/codezjx/article/details/38544823,未经本人允许请勿用于商业用途.感谢支持! 关 ...

  5. JS数组追加数组採用push.apply的坑

    JS数组追加数组没有现成的函数,这么多年我已经习惯了a.push.apply(a, b);这样的自以为非常酷的,不须要写for循环的写法,一直也没遇到什么问题,直到今天我要append的b是个非常大的 ...

  6. V5

    系统设置--关于手机--版本号点5下--进去开发模式--打开开发选项--打开USB调试.然后在连接第三方助手软件 http://bbs.ztehn.com/thread-19037-1-1.html

  7. 关于SVN配置文件的一个小例子

    1   背景假设 厦门央瞬公司是一家电子元器件设备供应商,其中有个ARM部门,专门负责ARM芯片的方案设计.销售,并在北京.上海各设立了一个办事处.对于工作日志,原先采用邮件方式发给经理,但是这种方式 ...

  8. embedded dylibs/frameworks are only supported on iOS 8.0 and later 错误解决

    ld: warning: embedded dylibs/frameworks only run on iOS 8 or later ld: embedded dylibs/frameworks ar ...

  9. java注解(一)

    虽然平时有使用注解,不过没有深入了解,今天无聊,重新从基础深入了解整理下: java注解是附加在代码中的一些元信息,用于一些工具在编译.运行时进行解析和使用,起到说明.配置的功能.     注解不会也 ...

  10. leetcode第一刷_Convert Sorted List to Binary Search Tree

    好,二叉搜索树粉末登场,有关他的问题有这么几个,给你一个n,如何求全部的n个节点的二叉搜索树个数?能不能把全部的这些二叉搜索树打印出来? 这道题倒不用考虑这么多,直接转即可了,我用的思想是分治,每次找 ...