우선, 이전 서브스레드 실행 연산에 관련되지 않은 내용인 ngx_process_events_and_timers가 있다는 점을 계속해서 상기해 보겠습니다. 오늘은 이 함수에 대해 공부하겠습니다.
이 기사의 출처: http://blog.csdn.net/lengzijian/article/details/7601730
섹션 19의 일부 스크린샷을 살펴보겠습니다.
오늘은 이벤트 중심 기능을 주로 설명하겠습니다. 사진에서 빨간색 부분:
[cpp] view
일반 카피프린트?
- src/event/ngx_event.c
-
- void
- ngx_process_events_and_timers(ngx_cycle_t *cycle)
- {
- ngx_uint_t 플래그;
- ngx_msec_t 타이머, 델타;
-
- if (ngx_timer_solution) {
- 타이머 = NGX_TIMER_INFINITE;
- 플래그 = 0;
-
- } 그 외 {
- 타이머 = ngx_event_find_timer();
- 플래그 = NGX_UPDATE_TIME;
- }
-
- /*
- ngx_use_accept_mutex变weight代表是否使用accept互斥体
- 默认是使用,可通过accept_mutex off;指令关闭;
- 뮤텍스 수락 적작용就是避免惊群,同时实现负载均衡
- */
- (ngx_use_accept_mutex) {
-
- /*
- ngx_accept_disabled变weight 에서ngx_event_accept函数中计算。
- 如果ngx_accept_disabled大于0,就表示该进程接受的链接过多,
- 因此放弃一次争抢수락 mutex의 机会,同时将自己减一。
- 然后 继续处理已有连接上的事件。
- nginx就利用这一点实现了继承关于连接的基本负载均衡。
- */
- 인 경우 (ngx_accept_disabled > 0) {
- ngx_accept_ 장애인--;
-
- } 그밖에 {
- /*
- 尝试锁accept mutex,只有成功获取锁的进程,才会将들어보세요套接字放到epoll中。
- 因此,这就保证了只有一个进程拥有监听套接口,故所有进程阻塞在epoll_wait时,
- 才不会惊群现象。
- */
-
🎜 > (ngx_trylock_accept_mutex(cycle) == NGX_ERROR) { 🎜> >
- 프로세스가 잠금을 획득하면 NGX_POST_EVENTS 플래그가 추가됩니다.
이 플래그의 기능은 생성된 모든 이벤트를 - 대기열에 넣은 다음 이벤트가 해제된 후 천천히 이벤트를 처리하는 것입니다. .
처리 시간이 매우 오래 걸릴 수 있으므로 잠금을 먼저 해제하고 처리하지 않으면 프로세스가 오랫동안 잠금을 차지하게 됩니다. , - 결과적으로 다른 프로세스가 잠금을 얻을 수 없으므로 ACCEPT의 효율성이 낮습니다.
- _POST_EVENTS
-
/*-
획득된 프로세스가 없으며 물론 NGX_POST_EVENTS 플래그도 필요하지 않습니다.
- 하지만 잠금을 위해 싸우기 전에 지연 시간을 설정해야 합니다.> == NGX_TIMER_INFINITE
-
|| 타이머 > ) 타이머 = cept_mutex_delay;
-
} }
-
🎜>
- }
- 델타 = ngx_current_msec;
-
- /*다음으로 epoll은 대기 이벤트를 시작합니다.
ngx_process_events의 구체적인 구현은 epoll 모듈의 ngx_epoll_process_events 함수에 해당합니다. -
-
자세한 내용은 나중에 설명하겠습니다
* /-
- (void) ngx_process_events(주기, 타이머, 플래그) ;
- //이 대기 이벤트의 시간 소비 통계
- delta = ngx_current_msec - 델타
-
ngx_log_debug1(NGX_LOG_DEBUG_EVENT, 사이클->log, 0, " 타이머 델타: %M", 델타);
-
> -
- ngx_posted_accept_events는 epoll이 청취 소켓에서 기다리는 수락 이벤트를 임시로 저장하는 이벤트 queue입니다.
- 위에서 언급한 NGX_POST_EVENTS 플래그가 사용된 후 모든 수락 이벤트는 이 대기열
*/-
-
- ngx_event_process_posted (cycle, &ngx_posted_accept_events) 🎜> //모든 승인 이벤트 처리 후 잠금이 유지되면 해제합니다.
- if (ngx_accept_mutex_held) { ngx_shmtx_unlock(&ngx_accept_mutex) 🎜>
-
delta는 이전에 계산된 시간 소모입니다. 밀리초 수준의 시간 소모가 있는 경우
경우입니다. timeout을 사용하는 경우, time rbtree에서 만료된 타이머를 삭제하고 해당 이벤트의 핸들러 함수를 호출하여 처리 -
- */
- if
(델타) { -
ngx_event_expire_timers( );
- } ,
- > /*
일반 이벤트 처리(연결에서 얻은 이벤트 읽기 및 쓰기), -
-
이벤트마다 고유한 핸들러 메소드가 있으므로
*/-
- if (ngx_posted_events) {
-
if- (ngx_threaded) {
- ngx _wakeup_worker_thread(주기)
- } else
{ - ngx_event_process_posted(cycle, &ngx_posted_events)
- } }
- }
앞서 말한 accept 이벤트는 실제로 소켓에서 새 이벤트를 수신합니다. 승인 시간 방법은 아래와 같습니다: ngx_event_accept:
- [cpp] 보기
일반 카피프린트?
- src/event/ngx_event_accept.c
-
- void
- ngx_event_accept(ngx_event_t *ev)
- {
- socklen_t socklen;
- ngx_err_t 오류;
- ngx_log_t *log;
- ngx_socket_t s;
- ngx_event_t *rev, *wev;
- ngx_listening_t *ls;
- ngx_connection_t *c, *lc;
- ngx_event_conf_t *ecf;
- u_char sa[NGX_SOCKADDRLEN];
-
- //省略part代码
-
- lc = ev->data;
- ls = lc->듣기;
- ev->준비 = 0;
-
- ngx_log_debug2(NGX_LOG_DEBUG_EVENT, ev->log, 0,
- "%V에서 수락, 준비: %d", &ls->addr_text, ev->사용 가능);
-
- 하세요 {
- socklen = NGX_SOCKADDRLEN;
- //accept 一个 new连接
- s = accept(lc-> ;fd, (struct sockaddr *) sa, &socklen);
- //省略part代码
-
- /*
-
accept到一个新的连接后,就重新计算ngx_accept_disabled的值,
- 它主要是用来做负载均衡,之前有提过。
-
- 这里,我们可以看到他的就只方式
- “总连接数的八分之一- 剩余的连接数“
- 总连接指每个进程设数,这个数字可以再配置文件中指정입니다.
- ngx_accept_disabled가 0보다 크고 연결이 과부하되었습니다
- 🎜>
-
ngx_accept_disabled = ngx_cycle->connection_n / 8 - - ngx_cycle- >free_connection_n;
>-
c = ngx_get_connection(s, ev->log)
-
-
//새 링크에 대한 메모리 풀 생성
-
-
>
-
c->pool = ngx_create_pool(ls->pool_size, ev->log); 🎜>
- 이온(c );
-
c->sockaddr = ngx_palloc(c->pool, socklen); > if (c->sockaddr == NULL) {
-
ngx_close_accepted_connection(c); > return; c->sockaddr, sa, socklen) 🎜>(ngx_log_t));
>- ngx_close_accepted_connection(c)
- 🎜> }
-
- /* aio에 대한 차단 모드 및 기타에 대한 비차단 모드 설정*/
-
- if (ngx_inherited_nonblocking) {
- 인 경우 (ngx_event_flags & NGX_USE_AIO_EVENT) {
- if (ngx_blocking( s) == -1) {
- ngx_log_error(NGX_LOG_ALERT, ev->log, ngx_socket_errno,
- ngx_blocking_n " 실패했습니다");
- ngx_close_accepted_connection(c);
- 반품;
- }
- }
-
- } else {
- //我们使用epoll模型 ,这里我们设置连接为논블로킹
- (!(ngx_event_flags & (NGX_USE_AIO_EVENT|NGX_USE_RTSIG_EVENT))) {
- if (ngx_nonblocking(s) == -1) {
- ngx_log _error(NGX_LOG_ALERT, ev->log , ngx_socket_errno,
- ngx_nonblocking_n " 실패했습니다");
- ngx_close_accepted_connection(c);
- 반품;
- }
- }
- }
-
- *log = ls->log;
- //初始化新的连接
- c->recv = ngx_rec v;
- c->send = ngx_send;
- c->recv_chain = ngx_recv_chain;
- c->send_chain = ngx_send_chain;
-
- c->log = log;
- c->풀->log = log;
-
- c->socklen = socklen;
- c->듣기 = ls;
- c->local_sockaddr = ls->sockaddr;
-
- c->unexpected_eof = 1;
-
- #if (NGX_HAVE_UNIX_DOMAIN)
- if (c->sockaddr->sa_family == AF_UNIX) {
- c->tcp_nopush = NGX_ TCP_NOPUSH_DISABLED ;
- c->tcp_nodelay = NGX_TCP_NODELAY_DISABLED;
- #if (NGX_SOLARIS)
- /* Solaris의 sendfilev() AF_NCA, AF_INET 및 AF_INET6 지원*/
- c->sendfile = 0;
- #endif
- }
- #endif
-
- rev = c->read;
- wev = c->write;
-
- wev->ready = 1;
-
-
(ngx_event_flags & (NGX_USE_AIO_EVENT|NGX_USE_RTSIG_EVENT)) {
- /* rtsig, aio, iocp */
- rev->ready = 1;
- }
-
- (ev- >deferred_accept) {
- rev->준비 = 1;
- #if (NGX_HAVE_KQUEUE)
- rev->사용 가능 = 1;
- #endif
- }
-
- rev->log = log;
- wev->log = log;
-
- /*
- * TODO: MT: - ngx_atomic_fetch_add()
- * 또는 중요한 섹션 또는 가벼운 뮤텍스에 의한 보호
- *
- * TODO: MP: - 공유 메모리에 할당됨
- * - ngx_atomic_fetch_add()
- * 또는 중요한 섹션이나 가벼운 뮤텍스에 의한 보호
- * /
-
- c->number = ngx_atomic_fetch_add(ngx_connection_counter, 1);
-
- 인 경우 (ngx_add_conn && (ngx_event_flags & NGX_USE_EPOLL_EVENT) == 0) {
- 인 경우 (ngx_add_conn(c) == NGX_ERROR) {
- ngx_close_accepted_connection(c );
- 반품;
- }
- }
-
- 로그->데이터 = NULL;
- 로그->핸들러 = NULL;
-
- /*
- 这里듣기 핸들러很重要,它将完成新连接的最后初始化工事,
- 동종挂에는 핸들러가 있습니다.函数,
- 就是ngx_http_init_connection 지금 지后http模块中재详细介绍
- */
- ls->handler(c);
-
- if (ngx_event_flags & NGX_USE_KQUEUE_EVENT) {
- ev->사용 가능--;
- }
-
- } 동안 (ev ->사용 가능);
- }
accpt事件的handler방식也就是如此了。지后就是每个连接的读写事件handler방식,这一part分会直接将我们引入http模块,我们还不急,还要school习下nginx经典模块epoll。
위 내용은 nginx 소스 코드 연구 노트(21) - 이벤트 모듈 2 - 대기열 콘텐츠를 포함한 이벤트 중심 코어 ngx_process_events_and_timers를 소개합니다. PHP 튜토리얼에 관심이 있는 친구들에게 도움이 되기를 바랍니다.