02Nginx源码分析之总结内存开辟的相关函数(ngx_alloc.c,ngx_palloc.c)

02Nginx源码分析之总结内存开辟的相关函数(ngx_alloc.c,ngx_palloc.c)

前言:
写这一篇的目的是为了方便接下来的阅读,否则每次都要对应查找相应的函数,非常麻烦。

并且注意:
下面所说的内存对齐,实际上是通过倍数来对齐的,例如按照256对齐,此时只是将返回的首地址变成256的倍数,若想对齐,必须调用ngx_memalign时的参1也传256,这样就可以对齐,即0-256-512,第一次获取内存返回首地址是0(不需要关心开辟的大小),那么第二次开辟内存按照256对齐后,它下次返回的首地址必然是256,所以这样就对齐了。但是如果你第二次对齐的倍数不是256,而是4,8,16,32这些,那么就有可能造成不对齐的情况。

这就是本篇所说的内存对齐,这个内存对齐相对于对齐宏ngx_align来说,更强调对齐首地址。而后者是将整个对齐倍数(即若对齐为16,开辟大小为9,那么调用对齐宏实际占用16的大小)进行对齐,需要区别下一篇所说的内存对齐。

1 首先说ngx_alloc.c源文件中的开辟函数
1)ngx_alloc函数
作用:单独调用malloc开辟内存。

/*
 * 功能:调用malloc开辟内存
*/
void *
ngx_alloc(size_t size, ngx_log_t *log)
{
    void  *p;

    p = malloc(size);
    if (p == NULL) {
        ngx_log_error(NGX_LOG_EMERG, log, ngx_errno,
                      "malloc(%uz) failed", size);
    }

    ngx_log_debug2(NGX_LOG_DEBUG_ALLOC, log, 0, "malloc: %p:%uz", p, size);

    return p;
}

2)ngx_calloc函数
和上面其实一样,只不过多了一步清零工作。

/*
 * 功能:调用ngx_alloc开辟内存并进行清零操作
*/
void *
ngx_calloc(size_t size, ngx_log_t *log)
{
    void  *p;

    p = ngx_alloc(size, log);

    if (p) {
        ngx_memzero(p, size);
    }

    return p;
}

3)ngx_memalign函数
实际和上面也是一样的,也是开辟内存,只不过多了内存对齐的工作。所以要想内存对齐则调用此函数,否则调用上面的函数即可。

/*
 * 功能:根据宏定义取相应函数,但是功能实现都是一样的。
 * posix_memalign和memalign都是开辟一个size大小的内存空间,并且返回的值与alignment成倍数即内存对齐。它与另一个宏函数ngx_align_ptr是ngx使大内存内存对齐的关键
*/


#if (NGX_HAVE_POSIX_MEMALIGN)

void *
ngx_memalign(size_t alignment, size_t size, ngx_log_t *log)
{
    void  *p;
    int    err;

    err = posix_memalign(&p, alignment, size);//开辟一个内存块,并且传出的p的首地址与alignment成倍数关系即内存对齐

    if (err) {
        ngx_log_error(NGX_LOG_EMERG, log, err,
                      "posix_memalign(%uz, %uz) failed", alignment, size);
        p = NULL;
    }

    ngx_log_debug3(NGX_LOG_DEBUG_ALLOC, log, 0,
                   "posix_memalign: %p:%uz @%uz", p, size, alignment);

    return p;
}

#elif (NGX_HAVE_MEMALIGN)

void *
ngx_memalign(size_t alignment, size_t size, ngx_log_t *log)
{
    void  *p;

    p = memalign(alignment, size);//开辟一个内存块,并且返回的p的首地址与alignment成倍数关系即内存对齐,与上面区别是一个是传出一个是返回值返回
    if (p == NULL) {
        ngx_log_error(NGX_LOG_EMERG, log, ngx_errno,
                      "memalign(%uz, %uz) failed", alignment, size);
    }

    ngx_log_debug3(NGX_LOG_DEBUG_ALLOC, log, 0,
                   "memalign: %p:%uz @%uz", p, size, alignment);

    return p;
}

2 ngx_alloc.c源文件中的开辟函数
1)ngx_palloc函数和ngx_pnalloc函数
这两个函数是最重要的。
但这里只是简单说一下ngx_palloc和ngx_pnalloc,具体里面的ngx_palloc_small和ngx_palloc_large可以到下一篇内存池的文章中理解。
理清ngx_palloc和ngx_pnalloc:实际上这两个函数是一样的,只不过ngx_palloc穿1会进行内存对齐,后者穿0不对齐。所以单独讲ngx_palloc即可。

