Nginx源码分析 - Event事件篇 - Event模块和配置的初始化(16)

目录

一、Event模块的数据结构

1. event事件模块的配置

2. ngx_events_module 事件模块

3. ngx_event_core_module 事件核心模块

4. ngx_event_conf_t 事件conf结构

二、Event模块结构图

三、Event模块的初始化

1. ngx_event_module_init 模块初始化函数

四、Event模块配置的初始化

1. ngx_events_module 模块配置初始化

2. ngx_conf_parse 解析顶层“event”的配置

3. ngx_events_block 解析events块block中的内容

4. ngx_event_core_create_conf和ngx_event_core_init_conf

5. ngx_event_get_conf获取event配置

一、Event模块的数据结构

1. event事件模块的配置

events {  
        use epoll;  
        worker_connections  1024;  
}  

event事件模块,配置分为两层:ngx_events_module 事件模块 和 ngx_event_core_module 事件核心模块。

ngx_events_module:模块类型NGX_CORE_MODULE,所以此模块在最外层核心模块解析“events”命令的时候会回调ngx_events_block函数。
ngx_event_core_module:模块类型NGX_EVENT_MODULE,所以此模块在ngx_events_block函数被回调后,才能解析配置信息
最外层的events模块,类型NGX_CORE_MODULE,属于核心模块,核心模块在最开始配置文件初始化的时候,就会调用指令的命令集。所以在核心模块启动的时候就会调用events的模块配置解析指令函数:ngx_events_block2. ngx_events_module 事件模块

 

2. ngx_event_core_module 事件核心模块

3. ngx_event_conf_t 事件conf结构

二、Event模块结构图

 

三、Event模块的初始化

1. ngx_event_module_init 模块初始化函数

ngx_event_module_init方法为事件核心模块的初始化函数。

/**
 * event事件核心模块初始化函数
 */
