#内存管理的艺术# 之 Nginx slab的实现 — 第一篇“基本布局”

#内存管理的艺术# 之 Nginx slab的实现 — 第一篇“基本布局”

访问这里,获取更多原创内容。

    说明:本系列的文章基于Nginx-1.5.0版本代码。

    Nginx slab分配器用于管理和分配小于一页的内存申请,但实际上大于一页的内存分配也是统一实现的,  具体代码在core/ngx_slab.c文件中,对应的头文件是core/ngx_slab.h。

    ngx_slab.h头文件中定义了两个重要的数据结构:

ngx_slab_pool_t;/*整个内存区的管理结构*/
ngx_slab_page_t;/*用于表示page页内存管理单元和slot分级管理单元*/

    同时还声明了对外提供的几个函数原型,分别是:

    用于初始化内存管理结构的:

void ngx_slab_init(ngx_slab_pool_t *pool);

    用于内存分配的:

void *ngx_slab_alloc(ngx_slab_pool_t *pool, size_t size);
void *ngx_slab_alloc_locked(ngx_slab_pool_t *pool, size_t size);
void *ngx_slab_calloc(ngx_slab_pool_t *pool, size_t size);
void *ngx_slab_calloc_locked(ngx_slab_pool_t *pool, size_t size);
/*alloc 和 calloc的区别在于是否在分配的同时将内存清零*/

    用于内存释放的:

void ngx_slab_free(ngx_slab_pool_t *pool, void *p);
void ngx_slab_free_locked(ngx_slab_pool_t *pool, void *p);

    这一次我们先从ngx_slab_init()函数开始,看看slab分配器的基本结构是怎么样的。

    由于slab管理和分配的内存是以2的幂次方为基准向上取整的,也就是说,如果你想申请一块20bytes的内存,那么就向上取整到32bytes。   

#内存管理的艺术# 之 Nginx slab的实现 --- 第一篇“基本布局”

    ngx_slab.c的实现中将内存分配分为两大类:

  • 基于页(page)的内存分配,由page页内存管理单元来进行管理,其实现相对简单,因为整个内存池的基本结构就是以页为单位进行划分的。

  • 基于块(chunk)的内存分配,将一页划分为若干块,实现相对复杂,除了page页内存管理单元外还引入了分级内存管理单元(slot数组)来共同管理;

    实际上,page页内存管理单元和slot分级管理单元都是由ngx_slab_page_t结构来表示的,slot分级管理数组紧跟在ngx_slab_pool_t结构之后,page页内存管理数组又紧跟在slot分级管理数组之后。