ngx_palloc:
1)若开辟的内存小于等于内存池的max,则调用小内存开辟,否则调用大内存开辟。
2)开辟小内存ngx_palloc_small的逻辑:在内存池的链表中查找是否有可用的内存,没有则调用ngx_palloc_block开辟新的内存池中给samll,此时内存必定在内存池链表上。
3)开辟大内存ngx_palloc_large的逻辑:调用ngx_alloc开辟一块大内存,交给对应的large结构体管理然后返回。实际上每次开辟大内存他会对前三个管理结构体判断其中的alloc是否为空,若为空则交给前三个管理,否则跳出交给新创建的large结构体管理,此时内存必定在大内存块链表上。

上面的这几句话总结对后面的内存对齐的这几个函数理解非常有用。

void *
ngx_palloc(ngx_pool_t *pool, size_t size)
{
#if !(NGX_DEBUG_PALLOC)
    if (size <= pool->max) {
        return ngx_palloc_small(pool, size, 1);
    }
#endif

    return ngx_palloc_large(pool, size);
}


//ngx_pnalloc同理。
void *
ngx_pnalloc(ngx_pool_t *pool, size_t size)
{
#if !(NGX_DEBUG_PALLOC)
    if (size <= pool->max) {
        return ngx_palloc_small(pool, size, 0);
    }
#endif

    return ngx_palloc_large(pool, size);
}

2)ngx_pcalloc函数
调用上面的ngx_palloc,c代表callback。

/*
 * 重新说一篇ngx_palloc()的作用:调用ngx_palloc_small或者ngx_palloc_large申请一片内存返回。申请的内存在内存池或者大数据块上。
 * 它的作用和ngx_palloc可以是一样的,只不过多了一步清零操作。
*/
void *
ngx_pcalloc(ngx_pool_t *pool, size_t size)
{
    void *p;

    p = ngx_palloc(pool, size);
    if (p) {
        ngx_memzero(p, size);
    }

    return p;
}

3)ngx_pmemalign函数
注意函数名多了个p对比最上面的ngx_alloc.c源文件,这个函数比较少被调用,所以只需要了解一下即可。

/*
 * 它的作用和ngx_palloc_large可以是基本一样的,有两个不同点:
 * 1)调用内存对齐的函数ngx_memalign创建内存,而非像ngx_palloc_large内调用不内存对齐的ngx_alloc。
 * 2)不再去检查前面三个大内存是否为空,而是直接将新创建的大内存p按照头插法放进大内存链表管理。
*/
void *
ngx_pmemalign(ngx_pool_t *pool, size_t size, size_t alignment)
{
    void              *p;
    ngx_pool_large_t  *large;

	// 1)创建一个内存对齐的大内存
    p = ngx_memalign(alignment, size, pool->log);
    if (p == NULL) {
        return NULL;
    }

	// 2)创建管理大内存的结构体,并在下面赋值进行管理该内存
    large = ngx_palloc_small(pool, sizeof(ngx_pool_large_t), 1);
    if (large == NULL) {
        ngx_free(p);
        return NULL;
    }

	//头插法插进大内存链表
    large->alloc = p;
    large->next = pool->large;
    pool->large = large;

    return p;
}

3 总结
看懂上面后,以后只需瞄一眼这里就可,非常方便。

1)ngx_alloc函数
作用:单独调用malloc开辟内存。
2)ngx_calloc函数
调用ngx_alloc增加清零操作。可认为与ngx_alloc一样。
3)ngx_memalign函数
开辟内存并内存对齐。

4)ngx_palloc函数和ngx_pnalloc函数
这两个函数调用small和large开辟内存。
1)ngx_palloc_samll首先在已有内存池中查找可用内存,没有则调用block开辟新内存,并给新内存池管理再插入到内存池链表,所以调用这两个函数的内存在内存池链表上。但实际并不连续,只是内存池结构体连续。所以调用samll必定在内存池链表。
2)而ngx_palloc_large是调用ngx_alloc直接开辟内存后交给ngx_pool_large_t结构体管理,再插入大内存块链表。所以该函数开辟的内存不在内存池。而是必定在大内存块链表中。
所以:开辟的内存在内存池链表或者在大内存块链表。

5)ngx_pcalloc函数
调用ngx_palloc,增加清零操作。可认为与ngx_palloc一样。

额外函数(较少使用)6)ngx_pmemalign函数
调用ngx_memalign开辟内存对齐的空间,然后交给ngx_pool_large_t结构体管理,再插入大内存块链表。
所以可认为与ngx_palloc_large一样,只不过ngx_palloc_large少了内存对齐但多了检查前三个结构的步骤。

所以可以总结:
1)前三个都是开辟无任何关系管理的内存。
2)后三个需要进行区分:
1:ngx_palloc和ngx_pnalloc开辟的内存在内存池链表或者在大内存块链表。
2ngx_palloc_samll(即block)开辟的内存必定在内存池结构体链表上管理。
3:ngx_palloc_large和ngx_pmemalign开辟的内存必定在大内存块结构体的链表上管理。
  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值