static ngx_int_t ngx_event_module_init(ngx_cycle_t *cycle) {
	void ***cf;
	u_char *shared;
	size_t size, cl;
	ngx_shm_t shm;
	ngx_time_t *tp;
	ngx_core_conf_t *ccf;
	ngx_event_conf_t *ecf;
 
	/* 获取配置信息 */
	cf = ngx_get_conf(cycle->conf_ctx, ngx_events_module);
	ecf = (*cf)[ngx_event_core_module.ctx_index];
 
	if (!ngx_test_config && ngx_process <= NGX_PROCESS_MASTER) {
		ngx_log_error(NGX_LOG_NOTICE, cycle->log, 0,
				"using the \"%s\" event method", ecf->name);
	}
 
	ccf = (ngx_core_conf_t *) ngx_get_conf(cycle->conf_ctx, ngx_core_module);
 
	ngx_timer_resolution = ccf->timer_resolution;
 
#if !(NGX_WIN32)
	{
		ngx_int_t limit;
		struct rlimit rlmt;
 
		if (getrlimit(RLIMIT_NOFILE, &rlmt) == -1) {
			ngx_log_error(NGX_LOG_ALERT, cycle->log, ngx_errno,
					"getrlimit(RLIMIT_NOFILE) failed, ignored");
 
		} else {
			if (ecf->connections > (ngx_uint_t) rlmt.rlim_cur
					&& (ccf->rlimit_nofile == NGX_CONF_UNSET
							|| ecf->connections
									> (ngx_uint_t) ccf->rlimit_nofile)) {
				limit = (ccf->rlimit_nofile == NGX_CONF_UNSET) ?
						(ngx_int_t) rlmt.rlim_cur : ccf->rlimit_nofile;
 
				ngx_log_error(NGX_LOG_WARN, cycle->log, 0,
						"%ui worker_connections exceed "
								"open file resource limit: %i",
						ecf->connections, limit);
			}
		}
	}
#endif /* !(NGX_WIN32) */
 
	if (ccf->master == 0) {
		return NGX_OK;
	}
 
	if (ngx_accept_mutex_ptr) {
		return NGX_OK;
	}
 
	/* cl should be equal to or greater than cache line size */
 
	cl = 128;
 
	size = cl /* ngx_accept_mutex */
	+ cl /* ngx_connection_counter */
	+ cl; /* ngx_temp_number */
 
#if (NGX_STAT_STUB)
 
	size += cl /* ngx_stat_accepted */
	+ cl /* ngx_stat_handled */
	+ cl /* ngx_stat_requests */
	+ cl /* ngx_stat_active */
	+ cl /* ngx_stat_reading */
	+ cl /* ngx_stat_writing */
	+ cl; /* ngx_stat_waiting */
 
#endif
 
	shm.size = size;
	shm.name.len = sizeof("nginx_shared_zone") - 1;
	shm.name.data = (u_char *) "nginx_shared_zone";
	shm.log = cycle->log;
 
	if (ngx_shm_alloc(&shm) != NGX_OK) {
		return NGX_ERROR;
	}
 
	shared = shm.addr;
 
	ngx_accept_mutex_ptr = (ngx_atomic_t *) shared;
	ngx_accept_mutex.spin = (ngx_uint_t) -1;
 
	if (ngx_shmtx_create(&ngx_accept_mutex, (ngx_shmtx_sh_t *) shared,
			cycle->lock_file.data) != NGX_OK) {
		return NGX_ERROR;
	}
 
	ngx_connection_counter = (ngx_atomic_t *) (shared + 1 * cl);
 
	(void) ngx_atomic_cmp_set(ngx_connection_counter, 0, 1);
 
	ngx_log_debug2(NGX_LOG_DEBUG_EVENT, cycle->log, 0,
			"counter: %p, %uA",
			ngx_connection_counter, *ngx_connection_counter);
 
	ngx_temp_number = (ngx_atomic_t *) (shared + 2 * cl);
 
	tp = ngx_timeofday();
 
	ngx_random_number = (tp->msec << 16) + ngx_pid;
 
#if (NGX_STAT_STUB)
 
	ngx_stat_accepted = (ngx_atomic_t *) (shared + 3 * cl);
	ngx_stat_handled = (ngx_atomic_t *) (shared + 4 * cl);
	ngx_stat_requests = (ngx_atomic_t *) (shared + 5 * cl);
	ngx_stat_active = (ngx_atomic_t *) (shared + 6 * cl);
	ngx_stat_reading = (ngx_atomic_t *) (shared + 7 * cl);
	ngx_stat_writing = (ngx_atomic_t *) (shared + 8 * cl);
	ngx_stat_waiting = (ngx_atomic_t *) (shared + 9 * cl);
 
#endif
 
	return NGX_OK;
}

四、Event模块配置的初始化

1. ngx_events_module 模块配置初始化

ngx_init_cycle方法中的模块初始化。ngx_events_module类型为NGX_CORE_MODULE,所以在ngx_init_cycle中就会进行核心模块的初始化。

但是ngx_events_module中的create_conf方法为NULL,所以不会调用创建配置的方法。

    /*
     * 核心模块的配置文件创建
     * 配置创建调用nginx.c 中的 ngx_core_module_create_conf
     * */
    for (i = 0; cycle->modules[i]; i++) {
        if (cycle->modules[i]->type != NGX_CORE_MODULE) {
            continue;
        }
 
        module = cycle->modules[i]->ctx;
 
        if (module->create_conf) {
            rv = module->create_conf(cycle); //模块回调函数,创建模块的配置信息
            if (rv == NULL) {
                ngx_destroy_pool(pool);
                return NULL;
            }
            cycle->conf_ctx[cycle->modules[i]->index] = rv; //配置文件复制
        }
    }

2. ngx_conf_parse 解析顶层“event”的配置

ngx_init_cycle方法中会调用ngx_conf_parse方法,并且解析的/usr/local/nginx/conf/nginx.conf配置文件。此次调用只解析最顶层的配置信息“events”,而不会解析{}块中的内容
 

    /* 解析命令行中的配置参数;例如:nginx -t -c /usr/local/nginx/conf/nginx.conf */
    if (ngx_conf_param(&conf) != NGX_CONF_OK) {
        environ = senv;
        ngx_destroy_cycle_pools(&conf);
        return NULL;
    }
 
    /* 解析配置文件/usr/local/nginx/conf/nginx.conf 信息 */
    if (ngx_conf_parse(&conf, &cycle->conf_file) != NGX_CONF_OK) {
        environ = senv;
        ngx_destroy_cycle_pools(&conf);
        return NULL;
    }

