整个配置解析主要是函数ngx_init_cycle(&init_cycle)进行处理。
ngx_init_cycle(&init_cycle)
ngx_time_update()//时间更新,也是在main函数里面讲过
/* * 通过加锁和解锁,来更新如下时间 ngx_cached_time = tp; ngx_cached_http_time.data = p0; ngx_cached_err_log_time.data = p1; ngx_cached_http_log_time.data = p2; ngx_cached_http_log_iso8601.data = p3; */
log = old_cycle->log;//错误日志对象
pool = ngx_create_pool(NGX_CYCLE_POOL_SIZE,log);//分配一块16k的内存池,第一个节点大约是16k,如果新增小块内存节点,那么该节点最大为4095
cycle->pool = pool; cycle->log = log; cycle->new_log.log_level = NGX_LOG_ERR; cycle->conf_prefix; //设置配置文件dir cycle->prefix; //设置运行环境dir cycle->conf_file; //设置配置文件绝对路径 cycle->conf_param; //保存参数 cycle->pathes; //路径数组 cycle->open_files; //分配打开的文件描述符,其中这个结构是一 个链表,每个链表节点都有一个n个size的 cycle->shared_memory; //分配n个共享内存节点,这个是一个list, 每个节点都存在n个ngx_shm_zone_t结构,通过next指向链表后面的节点。 cycle->listening; //listening数组 cycle->conf_ctx = ngx_pcalloc(pool, ngx_max_module*sizeof(void*));
为cycle分配ngx_max_module个上下文指针
首先创建核心模块的上下文结构,保存在cycle->conf_ctx相应的结构里面
核心模块如下:
NGX_CORE_MODULE index Ngx_core_module 0 Ngx_errlog_module 1 Ngx_reg_module 6 Ngx_events_module 3 Ngx_http_module 7
核心代码
for (i = 0; ngx_modules[i]; i++) { if (ngx_modules[i]->type != NGX_CORE_MODULE) { continue; } module = ngx_modules[i]->ctx; if (module->create_conf) { rv = module->create_conf(cycle); if (rv == NULL) { ngx_destroy_pool(pool); return NULL; } cycle->conf_ctx[ngx_modules[i]->index] = rv; } }
nginx首先创建NGX_CORE_MODULE是为了创建后面的模块上下文做铺垫,也就是
NGX_CORE_MODULE是其他模块的基础,如下:
其中Ngx_errlog_module、ngx_events_module、ngx_http_module模块都没有create_conf
方法,是根据用户的conf实际动态的进行创建。
conf.args = ngx_array_create(pool, 10, sizeof(ngx_str_t)); if (conf.args == NULL) { ngx_destroy_pool(pool); return NULL; } conf.temp_pool = ngx_create_pool(NGX_CYCLE_POOL_SIZE, log); if (conf.temp_pool == NULL) { ngx_destroy_pool(pool); return NULL; } conf.ctx = cycle->conf_ctx; conf.cycle = cycle; conf.pool = pool; conf.log = log; conf.module_type = NGX_CORE_MODULE; conf.cmd_type = NGX_MAIN_CONF;
为conf创建参数,用于保存配置文件的用户配置key和value,同时创建temp_pool内
存池,conf->ctx指向cycle->conf_ctx,保存cycle、pool、log,设置模块类型和命令类型,
开始解析主配置文件变量。
解析配置文件的过程主要是ngx_conf_parse进行间接递归调用,主要分为main作用域 、event作用域、http作用域、server作用域、location作用域,然后接下来就进行分析。
在解析过程中,对配置文件(nginx.conf)分为三类:
enum{ parse_file = 0, //解析文件,比如nginx.conf 还有include文件 parse_block, //解析块,比如http{}块 parse_param //解析参数 比如deamon off }
进入ngx_conf_parse函数,根据是否是解析文件、解析块、解析参数,来设置不同的值,然后进入ngx_conf_read_token,这个函数,主要是读取文件,然后设置一些旗变量,保存用户配置到conf->args里面,其中在解析配置文件(nginx.conf)的时候使用一个buffer内存,主要读取文件,然后解析buffer,解析过程是流式,如果buffer后面还有未解析完,但是没有遇到分号or大括号之类的,就会把未解析完移动到buffer的头部,然后在继续读文件,进行解析,每调用玩一次ngx_conf_read_token就会解析完一个命令,如果配置文件配置有问题,就会报错。解析完一个用户的设置后,保存在conf->args数组中。如果发现cf->handler有值,就调用cf->handler进行回调处理(一般处理mine.types文件),如果cf->handler为空,那么就调用ngx_conf_handler,遍历每一个模块的cmd,来case,找到该命令,并且调用命令的set回调方法,如果cmd是一个key
value的变量设置,给相应模块变量设置完后,就调用ngx_conf_read_token,如果是一个块的key,比如http、event、location,他们的回调函数(cmd->set)首先是创建上下文结构变量,然后间接递归ngx_conf_parse,当然就会设置不同的上下文。