nginx_palloc
来源:互联网 发布:新浪微博加粉软件 编辑:程序博客网 时间:2024/06/14 03:45
http://wenku.baidu.com/view/31fe434dfe4733687e21aae1.html
http://apps.hi.baidu.com/share/detail/33027791
http://down.chinaz.com/server/201201/1577_1.htm
http://www.abo321.org/archives/109
http://www.cnblogs.com/sld666666/archive/2010/06/27/1766255.html
多个pool 使用d.next指针来窜起来,large链表指向大块的分配,
特定的分配算法有特定的应用场合!!
/*
* Copyright (C) Igor Sysoev
*/
#ifndef _NGX_PALLOC_H_INCLUDED_
#define _NGX_PALLOC_H_INCLUDED_
#include <ngx_config.h>
#include <ngx_core.h>
/*
* NGX_MAX_ALLOC_FROM_POOL should be (ngx_pagesize - 1), i.e. 4095 on x86.
* On Windows NT it decreases a number of locked pages in a kernel.
*/
#define NGX_MAX_ALLOC_FROM_POOL (ngx_pagesize - 1)
#define NGX_DEFAULT_POOL_SIZE (16 * 1024)
#define NGX_POOL_ALIGNMENT 16
#define NGX_MIN_POOL_SIZE \
ngx_align((sizeof(ngx_pool_t) + 2 * sizeof(ngx_pool_large_t)), \
NGX_POOL_ALIGNMENT)
typedef void (*ngx_pool_cleanup_pt)(void *data);
typedef struct ngx_pool_cleanup_s ngx_pool_cleanup_t;
struct ngx_pool_cleanup_s {
ngx_pool_cleanup_pt handler;
void *data;
ngx_pool_cleanup_t *next;
};
typedef struct ngx_pool_large_s ngx_pool_large_t;
struct ngx_pool_large_s {
ngx_pool_large_t *next;
void *alloc;
};
typedef struct {
u_char *last;
u_char *end;
ngx_pool_t *next;
ngx_uint_t failed;
} ngx_pool_data_t;
struct ngx_pool_s {
ngx_pool_data_t d;
size_t max;
ngx_pool_t *current;
ngx_chain_t *chain;
ngx_pool_large_t *large;
ngx_pool_cleanup_t *cleanup;
ngx_log_t *log;
};
typedef struct {
ngx_fd_t fd;
u_char *name;
ngx_log_t *log;
} ngx_pool_cleanup_file_t;
void *ngx_alloc(size_t size, ngx_log_t *log);
void *ngx_calloc(size_t size, ngx_log_t *log);
ngx_pool_t *ngx_create_pool(size_t size, ngx_log_t *log);
void ngx_destroy_pool(ngx_pool_t *pool);
void ngx_reset_pool(ngx_pool_t *pool);
void *ngx_palloc(ngx_pool_t *pool, size_t size);
void *ngx_pnalloc(ngx_pool_t *pool, size_t size);
void *ngx_pcalloc(ngx_pool_t *pool, size_t size);
void *ngx_pmemalign(ngx_pool_t *pool, size_t size, size_t alignment);
ngx_int_t ngx_pfree(ngx_pool_t *pool, void *p);
ngx_pool_cleanup_t *ngx_pool_cleanup_add(ngx_pool_t *p, size_t size);
void ngx_pool_run_cleanup_file(ngx_pool_t *p, ngx_fd_t fd);
void ngx_pool_cleanup_file(void *data);
void ngx_pool_delete_file(void *data);
#endif /* _NGX_PALLOC_H_INCLUDED_ */
/*
* Copyright (C) Igor Sysoev
*/
#include <ngx_config.h>
#include <ngx_core.h>
static void *ngx_palloc_block( ngx_pool_t *pool, size_t size );
static void *ngx_palloc_large( ngx_pool_t *pool, size_t size );
ngx_pool_t *
ngx_create_pool( size_t size, ngx_log_t *log )
{
ngx_pool_t *p;
// 地址从16的倍数开始
p = ngx_memalign( NGX_POOL_ALIGNMENT, size, log );
if ( p == NULL )
{
return NULL;
}
//创建新pool,last从第一个位置开始,分配后会向下移动
p->d.last = (u_char *) p + sizeof ( ngx_pool_t );
p->d.end = (u_char *) p + size; // end指向最后,不变
p->d.next = NULL; // 下一个pool为null
p->d.failed = 0; // 失败次数初始化为0
size = size - sizeof ( ngx_pool_t );
p->max = (size < NGX_MAX_ALLOC_FROM_POOL) ? size : NGX_MAX_ALLOC_FROM_POOL;
p->current = p; //当前pool指向p
p->chain = NULL;
p->large = NULL;
p->cleanup = NULL;
p->log = log;
return p;
}
void
ngx_destroy_pool( ngx_pool_t *pool )
{
ngx_pool_t *p, *n;
ngx_pool_large_t *l;
ngx_pool_cleanup_t *c;
//调用cleanup链表中的每个cleanup的handler函数来回收c的data
for ( c = pool->cleanup; c; c = c->next )
{
if ( c->handler )
{
ngx_log_debug1( NGX_LOG_DEBUG_ALLOC, pool->log, 0,
"run cleanup: %p", c );
c->handler( c->data );
}
}
//free 掉large链表中的所有alloc不为null的项,free large块时已经释放了一部分了
for ( l = pool->large; l; l = l->next )
{
ngx_log_debug1(NGX_LOG_DEBUG_ALLOC, pool->log, 0, "free: %p", l->alloc);
if ( l->alloc )
{
ngx_free(l->alloc);
}
}
#if (NGX_DEBUG)
/*
* we could allocate the pool->log from this pool
* so we can not use this log while the free()ing the pool
*/
for (p = pool, n = pool->d.next; /* void */; p = n, n = n->d.next)
{
ngx_log_debug2(NGX_LOG_DEBUG_ALLOC, pool->log, 0,
"free: %p, unused: %uz", p, p->d.end - p->d.last);
if (n == NULL)
{
break;
}
}
#endif
//释放pool开始的,d的next指向的各个pool,这里才释放pool,程序运行时是不会释放的
//考虑http处理时的特性!!!!!不是多次分配和释放,而是一次性大量分配,然后一次性释放所有!!
for ( p = pool, n = pool->d.next; /* void */; p = n, n = n->d.next )
{
ngx_free( p );
if ( n == NULL )
{
break;
}
}
}
void
ngx_reset_pool( ngx_pool_t *pool )
{
ngx_pool_t *p;
ngx_pool_large_t *l;
// free掉large链表中的所有项
for ( l = pool->large; l; l = l->next)
{
if ( l->alloc )
{
ngx_free( l->alloc );
}
}
pool->large = NULL;
//不free pool指向的所有项,仅仅只是调整d.last的指针指向最初位置,调整pool开始的所有的pool链表
for ( p = pool; p; p = p->d.next )
{
p->d.last = (u_char *) p + sizeof(ngx_pool_t);
}
}
//指针按照4字节对齐分配
void *
ngx_palloc( ngx_pool_t *pool, size_t size )
{
u_char *m;
ngx_pool_t *p;
if ( size <= pool->max )
{
p = pool->current;
//从current开始,遍历所有的pool,如果找到了,就调整last指针,如果没有找到就看下一个pool
do
{
m = ngx_align_ptr( p->d.last, NGX_ALIGNMENT );
if ( (size_t) (p->d.end - m) >= size )
{
p->d.last = m + size;
return m;
}
p = p->d.next;
} while ( p );
//所有的pool中都没有,只能再次分配一个全新的pool了,挂接到当前pool链表中
return ngx_palloc_block( pool, size );
}
//比最大块还要大,只能malloc一下
return ngx_palloc_large( pool, size );
}
//指针没有按照4字节对齐的分配
void *
ngx_pnalloc( ngx_pool_t *pool, size_t size )
{
u_char *m;
ngx_pool_t *p;
if ( size <= pool->max )
{
//从上一次成功分配空间的指针current开始,向下找,一直找到last,
p = pool->current;
do
{
m = p->d.last;
if ( (size_t) ( p->d.end - m ) >= size )
{
p->d.last = m + size;
return m;
}
p = p->d.next;
} while ( p );
//从current找到last还是没找到,则只能再次分配一个block了
return ngx_palloc_block( pool, size );
}
return ngx_palloc_large( pool, size );
}
//分配一个新的pool,链接到pool的d.next指向的pool链表中
static void *
ngx_palloc_block( ngx_pool_t *pool, size_t size )
{
u_char *m;
size_t psize;
ngx_pool_t *p, *newp, *current;
psize = (size_t) ( pool->d.end - (u_char *) pool );
m = ngx_memalign( NGX_POOL_ALIGNMENT, psize, pool->log );
if ( m == NULL )
{
return NULL;
}
newp = (ngx_pool_t *) m;
newp->d.end = m + psize;
newp->d.next = NULL;
newp->d.failed = 0;
m += sizeof ( ngx_pool_data_t );
m = ngx_align_ptr( m, NGX_ALIGNMENT );
newp->d.last = m + size;
//current指向的块可能已经失败次数过多了,则不看current了,向下找,找到一个failed小于4的项
//d.next为单向链表,只向后找,不向前找,current一旦新增了就只向后看了,分配的时候一般都是先用前面的,
//前面的不够了才用后面的
current = pool->current;
for ( p = current; p->d.next; p = p->d.next )
{
if ( p->d.failed++ > 4 )
{
current = p->d.next;
}
}
p->d.next = newp; // 链接到d.next pool 链表中
//调整current指针为当前failed小于4的项,可能分配新项
pool->current = current ? current : newp;
return m;
}
static void *
ngx_palloc_large( ngx_pool_t *pool, size_t size )
{
void *p;
ngx_uint_t n;
ngx_pool_large_t *large;
//分配size大小的项
p = ngx_alloc( size, pool->log );
if ( p == NULL )
{
return NULL;
}
n = 0;
//在large链表中找到一项,alloc设置为刚分配的p,只扫描3项
for ( large = pool->large; large; large = large->next )
{
if ( large->alloc == NULL )
{
large->alloc = p;
return p;
}
if ( n++ > 3 )
{
break;
}
}
//分配一个ngx_pool_large_t结构,将之加入到pool的large链表的头部
large = ngx_palloc( pool, sizeof ( ngx_pool_large_t ) );
if ( large == NULL )
{
ngx_free( p );
return NULL;
}
large->alloc = p;
large->next = pool->large;
pool->large = large;
return p;
}
//分配一个large_t结构,分配size大小的内存,加入到pool的large链表的头部
void *
ngx_pmemalign( ngx_pool_t *pool, size_t size, size_t alignment )
{
void *p;
ngx_pool_large_t *large;
p = ngx_memalign( alignment, size, pool->log );
if ( p == NULL )
{
return NULL;
}
large = ngx_palloc( pool, sizeof ( ngx_pool_large_t ) );
if ( large == NULL )
{
ngx_free( p );
return NULL;
}
large->alloc = p;
large->next = pool->large;
pool->large = large;
return p;
}
//只处理large链表,不管pool里面的内存,在destroy的时候一次性干掉之
ngx_int_t
ngx_pfree( ngx_pool_t *pool, void *p )
{
ngx_pool_large_t *l;
//首先在large链表中找,如果是large块,则直接free
for ( l = pool->large; l; l = l->next)
{
if ( p == l->alloc )
{
ngx_log_debug1(NGX_LOG_DEBUG_ALLOC, pool->log, 0,
"free: %p", l->alloc);
ngx_free( l->alloc );
l->alloc = NULL;
return NGX_OK;
}
}
return NGX_DECLINED;
}
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;
}
ngx_pool_cleanup_t *
ngx_pool_cleanup_add( ngx_pool_t *p, size_t size )
{
ngx_pool_cleanup_t *c;
c = ngx_palloc( p, sizeof ( ngx_pool_cleanup_t ) );
if ( c == NULL )
{
return NULL;
}
if ( size )
{
c->data = ngx_palloc( p, size );
if ( c->data == NULL )
{
return NULL;
}
}
else
{
c->data = NULL;
}
c->handler = NULL;
c->next = p->cleanup;
p->cleanup = c;
ngx_log_debug1( NGX_LOG_DEBUG_ALLOC, p->log, 0, "add cleanup: %p", c );
return c;
}
void
ngx_pool_run_cleanup_file( ngx_pool_t *p, ngx_fd_t fd )
{
ngx_pool_cleanup_t *c;
ngx_pool_cleanup_file_t *cf;
for ( c = p->cleanup; c; c = c->next )
{
if ( c->handler == ngx_pool_cleanup_file )
{
cf = c->data;
if ( cf->fd == fd )
{
c->handler( cf );
c->handler = NULL;
return;
}
}
}
}
void
ngx_pool_cleanup_file( void *data )
{
ngx_pool_cleanup_file_t *c = data;
ngx_log_debug1( NGX_LOG_DEBUG_ALLOC, c->log, 0, "file cleanup: fd:%d",
c->fd );
if ( ngx_close_file( c->fd ) == NGX_FILE_ERROR )
{
ngx_log_error(NGX_LOG_ALERT, c->log, ngx_errno,
ngx_close_file_n " \"%s\" failed", c->name);
}
}
void
ngx_pool_delete_file( void *data )
{
ngx_pool_cleanup_file_t *c = data;
ngx_err_t err;
ngx_log_debug2(NGX_LOG_DEBUG_ALLOC, c->log, 0, "file cleanup: fd:%d %s",
c->fd, c->name);
if ( ngx_delete_file( c->name ) == NGX_FILE_ERROR )
{
err = ngx_errno;
if ( err != NGX_ENOENT )
{
ngx_log_error( NGX_LOG_CRIT, c->log, err,
ngx_delete_file_n " \"%s\" failed", c->name );
}
}
if ( ngx_close_file( c->fd ) == NGX_FILE_ERROR )
{
ngx_log_error( NGX_LOG_ALERT, c->log, ngx_errno,
ngx_close_file_n " \"%s\" failed", c->name );
}
}
#if 0
static void *
ngx_get_cached_block(size_t size)
{
void *p;
ngx_cached_block_slot_t *slot;
if (ngx_cycle->cache == NULL)
{
return NULL;
}
slot = &ngx_cycle->cache[(size + ngx_pagesize - 1) / ngx_pagesize];
slot->tries++;
if (slot->number)
{
p = slot->block;
slot->block = slot->block->next;
slot->number--;
return p;
}
return NULL;
}
#endif
- nginx_palloc
- ClientScript.RegisterStartupScript使用说明
- GridView DataSource为null时,显示表头
- c#委托事件与lambda (整理)
- 解决IIS出现server application unavailable问题的方法
- java张孝祥(4) 枚举
- nginx_palloc
- VB工程--百例43--图形绘制
- Linux如何在系统启动时自动加载内核模块
- Android UI开发第六篇——仿QQ的滑动Tab
- win2008 R2 64位系统下配置DCOM权限
- USB CYPRESS 68013A开发重点讲解
- HTTP协议详解(转)
- VC输出“烫”和“屯”
- linux中删除文件后磁盘空间没有释放的解决