3. ngx_events_block 解析events块block中的内容

ngx_events_block方法为ngx_events_commands命令集的回调函数。在最顶层解析nginx.conf文件的时候,会进行核心模块的命令集遍历。(参考: Nginx源码分析 - 主流程篇 - 解析配置文件    中的ngx_conf_handler)   会遍历模块命令集的cmd->set方法。ngx_events_block中主要创建ngx_event_core_module事件的核心模块以及配置信息。

/**
 * 模块解析
 * 事件模块配置如下:
 * events {
    worker_connections  1024;
    }
         光使用核心配置的方式,只能解析到 events 这一层。
         如果需要继续往{}中的内容解析,就得重新调用ngx_conf_parse进行解析
 */
static char *
ngx_events_block(ngx_conf_t *cf, ngx_command_t *cmd, void *conf) {
	char *rv;
	void ***ctx;
	ngx_uint_t i;
	ngx_conf_t pcf;
	ngx_event_module_t *m;
 
	if (*(void **) conf) {
		return "is duplicate";
	}
 
	/* count the number of the event modules and set up their indices */
 
	ngx_event_max_module = ngx_count_modules(cf->cycle, NGX_EVENT_MODULE);
 
	/* 分配内存空间 */
	ctx = ngx_pcalloc(cf->pool, sizeof(void *));
	if (ctx == NULL) {
		return NGX_CONF_ERROR ;
	}
 
	*ctx = ngx_pcalloc(cf->pool, ngx_event_max_module * sizeof(void *));
	if (*ctx == NULL) {
		return NGX_CONF_ERROR ;
	}
 
	*(void **) conf = ctx;
 
	/* 模块初始化,如果是NGX_EVENT_MODULE,则调用模块的create_conf方法 */
	for (i = 0; cf->cycle->modules[i]; i++) {
		if (cf->cycle->modules[i]->type != NGX_EVENT_MODULE) {
			continue;
		}
 
		m = cf->cycle->modules[i]->ctx;
 
		if (m->create_conf) {
			(*ctx)[cf->cycle->modules[i]->ctx_index] = m->create_conf(
					cf->cycle);
			if ((*ctx)[cf->cycle->modules[i]->ctx_index] == NULL) {
				return NGX_CONF_ERROR ;
			}
		}
	}
 
	/* event*/
	pcf = *cf;
	cf->ctx = ctx;
	cf->module_type = NGX_EVENT_MODULE;
	cf->cmd_type = NGX_EVENT_CONF;
 
	/* 调用配置解析,这次解析的是 块中的内容,非文件内容 */
	rv = ngx_conf_parse(cf, NULL);
 
	*cf = pcf;
 
	if (rv != NGX_CONF_OK) {
		return rv;
	}
 
	/* 初始化 模块的init_conf 方法*/
	for (i = 0; cf->cycle->modules[i]; i++) {
		if (cf->cycle->modules[i]->type != NGX_EVENT_MODULE) {
			continue;
		}
 
		m = cf->cycle->modules[i]->ctx;
 
		if (m->init_conf) {
			rv = m->init_conf(cf->cycle,
					(*ctx)[cf->cycle->modules[i]->ctx_index]);
			if (rv != NGX_CONF_OK) {
				return rv;
			}
		}
	}
 
	return NGX_CONF_OK;
}

4. ngx_event_core_create_conf和ngx_event_core_init_conf

  • ngx_event_core_create_conf:主要是创建event事件核心模块
  • ngx_event_core_init_conf:初始化event事件核心模块
/**
 * 创建Event的核心配置文件
 */