void
ngx_slab_init(ngx_slab_pool_t *pool)
{
    u_char            *p;
    size_t            size;
    ngx_int_t         m;
    ngx_uint_t        i, n, pages;
    ngx_slab_page_t   *slots;

    /* STUB */
    if (ngx_slab_max_size == 0) {
        /*slab分配器最大分配大小(slab分配器用于分配小于一页的内存申请,由于实际会以2的幂次方为基准向上取整,因此超过1/2页大小的内存申请也被取整为1页)*/
        ngx_slab_max_size = ngx_pagesize / 2;
        /*当使用bit位来标记块的使用情况时,如果想用一个uintptr_t类型的数来标记一整页中的所有块,则需要将一页分为(8 * sizeof(uintptr_t)个块,每块大小为ngx_slab_exact_size*/
        ngx_slab_exact_size = ngx_pagesize / (8 * sizeof(uintptr_t));
        /*根据ngx_slab_exact_size计算对应的块大小移位ngx_slab_exact_shift*/
        for (n = ngx_slab_exact_size; n >>= 1; ngx_slab_exact_shift++) {
            /* void */
        }
    }
    /**/

    /*slab的最小分配单元,当申请的内存比min_size还小时,则取整到min_size;
    通常min_shift=3,则min_size=8*/
    pool->min_size = 1 << pool->min_shift;

    p = (u_char *) pool + sizeof(ngx_slab_pool_t);
    size = pool->end - p;/*除ngx_slab_pool_t之外的剩余空间大小*/

    ngx_slab_junk(p, size);

    /*slot分级管理数组的起始地址*/
    slots = (ngx_slab_page_t *) p;
    /*从最小块大小到页大小之间的分级数*/
    n = ngx_pagesize_shift - pool->min_shift;

    for (i = 0; i < n; i++) {
        slots[i].slab = 0;
        slots[i].next = &slots[i];/*表明分级数组中还没有要管理的页*/
        slots[i].prev = 0;
    }

    p += n * sizeof(ngx_slab_page_t);
    
    /*每一个实际的page页都对应一个页内存管理单元(后面会看到,反过来则不成立),这里会有疑问为什么size上没有减去slot分级数组占用的空间,下面会说明*/
    pages = (ngx_uint_t) (size / (ngx_pagesize + sizeof(ngx_slab_page_t)));

    /*初始化页内存管理数组*/
    ngx_memzero(p, pages * sizeof(ngx_slab_page_t));

    pool->pages = (ngx_slab_page_t *) p;

    pool->free.prev = 0;
    pool->free.next = (ngx_slab_page_t *) p;

    /*页内存管理单元中的slab字段记录了其后跟随的连续空闲内存页数*/
    pool->pages->slab = pages;
    pool->pages->next = &pool->free;
    pool->pages->prev = (uintptr_t) &pool->free;

    /*将实际的page页起始地址对齐到pagesize*/
    pool->start = (u_char *)
                  ngx_align_ptr((uintptr_t) p + pages * sizeof(ngx_slab_page_t),
                                ngx_pagesize);

    /*对齐后剩余的内存空间可能不足pages个页,需要进行调整;另外也可以看出实际的内存页数可能会少于page页管理单元的数目,多余的几个就空闲在最后好了,这也是为什么上面计算pages时size并没有减去slot分级数组大小的原因,因为一切都是由最终对齐后的内存空间大小决定的,所以前面也就不必要求那么精确了*/
    m = pages - (pool->end - pool->start) / ngx_pagesize;
    if (m > 0) {
        pages -= m;
        pool->pages->slab = pages;
    }

    pool->log_ctx = &pool->zero;
    pool->zero = '
void
ngx_slab_init(ngx_slab_pool_t *pool)
{
    u_char            *p;
    size_t            size;
    ngx_int_t         m;
    ngx_uint_t        i, n, pages;
    ngx_slab_page_t   *slots;
    /* STUB */
    if (ngx_slab_max_size == 0) {
        /*slab分配器最大分配大小(slab分配器用于分配小于一页的内存申请,由于实际会以2的幂次方为基准向上取整,因此超过1/2页大小的内存申请也被取整为1页)*/
        ngx_slab_max_size = ngx_pagesize / 2;
        /*当使用bit位来标记块的使用情况时,如果想用一个uintptr_t类型的数来标记一整页中的所有块,则需要将一页分为(8 * sizeof(uintptr_t)个块,每块大小为ngx_slab_exact_size*/
        ngx_slab_exact_size = ngx_pagesize / (8 * sizeof(uintptr_t));
        /*根据ngx_slab_exact_size计算对应的块大小移位ngx_slab_exact_shift*/
        for (n = ngx_slab_exact_size; n >>= 1; ngx_slab_exact_shift++) {
            /* void */
        }
    }
    /**/
    /*slab的最小分配单元,当申请的内存比min_size还小时,则取整到min_size;
    通常min_shift=3,则min_size=8*/
    pool->min_size = 1 << pool->min_shift;
    p = (u_char *) pool + sizeof(ngx_slab_pool_t);
    size = pool->end - p;/*除ngx_slab_pool_t之外的剩余空间大小*/
    ngx_slab_junk(p, size);
    /*slot分级管理数组的起始地址*/
    slots = (ngx_slab_page_t *) p;
    /*从最小块大小到页大小之间的分级数*/
    n = ngx_pagesize_shift - pool->min_shift;
    for (i = 0; i < n; i++) {
        slots[i].slab = 0;
        slots[i].next = &slots[i];/*表明分级数组中还没有要管理的页*/
        slots[i].prev = 0;
    }
    p += n * sizeof(ngx_slab_page_t);
    
    /*每一个实际的page页都对应一个页内存管理单元(后面会看到,反过来则不成立),这里会有疑问为什么size上没有减去slot分级数组占用的空间,下面会说明*/
    pages = (ngx_uint_t) (size / (ngx_pagesize + sizeof(ngx_slab_page_t)));
    /*初始化页内存管理数组*/
    ngx_memzero(p, pages * sizeof(ngx_slab_page_t));
    pool->pages = (ngx_slab_page_t *) p;
    pool->free.prev = 0;
    pool->free.next = (ngx_slab_page_t *) p;
    /*页内存管理单元中的slab字段记录了其后跟随的连续空闲内存页数*/
    pool->pages->slab = pages;
    pool->pages->next = &pool->free;
    pool->pages->prev = (uintptr_t) &pool->free;
    /*将实际的page页起始地址对齐到pagesize*/
    pool->start = (u_char *)
ngx_align_ptr((uintptr_t) p + pages * sizeof(ngx_slab_page_t),
ngx_pagesize);
    /*对齐后剩余的内存空间可能不足pages个页,需要进行调整;另外也可以看出实际的内存页数可能会少于page页管理单元的数目,多余的几个就空闲在最后好了,这也是为什么上面计算pages时size并没有减去slot分级数组大小的原因,因为一切都是由最终对齐后的内存空间大小决定的,所以前面也就不必要求那么精确了*/
    m = pages - (pool->end - pool->start) / ngx_pagesize;
    if (m > 0) {
        pages -= m;
        pool->pages->slab = pages;
    }
    pool->log_ctx = &pool->zero;
    pool->zero = '\0';
}
'; }

    初始化完成之后,整个内存结构布局就是这个样子滴:

004735_vI7v_2310891.png

    图中的各个标记基本保持与ngx_slab_init()函数中一致, 其中,N = pages – m,即经过对齐调整后的实际内存页数。

 

    有了这个图做铺垫,我们再讨论page页的分配时就容易多了。

 

 

 

转载于:https://my.oschina.net/u/2310891/blog/672539

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。

发布者:全栈程序员-用户IM,转载请注明出处:https://javaforall.cn/109031.html原文链接:https://javaforall.cn

【正版授权,激活自己账号】: Jetbrains全家桶Ide使用,1年售后保障,每天仅需1毛

【官方授权 正版激活】: 官方授权 正版激活 支持Jetbrains家族下所有IDE 使用个人JB账号...

(0)
blank

相关推荐

  • dex文件格式

    dex文件格式dex文件格式Android4.0源码Dalvik/docs目录下文档dex-format.html有详细介绍dex文件格式1.dex文件中的数据结构dex文件使用到的数据类型u1~u8表示1到8字节的无符号数,而sleb128、uleb128与uleb128p1则是dex文件中特有的LEB128数据类型。每个LEB128由1~5个字节组成,所有的字节组合在一起表示一…

  • linux设置法语键盘布局,法语键盘布局图「建议收藏」

    linux设置法语键盘布局,法语键盘布局图「建议收藏」《法语键盘布局图》由会员分享,可在线阅读,更多相关《法语键盘布局图(3页珍藏版)》请在人人文库网上搜索。1、法语键盘布局(适用环境:法语输入法,法语键盘布局)法语键盘布局介绍一、与英文键盘(普通键盘)的区别1.a和q的位置互换了2.z和W位置互换3.m和“分号和冒号键”是方便的表达方式,本文中提到的“特定键”是指国际标准键盘的键(即市场上销售的普通键盘键上印刷的字符),下同。“位置互换二。键盘上的…

    2022年10月23日
  • createthread dll「建议收藏」

    createthread dll「建议收藏」CreateThreadapi内部会调用waitforsingleobject等待互斥量对象。目的是同步顺序执行dll初始化。当该方法创建完线程内核对象和线程盏后,该函数内部会调用进程映射中所有dll的dllmain方法进行初始化。因此在自己写的dll中不要创建线程并使用waitforsingleobject等待线程创建。因为如果A线程创建的时候调用了dll中的dllmain函数,并且该

  • docker打开2375监听端口「建议收藏」

    docker打开2375监听端口「建议收藏」由于在使用caliper时,需要用到Docker的监听端口,所以此步骤如下:1、修改/usr/lib/systemd/system/docker.service,在[service]的ExecStart,添加-Htcp://0.0.0.0:2375ExecStart=/usr/bin/dockerd-Htcp://0.0.0.0:2375-Hfd://–containerd=/run/containerd/containerd.sock2、刷新配置文件,重启docker

  • 图片触及翻转效果 css3

    图片触及翻转效果 css3

  • 你知道如何从零开始学c++游戏编程吗「建议收藏」

    你知道如何从零开始学c++游戏编程吗「建议收藏」在软件开发中,游戏开发这个方向看起来目标很明确,但其实是个领域很广的方向,入门的时候如果得不到指点一二,很容易误入歧途,相反,如果走这条路之前能得到前人的一些指路,是可以事半功倍的。平台与编程语言选择首先,游戏开发的平台就有很多类型:个人主机平台:Windows、Linux、MacOC;移动平台:iOS、Android、WindowsPhone、BlackBerryOS、Symbian;专业主…

发表回复

您的电子邮箱地址不会被公开。

关注全栈程序员社区公众号