static void *
ngx_event_core_create_conf(ngx_cycle_t *cycle) {
    ngx_event_conf_t *ecf;
 
    /* 分配配置文件内容 */
    ecf = ngx_palloc(cycle->pool, sizeof(ngx_event_conf_t));
    if (ecf == NULL) {
        return NULL;
    }
 
    /* 设置默认值 */
    ecf->connections = NGX_CONF_UNSET_UINT;
    ecf->use = NGX_CONF_UNSET_UINT;
    ecf->multi_accept = NGX_CONF_UNSET;
    ecf->accept_mutex = NGX_CONF_UNSET;
    ecf->accept_mutex_delay = NGX_CONF_UNSET_MSEC;
    ecf->name = (void *) NGX_CONF_UNSET;
 
#if (NGX_DEBUG)
 
    if (ngx_array_init(&ecf->debug_connection, cycle->pool, 4,
                    sizeof(ngx_cidr_t)) == NGX_ERROR)
    {
        return NULL;
    }
 
#endif
 
    return ecf;
}
 
/**
 * 初始化Event的核心配置文件
 */
static char *
ngx_event_core_init_conf(ngx_cycle_t *cycle, void *conf) {
    ngx_event_conf_t *ecf = conf;
 
#if (NGX_HAVE_EPOLL) && !(NGX_TEST_BUILD_EPOLL)
    int fd;
#endif
    ngx_int_t i;
    ngx_module_t *module;
    ngx_event_module_t *event_module;
 
    module = NULL;
 
#if (NGX_HAVE_EPOLL) && !(NGX_TEST_BUILD_EPOLL)
 
    fd = epoll_create(100);
 
    if (fd != -1) {
        (void) close(fd);
        module = &ngx_epoll_module;
 
    } else if (ngx_errno != NGX_ENOSYS) {
        module = &ngx_epoll_module;
    }
 
#endif
 
#if (NGX_HAVE_DEVPOLL) && !(NGX_TEST_BUILD_DEVPOLL)
 
    module = &ngx_devpoll_module;
 
#endif
 
#if (NGX_HAVE_KQUEUE)
 
    module = &ngx_kqueue_module;
 
#endif
 
#if (NGX_HAVE_SELECT)
 
    if (module == NULL) {
        module = &ngx_select_module;
    }
 
#endif
 
    /**
     * 查询使用的事件模型:epoll、kqueue等
     * 因为在模块初始化的时候,epoll\kqueue等event的模型模块都会被初始化
     * 但是每个服务器只能选择一种相应的事件模型,所以选择一个适合自己的模块
     */
    if (module == NULL) {
        for (i = 0; cycle->modules[i]; i++) {
 
            if (cycle->modules[i]->type != NGX_EVENT_MODULE) {
                continue;
            }
 
            event_module = cycle->modules[i]->ctx;
 
            if (ngx_strcmp(event_module->name->data, event_core_name.data)
                    == 0) {
                continue;
            }
 
            module = cycle->modules[i];
            break;
        }
    }
 
    if (module == NULL) {
        ngx_log_error(NGX_LOG_EMERG, cycle->log, 0, "no events module found");
        return NGX_CONF_ERROR ;
    }
 
    ngx_conf_init_uint_value(ecf->connections, DEFAULT_CONNECTIONS);
    cycle->connection_n = ecf->connections;
 
    /**
     * 存储使用的事件模型模块索引 例如:epoll、kqueue
     * nginx.conf中存储的是:use epoll;
     * 这里会找到cycle->modules的具体模块的索引值,存储最终的索引值
     */
    ngx_conf_init_uint_value(ecf->use, module->ctx_index);
 
    event_module = module->ctx;
    ngx_conf_init_ptr_value(ecf->name, event_module->name->data);
 
    ngx_conf_init_value(ecf->multi_accept, 0);
    ngx_conf_init_value(ecf->accept_mutex, 1);
    ngx_conf_init_msec_value(ecf->accept_mutex_delay, 500);
 
    return NGX_CONF_OK;
}

5. ngx_event_get_conf获取event配置

获取event的配置,先获取ngx_events_module配置,然后再到ngx_events_module模块上找到ngx_event_core_module 事件核心模块的配置

#define ngx_event_get_conf(conf_ctx, module)                                  \
             (*(ngx_get_conf(conf_ctx, ngx_events_module))) [module.ctx_index];

参考地址:

1.https://initphp.blog.csdn.net/article/details/52434261

2. 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值