|  |  | 
|  | /* | 
|  | * Copyright (C) Igor Sysoev | 
|  | * Copyright (C) Nginx, Inc. | 
|  | */ | 
|  |  | 
|  |  | 
|  | #include <ngx_config.h> | 
|  | #include <ngx_core.h> | 
|  | #include <ngx_event.h> | 
|  |  | 
|  |  | 
|  | ngx_os_io_t  ngx_io; | 
|  |  | 
|  |  | 
|  | static void ngx_drain_connections(ngx_cycle_t *cycle); | 
|  |  | 
|  |  | 
|  | ngx_listening_t * | 
|  | ngx_create_listening(ngx_conf_t *cf, struct sockaddr *sockaddr, | 
|  | socklen_t socklen) | 
|  | { | 
|  | size_t            len; | 
|  | ngx_listening_t  *ls; | 
|  | struct sockaddr  *sa; | 
|  | u_char            text[NGX_SOCKADDR_STRLEN]; | 
|  |  | 
|  | ls = ngx_array_push(&cf->cycle->listening); | 
|  | if (ls == NULL) { | 
|  | return NULL; | 
|  | } | 
|  |  | 
|  | ngx_memzero(ls, sizeof(ngx_listening_t)); | 
|  |  | 
|  | sa = ngx_palloc(cf->pool, socklen); | 
|  | if (sa == NULL) { | 
|  | return NULL; | 
|  | } | 
|  |  | 
|  | ngx_memcpy(sa, sockaddr, socklen); | 
|  |  | 
|  | ls->sockaddr = sa; | 
|  | ls->socklen = socklen; | 
|  |  | 
|  | len = ngx_sock_ntop(sa, socklen, text, NGX_SOCKADDR_STRLEN, 1); | 
|  | ls->addr_text.len = len; | 
|  |  | 
|  | switch (ls->sockaddr->sa_family) { | 
|  | #if (NGX_HAVE_INET6) | 
|  | case AF_INET6: | 
|  | ls->addr_text_max_len = NGX_INET6_ADDRSTRLEN; | 
|  | break; | 
|  | #endif | 
|  | #if (NGX_HAVE_UNIX_DOMAIN) | 
|  | case AF_UNIX: | 
|  | ls->addr_text_max_len = NGX_UNIX_ADDRSTRLEN; | 
|  | len++; | 
|  | break; | 
|  | #endif | 
|  | case AF_INET: | 
|  | ls->addr_text_max_len = NGX_INET_ADDRSTRLEN; | 
|  | break; | 
|  | default: | 
|  | ls->addr_text_max_len = NGX_SOCKADDR_STRLEN; | 
|  | break; | 
|  | } | 
|  |  | 
|  | ls->addr_text.data = ngx_pnalloc(cf->pool, len); | 
|  | if (ls->addr_text.data == NULL) { | 
|  | return NULL; | 
|  | } | 
|  |  | 
|  | ngx_memcpy(ls->addr_text.data, text, len); | 
|  |  | 
|  | #if !(NGX_WIN32) | 
|  | ngx_rbtree_init(&ls->rbtree, &ls->sentinel, ngx_udp_rbtree_insert_value); | 
|  | #endif | 
|  |  | 
|  | ls->fd = (ngx_socket_t) -1; | 
|  | ls->type = SOCK_STREAM; | 
|  |  | 
|  | ls->backlog = NGX_LISTEN_BACKLOG; | 
|  | ls->rcvbuf = -1; | 
|  | ls->sndbuf = -1; | 
|  |  | 
|  | #if (NGX_HAVE_SETFIB) | 
|  | ls->setfib = -1; | 
|  | #endif | 
|  |  | 
|  | #if (NGX_HAVE_TCP_FASTOPEN) | 
|  | ls->fastopen = -1; | 
|  | #endif | 
|  |  | 
|  | return ls; | 
|  | } | 
|  |  | 
|  |  | 
|  | ngx_int_t | 
|  | ngx_clone_listening(ngx_cycle_t *cycle, ngx_listening_t *ls) | 
|  | { | 
|  | #if (NGX_HAVE_REUSEPORT) | 
|  |  | 
|  | ngx_int_t         n; | 
|  | ngx_core_conf_t  *ccf; | 
|  | ngx_listening_t   ols; | 
|  |  | 
|  | if (!ls->reuseport || ls->worker != 0) { | 
|  | return NGX_OK; | 
|  | } | 
|  |  | 
|  | ols = *ls; | 
|  |  | 
|  | ccf = (ngx_core_conf_t *) ngx_get_conf(cycle->conf_ctx, ngx_core_module); | 
|  |  | 
|  | for (n = 1; n < ccf->worker_processes; n++) { | 
|  |  | 
|  | /* create a socket for each worker process */ | 
|  |  | 
|  | ls = ngx_array_push(&cycle->listening); | 
|  | if (ls == NULL) { | 
|  | return NGX_ERROR; | 
|  | } | 
|  |  | 
|  | *ls = ols; | 
|  | ls->worker = n; | 
|  | } | 
|  |  | 
|  | #endif | 
|  |  | 
|  | return NGX_OK; | 
|  | } | 
|  |  | 
|  |  | 
|  | ngx_int_t | 
|  | ngx_set_inherited_sockets(ngx_cycle_t *cycle) | 
|  | { | 
|  | size_t                     len; | 
|  | ngx_uint_t                 i; | 
|  | ngx_listening_t           *ls; | 
|  | socklen_t                  olen; | 
|  | #if (NGX_HAVE_DEFERRED_ACCEPT || NGX_HAVE_TCP_FASTOPEN) | 
|  | ngx_err_t                  err; | 
|  | #endif | 
|  | #if (NGX_HAVE_DEFERRED_ACCEPT && defined SO_ACCEPTFILTER) | 
|  | struct accept_filter_arg   af; | 
|  | #endif | 
|  | #if (NGX_HAVE_DEFERRED_ACCEPT && defined TCP_DEFER_ACCEPT) | 
|  | int                        timeout; | 
|  | #endif | 
|  | #if (NGX_HAVE_REUSEPORT) | 
|  | int                        reuseport; | 
|  | #endif | 
|  |  | 
|  | ls = cycle->listening.elts; | 
|  | for (i = 0; i < cycle->listening.nelts; i++) { | 
|  |  | 
|  | ls[i].sockaddr = ngx_palloc(cycle->pool, sizeof(ngx_sockaddr_t)); | 
|  | if (ls[i].sockaddr == NULL) { | 
|  | return NGX_ERROR; | 
|  | } | 
|  |  | 
|  | ls[i].socklen = sizeof(ngx_sockaddr_t); | 
|  | if (getsockname(ls[i].fd, ls[i].sockaddr, &ls[i].socklen) == -1) { | 
|  | ngx_log_error(NGX_LOG_CRIT, cycle->log, ngx_socket_errno, | 
|  | "getsockname() of the inherited " | 
|  | "socket #%d failed", ls[i].fd); | 
|  | ls[i].ignore = 1; | 
|  | continue; | 
|  | } | 
|  |  | 
|  | if (ls[i].socklen > (socklen_t) sizeof(ngx_sockaddr_t)) { | 
|  | ls[i].socklen = sizeof(ngx_sockaddr_t); | 
|  | } | 
|  |  | 
|  | switch (ls[i].sockaddr->sa_family) { | 
|  |  | 
|  | #if (NGX_HAVE_INET6) | 
|  | case AF_INET6: | 
|  | ls[i].addr_text_max_len = NGX_INET6_ADDRSTRLEN; | 
|  | len = NGX_INET6_ADDRSTRLEN + sizeof("[]:65535") - 1; | 
|  | break; | 
|  | #endif | 
|  |  | 
|  | #if (NGX_HAVE_UNIX_DOMAIN) | 
|  | case AF_UNIX: | 
|  | ls[i].addr_text_max_len = NGX_UNIX_ADDRSTRLEN; | 
|  | len = NGX_UNIX_ADDRSTRLEN; | 
|  | break; | 
|  | #endif | 
|  |  | 
|  | case AF_INET: | 
|  | ls[i].addr_text_max_len = NGX_INET_ADDRSTRLEN; | 
|  | len = NGX_INET_ADDRSTRLEN + sizeof(":65535") - 1; | 
|  | break; | 
|  |  | 
|  | default: | 
|  | ngx_log_error(NGX_LOG_CRIT, cycle->log, ngx_socket_errno, | 
|  | "the inherited socket #%d has " | 
|  | "an unsupported protocol family", ls[i].fd); | 
|  | ls[i].ignore = 1; | 
|  | continue; | 
|  | } | 
|  |  | 
|  | ls[i].addr_text.data = ngx_pnalloc(cycle->pool, len); | 
|  | if (ls[i].addr_text.data == NULL) { | 
|  | return NGX_ERROR; | 
|  | } | 
|  |  | 
|  | len = ngx_sock_ntop(ls[i].sockaddr, ls[i].socklen, | 
|  | ls[i].addr_text.data, len, 1); | 
|  | if (len == 0) { | 
|  | return NGX_ERROR; | 
|  | } | 
|  |  | 
|  | ls[i].addr_text.len = len; | 
|  |  | 
|  | ls[i].backlog = NGX_LISTEN_BACKLOG; | 
|  |  | 
|  | olen = sizeof(int); | 
|  |  | 
|  | if (getsockopt(ls[i].fd, SOL_SOCKET, SO_TYPE, (void *) &ls[i].type, | 
|  | &olen) | 
|  | == -1) | 
|  | { | 
|  | ngx_log_error(NGX_LOG_CRIT, cycle->log, ngx_socket_errno, | 
|  | "getsockopt(SO_TYPE) %V failed", &ls[i].addr_text); | 
|  | ls[i].ignore = 1; | 
|  | continue; | 
|  | } | 
|  |  | 
|  | olen = sizeof(int); | 
|  |  | 
|  | if (getsockopt(ls[i].fd, SOL_SOCKET, SO_RCVBUF, (void *) &ls[i].rcvbuf, | 
|  | &olen) | 
|  | == -1) | 
|  | { | 
|  | ngx_log_error(NGX_LOG_ALERT, cycle->log, ngx_socket_errno, | 
|  | "getsockopt(SO_RCVBUF) %V failed, ignored", | 
|  | &ls[i].addr_text); | 
|  |  | 
|  | ls[i].rcvbuf = -1; | 
|  | } | 
|  |  | 
|  | olen = sizeof(int); | 
|  |  | 
|  | if (getsockopt(ls[i].fd, SOL_SOCKET, SO_SNDBUF, (void *) &ls[i].sndbuf, | 
|  | &olen) | 
|  | == -1) | 
|  | { | 
|  | ngx_log_error(NGX_LOG_ALERT, cycle->log, ngx_socket_errno, | 
|  | "getsockopt(SO_SNDBUF) %V failed, ignored", | 
|  | &ls[i].addr_text); | 
|  |  | 
|  | ls[i].sndbuf = -1; | 
|  | } | 
|  |  | 
|  | #if 0 | 
|  | /* SO_SETFIB is currently a set only option */ | 
|  |  | 
|  | #if (NGX_HAVE_SETFIB) | 
|  |  | 
|  | olen = sizeof(int); | 
|  |  | 
|  | if (getsockopt(ls[i].fd, SOL_SOCKET, SO_SETFIB, | 
|  | (void *) &ls[i].setfib, &olen) | 
|  | == -1) | 
|  | { | 
|  | ngx_log_error(NGX_LOG_ALERT, cycle->log, ngx_socket_errno, | 
|  | "getsockopt(SO_SETFIB) %V failed, ignored", | 
|  | &ls[i].addr_text); | 
|  |  | 
|  | ls[i].setfib = -1; | 
|  | } | 
|  |  | 
|  | #endif | 
|  | #endif | 
|  |  | 
|  | #if (NGX_HAVE_REUSEPORT) | 
|  |  | 
|  | reuseport = 0; | 
|  | olen = sizeof(int); | 
|  |  | 
|  | #ifdef SO_REUSEPORT_LB | 
|  |  | 
|  | if (getsockopt(ls[i].fd, SOL_SOCKET, SO_REUSEPORT_LB, | 
|  | (void *) &reuseport, &olen) | 
|  | == -1) | 
|  | { | 
|  | ngx_log_error(NGX_LOG_ALERT, cycle->log, ngx_socket_errno, | 
|  | "getsockopt(SO_REUSEPORT_LB) %V failed, ignored", | 
|  | &ls[i].addr_text); | 
|  |  | 
|  | } else { | 
|  | ls[i].reuseport = reuseport ? 1 : 0; | 
|  | } | 
|  |  | 
|  | #else | 
|  |  | 
|  | if (getsockopt(ls[i].fd, SOL_SOCKET, SO_REUSEPORT, | 
|  | (void *) &reuseport, &olen) | 
|  | == -1) | 
|  | { | 
|  | ngx_log_error(NGX_LOG_ALERT, cycle->log, ngx_socket_errno, | 
|  | "getsockopt(SO_REUSEPORT) %V failed, ignored", | 
|  | &ls[i].addr_text); | 
|  |  | 
|  | } else { | 
|  | ls[i].reuseport = reuseport ? 1 : 0; | 
|  | } | 
|  | #endif | 
|  |  | 
|  | #endif | 
|  |  | 
|  | if (ls[i].type != SOCK_STREAM) { | 
|  | continue; | 
|  | } | 
|  |  | 
|  | #if (NGX_HAVE_TCP_FASTOPEN) | 
|  |  | 
|  | olen = sizeof(int); | 
|  |  | 
|  | if (getsockopt(ls[i].fd, IPPROTO_TCP, TCP_FASTOPEN, | 
|  | (void *) &ls[i].fastopen, &olen) | 
|  | == -1) | 
|  | { | 
|  | err = ngx_socket_errno; | 
|  |  | 
|  | if (err != NGX_EOPNOTSUPP && err != NGX_ENOPROTOOPT | 
|  | && err != NGX_EINVAL) | 
|  | { | 
|  | ngx_log_error(NGX_LOG_NOTICE, cycle->log, err, | 
|  | "getsockopt(TCP_FASTOPEN) %V failed, ignored", | 
|  | &ls[i].addr_text); | 
|  | } | 
|  |  | 
|  | ls[i].fastopen = -1; | 
|  | } | 
|  |  | 
|  | #endif | 
|  |  | 
|  | #if (NGX_HAVE_DEFERRED_ACCEPT && defined SO_ACCEPTFILTER) | 
|  |  | 
|  | ngx_memzero(&af, sizeof(struct accept_filter_arg)); | 
|  | olen = sizeof(struct accept_filter_arg); | 
|  |  | 
|  | if (getsockopt(ls[i].fd, SOL_SOCKET, SO_ACCEPTFILTER, &af, &olen) | 
|  | == -1) | 
|  | { | 
|  | err = ngx_socket_errno; | 
|  |  | 
|  | if (err == NGX_EINVAL) { | 
|  | continue; | 
|  | } | 
|  |  | 
|  | ngx_log_error(NGX_LOG_NOTICE, cycle->log, err, | 
|  | "getsockopt(SO_ACCEPTFILTER) for %V failed, ignored", | 
|  | &ls[i].addr_text); | 
|  | continue; | 
|  | } | 
|  |  | 
|  | if (olen < sizeof(struct accept_filter_arg) || af.af_name[0] == '\0') { | 
|  | continue; | 
|  | } | 
|  |  | 
|  | ls[i].accept_filter = ngx_palloc(cycle->pool, 16); | 
|  | if (ls[i].accept_filter == NULL) { | 
|  | return NGX_ERROR; | 
|  | } | 
|  |  | 
|  | (void) ngx_cpystrn((u_char *) ls[i].accept_filter, | 
|  | (u_char *) af.af_name, 16); | 
|  | #endif | 
|  |  | 
|  | #if (NGX_HAVE_DEFERRED_ACCEPT && defined TCP_DEFER_ACCEPT) | 
|  |  | 
|  | timeout = 0; | 
|  | olen = sizeof(int); | 
|  |  | 
|  | if (getsockopt(ls[i].fd, IPPROTO_TCP, TCP_DEFER_ACCEPT, &timeout, &olen) | 
|  | == -1) | 
|  | { | 
|  | err = ngx_socket_errno; | 
|  |  | 
|  | if (err == NGX_EOPNOTSUPP) { | 
|  | continue; | 
|  | } | 
|  |  | 
|  | ngx_log_error(NGX_LOG_NOTICE, cycle->log, err, | 
|  | "getsockopt(TCP_DEFER_ACCEPT) for %V failed, ignored", | 
|  | &ls[i].addr_text); | 
|  | continue; | 
|  | } | 
|  |  | 
|  | if (olen < sizeof(int) || timeout == 0) { | 
|  | continue; | 
|  | } | 
|  |  | 
|  | ls[i].deferred_accept = 1; | 
|  | #endif | 
|  | } | 
|  |  | 
|  | return NGX_OK; | 
|  | } | 
|  |  | 
|  |  | 
|  | ngx_int_t | 
|  | ngx_open_listening_sockets(ngx_cycle_t *cycle) | 
|  | { | 
|  | int               reuseaddr; | 
|  | ngx_uint_t        i, tries, failed; | 
|  | ngx_err_t         err; | 
|  | ngx_log_t        *log; | 
|  | ngx_socket_t      s; | 
|  | ngx_listening_t  *ls; | 
|  |  | 
|  | reuseaddr = 1; | 
|  | #if (NGX_SUPPRESS_WARN) | 
|  | failed = 0; | 
|  | #endif | 
|  |  | 
|  | log = cycle->log; | 
|  |  | 
|  | /* TODO: configurable try number */ | 
|  |  | 
|  | for (tries = 5; tries; tries--) { | 
|  | failed = 0; | 
|  |  | 
|  | /* for each listening socket */ | 
|  |  | 
|  | ls = cycle->listening.elts; | 
|  | for (i = 0; i < cycle->listening.nelts; i++) { | 
|  |  | 
|  | if (ls[i].ignore) { | 
|  | continue; | 
|  | } | 
|  |  | 
|  | #if (NGX_HAVE_REUSEPORT) | 
|  |  | 
|  | if (ls[i].add_reuseport) { | 
|  |  | 
|  | /* | 
|  | * to allow transition from a socket without SO_REUSEPORT | 
|  | * to multiple sockets with SO_REUSEPORT, we have to set | 
|  | * SO_REUSEPORT on the old socket before opening new ones | 
|  | */ | 
|  |  | 
|  | int  reuseport = 1; | 
|  |  | 
|  | #ifdef SO_REUSEPORT_LB | 
|  |  | 
|  | if (setsockopt(ls[i].fd, SOL_SOCKET, SO_REUSEPORT_LB, | 
|  | (const void *) &reuseport, sizeof(int)) | 
|  | == -1) | 
|  | { | 
|  | ngx_log_error(NGX_LOG_ALERT, cycle->log, ngx_socket_errno, | 
|  | "setsockopt(SO_REUSEPORT_LB) %V failed, " | 
|  | "ignored", | 
|  | &ls[i].addr_text); | 
|  | } | 
|  |  | 
|  | #else | 
|  |  | 
|  | if (setsockopt(ls[i].fd, SOL_SOCKET, SO_REUSEPORT, | 
|  | (const void *) &reuseport, sizeof(int)) | 
|  | == -1) | 
|  | { | 
|  | ngx_log_error(NGX_LOG_ALERT, cycle->log, ngx_socket_errno, | 
|  | "setsockopt(SO_REUSEPORT) %V failed, ignored", | 
|  | &ls[i].addr_text); | 
|  | } | 
|  | #endif | 
|  |  | 
|  | ls[i].add_reuseport = 0; | 
|  | } | 
|  | #endif | 
|  |  | 
|  | if (ls[i].fd != (ngx_socket_t) -1) { | 
|  | continue; | 
|  | } | 
|  |  | 
|  | if (ls[i].inherited) { | 
|  |  | 
|  | /* TODO: close on exit */ | 
|  | /* TODO: nonblocking */ | 
|  | /* TODO: deferred accept */ | 
|  |  | 
|  | continue; | 
|  | } | 
|  |  | 
|  | s = ngx_socket(ls[i].sockaddr->sa_family, ls[i].type, 0); | 
|  |  | 
|  | if (s == (ngx_socket_t) -1) { | 
|  | ngx_log_error(NGX_LOG_EMERG, log, ngx_socket_errno, | 
|  | ngx_socket_n " %V failed", &ls[i].addr_text); | 
|  | return NGX_ERROR; | 
|  | } | 
|  |  | 
|  | if (ls[i].type != SOCK_DGRAM || !ngx_test_config) { | 
|  |  | 
|  | if (setsockopt(s, SOL_SOCKET, SO_REUSEADDR, | 
|  | (const void *) &reuseaddr, sizeof(int)) | 
|  | == -1) | 
|  | { | 
|  | ngx_log_error(NGX_LOG_EMERG, log, ngx_socket_errno, | 
|  | "setsockopt(SO_REUSEADDR) %V failed", | 
|  | &ls[i].addr_text); | 
|  |  | 
|  | if (ngx_close_socket(s) == -1) { | 
|  | ngx_log_error(NGX_LOG_EMERG, log, ngx_socket_errno, | 
|  | ngx_close_socket_n " %V failed", | 
|  | &ls[i].addr_text); | 
|  | } | 
|  |  | 
|  | return NGX_ERROR; | 
|  | } | 
|  | } | 
|  |  | 
|  | #if (NGX_HAVE_REUSEPORT) | 
|  |  | 
|  | if (ls[i].reuseport && !ngx_test_config) { | 
|  | int  reuseport; | 
|  |  | 
|  | reuseport = 1; | 
|  |  | 
|  | #ifdef SO_REUSEPORT_LB | 
|  |  | 
|  | if (setsockopt(s, SOL_SOCKET, SO_REUSEPORT_LB, | 
|  | (const void *) &reuseport, sizeof(int)) | 
|  | == -1) | 
|  | { | 
|  | ngx_log_error(NGX_LOG_EMERG, log, ngx_socket_errno, | 
|  | "setsockopt(SO_REUSEPORT_LB) %V failed", | 
|  | &ls[i].addr_text); | 
|  |  | 
|  | if (ngx_close_socket(s) == -1) { | 
|  | ngx_log_error(NGX_LOG_EMERG, log, ngx_socket_errno, | 
|  | ngx_close_socket_n " %V failed", | 
|  | &ls[i].addr_text); | 
|  | } | 
|  |  | 
|  | return NGX_ERROR; | 
|  | } | 
|  |  | 
|  | #else | 
|  |  | 
|  | if (setsockopt(s, SOL_SOCKET, SO_REUSEPORT, | 
|  | (const void *) &reuseport, sizeof(int)) | 
|  | == -1) | 
|  | { | 
|  | ngx_log_error(NGX_LOG_EMERG, log, ngx_socket_errno, | 
|  | "setsockopt(SO_REUSEPORT) %V failed", | 
|  | &ls[i].addr_text); | 
|  |  | 
|  | if (ngx_close_socket(s) == -1) { | 
|  | ngx_log_error(NGX_LOG_EMERG, log, ngx_socket_errno, | 
|  | ngx_close_socket_n " %V failed", | 
|  | &ls[i].addr_text); | 
|  | } | 
|  |  | 
|  | return NGX_ERROR; | 
|  | } | 
|  | #endif | 
|  | } | 
|  | #endif | 
|  |  | 
|  | #if (NGX_HAVE_INET6 && defined IPV6_V6ONLY) | 
|  |  | 
|  | if (ls[i].sockaddr->sa_family == AF_INET6) { | 
|  | int  ipv6only; | 
|  |  | 
|  | ipv6only = ls[i].ipv6only; | 
|  |  | 
|  | if (setsockopt(s, IPPROTO_IPV6, IPV6_V6ONLY, | 
|  | (const void *) &ipv6only, sizeof(int)) | 
|  | == -1) | 
|  | { | 
|  | ngx_log_error(NGX_LOG_EMERG, log, ngx_socket_errno, | 
|  | "setsockopt(IPV6_V6ONLY) %V failed, ignored", | 
|  | &ls[i].addr_text); | 
|  | } | 
|  | } | 
|  | #endif | 
|  | /* TODO: close on exit */ | 
|  |  | 
|  | if (!(ngx_event_flags & NGX_USE_IOCP_EVENT)) { | 
|  | if (ngx_nonblocking(s) == -1) { | 
|  | ngx_log_error(NGX_LOG_EMERG, log, ngx_socket_errno, | 
|  | ngx_nonblocking_n " %V failed", | 
|  | &ls[i].addr_text); | 
|  |  | 
|  | if (ngx_close_socket(s) == -1) { | 
|  | ngx_log_error(NGX_LOG_EMERG, log, ngx_socket_errno, | 
|  | ngx_close_socket_n " %V failed", | 
|  | &ls[i].addr_text); | 
|  | } | 
|  |  | 
|  | return NGX_ERROR; | 
|  | } | 
|  | } | 
|  |  | 
|  | ngx_log_debug2(NGX_LOG_DEBUG_CORE, log, 0, | 
|  | "bind() %V #%d ", &ls[i].addr_text, s); | 
|  |  | 
|  | if (bind(s, ls[i].sockaddr, ls[i].socklen) == -1) { | 
|  | err = ngx_socket_errno; | 
|  |  | 
|  | if (err != NGX_EADDRINUSE || !ngx_test_config) { | 
|  | ngx_log_error(NGX_LOG_EMERG, log, err, | 
|  | "bind() to %V failed", &ls[i].addr_text); | 
|  | } | 
|  |  | 
|  | if (ngx_close_socket(s) == -1) { | 
|  | ngx_log_error(NGX_LOG_EMERG, log, ngx_socket_errno, | 
|  | ngx_close_socket_n " %V failed", | 
|  | &ls[i].addr_text); | 
|  | } | 
|  |  | 
|  | if (err != NGX_EADDRINUSE) { | 
|  | return NGX_ERROR; | 
|  | } | 
|  |  | 
|  | if (!ngx_test_config) { | 
|  | failed = 1; | 
|  | } | 
|  |  | 
|  | continue; | 
|  | } | 
|  |  | 
|  | #if (NGX_HAVE_UNIX_DOMAIN) | 
|  |  | 
|  | if (ls[i].sockaddr->sa_family == AF_UNIX) { | 
|  | mode_t   mode; | 
|  | u_char  *name; | 
|  |  | 
|  | name = ls[i].addr_text.data + sizeof("unix:") - 1; | 
|  | mode = (S_IRUSR|S_IWUSR|S_IRGRP|S_IWGRP|S_IROTH|S_IWOTH); | 
|  |  | 
|  | if (chmod((char *) name, mode) == -1) { | 
|  | ngx_log_error(NGX_LOG_EMERG, cycle->log, ngx_errno, | 
|  | "chmod() \"%s\" failed", name); | 
|  | } | 
|  |  | 
|  | if (ngx_test_config) { | 
|  | if (ngx_delete_file(name) == NGX_FILE_ERROR) { | 
|  | ngx_log_error(NGX_LOG_EMERG, cycle->log, ngx_errno, | 
|  | ngx_delete_file_n " %s failed", name); | 
|  | } | 
|  | } | 
|  | } | 
|  | #endif | 
|  |  | 
|  | if (ls[i].type != SOCK_STREAM) { | 
|  | ls[i].fd = s; | 
|  | continue; | 
|  | } | 
|  |  | 
|  | if (listen(s, ls[i].backlog) == -1) { | 
|  | err = ngx_socket_errno; | 
|  |  | 
|  | /* | 
|  | * on OpenVZ after suspend/resume EADDRINUSE | 
|  | * may be returned by listen() instead of bind(), see | 
|  | * https://bugzilla.openvz.org/show_bug.cgi?id=2470 | 
|  | */ | 
|  |  | 
|  | if (err != NGX_EADDRINUSE || !ngx_test_config) { | 
|  | ngx_log_error(NGX_LOG_EMERG, log, err, | 
|  | "listen() to %V, backlog %d failed", | 
|  | &ls[i].addr_text, ls[i].backlog); | 
|  | } | 
|  |  | 
|  | if (ngx_close_socket(s) == -1) { | 
|  | ngx_log_error(NGX_LOG_EMERG, log, ngx_socket_errno, | 
|  | ngx_close_socket_n " %V failed", | 
|  | &ls[i].addr_text); | 
|  | } | 
|  |  | 
|  | if (err != NGX_EADDRINUSE) { | 
|  | return NGX_ERROR; | 
|  | } | 
|  |  | 
|  | if (!ngx_test_config) { | 
|  | failed = 1; | 
|  | } | 
|  |  | 
|  | continue; | 
|  | } | 
|  |  | 
|  | ls[i].listen = 1; | 
|  |  | 
|  | ls[i].fd = s; | 
|  | } | 
|  |  | 
|  | if (!failed) { | 
|  | break; | 
|  | } | 
|  |  | 
|  | /* TODO: delay configurable */ | 
|  |  | 
|  | ngx_log_error(NGX_LOG_NOTICE, log, 0, | 
|  | "try again to bind() after 500ms"); | 
|  |  | 
|  | ngx_msleep(500); | 
|  | } | 
|  |  | 
|  | if (failed) { | 
|  | ngx_log_error(NGX_LOG_EMERG, log, 0, "still could not bind()"); | 
|  | return NGX_ERROR; | 
|  | } | 
|  |  | 
|  | return NGX_OK; | 
|  | } | 
|  |  | 
|  |  | 
|  | void | 
|  | ngx_configure_listening_sockets(ngx_cycle_t *cycle) | 
|  | { | 
|  | int                        value; | 
|  | ngx_uint_t                 i; | 
|  | ngx_listening_t           *ls; | 
|  |  | 
|  | #if (NGX_HAVE_DEFERRED_ACCEPT && defined SO_ACCEPTFILTER) | 
|  | struct accept_filter_arg   af; | 
|  | #endif | 
|  |  | 
|  | ls = cycle->listening.elts; | 
|  | for (i = 0; i < cycle->listening.nelts; i++) { | 
|  |  | 
|  | ls[i].log = *ls[i].logp; | 
|  |  | 
|  | if (ls[i].rcvbuf != -1) { | 
|  | if (setsockopt(ls[i].fd, SOL_SOCKET, SO_RCVBUF, | 
|  | (const void *) &ls[i].rcvbuf, sizeof(int)) | 
|  | == -1) | 
|  | { | 
|  | ngx_log_error(NGX_LOG_ALERT, cycle->log, ngx_socket_errno, | 
|  | "setsockopt(SO_RCVBUF, %d) %V failed, ignored", | 
|  | ls[i].rcvbuf, &ls[i].addr_text); | 
|  | } | 
|  | } | 
|  |  | 
|  | if (ls[i].sndbuf != -1) { | 
|  | if (setsockopt(ls[i].fd, SOL_SOCKET, SO_SNDBUF, | 
|  | (const void *) &ls[i].sndbuf, sizeof(int)) | 
|  | == -1) | 
|  | { | 
|  | ngx_log_error(NGX_LOG_ALERT, cycle->log, ngx_socket_errno, | 
|  | "setsockopt(SO_SNDBUF, %d) %V failed, ignored", | 
|  | ls[i].sndbuf, &ls[i].addr_text); | 
|  | } | 
|  | } | 
|  |  | 
|  | if (ls[i].keepalive) { | 
|  | value = (ls[i].keepalive == 1) ? 1 : 0; | 
|  |  | 
|  | if (setsockopt(ls[i].fd, SOL_SOCKET, SO_KEEPALIVE, | 
|  | (const void *) &value, sizeof(int)) | 
|  | == -1) | 
|  | { | 
|  | ngx_log_error(NGX_LOG_ALERT, cycle->log, ngx_socket_errno, | 
|  | "setsockopt(SO_KEEPALIVE, %d) %V failed, ignored", | 
|  | value, &ls[i].addr_text); | 
|  | } | 
|  | } | 
|  |  | 
|  | #if (NGX_HAVE_KEEPALIVE_TUNABLE) | 
|  |  | 
|  | if (ls[i].keepidle) { | 
|  | value = ls[i].keepidle; | 
|  |  | 
|  | #if (NGX_KEEPALIVE_FACTOR) | 
|  | value *= NGX_KEEPALIVE_FACTOR; | 
|  | #endif | 
|  |  | 
|  | if (setsockopt(ls[i].fd, IPPROTO_TCP, TCP_KEEPIDLE, | 
|  | (const void *) &value, sizeof(int)) | 
|  | == -1) | 
|  | { | 
|  | ngx_log_error(NGX_LOG_ALERT, cycle->log, ngx_socket_errno, | 
|  | "setsockopt(TCP_KEEPIDLE, %d) %V failed, ignored", | 
|  | value, &ls[i].addr_text); | 
|  | } | 
|  | } | 
|  |  | 
|  | if (ls[i].keepintvl) { | 
|  | value = ls[i].keepintvl; | 
|  |  | 
|  | #if (NGX_KEEPALIVE_FACTOR) | 
|  | value *= NGX_KEEPALIVE_FACTOR; | 
|  | #endif | 
|  |  | 
|  | if (setsockopt(ls[i].fd, IPPROTO_TCP, TCP_KEEPINTVL, | 
|  | (const void *) &value, sizeof(int)) | 
|  | == -1) | 
|  | { | 
|  | ngx_log_error(NGX_LOG_ALERT, cycle->log, ngx_socket_errno, | 
|  | "setsockopt(TCP_KEEPINTVL, %d) %V failed, ignored", | 
|  | value, &ls[i].addr_text); | 
|  | } | 
|  | } | 
|  |  | 
|  | if (ls[i].keepcnt) { | 
|  | if (setsockopt(ls[i].fd, IPPROTO_TCP, TCP_KEEPCNT, | 
|  | (const void *) &ls[i].keepcnt, sizeof(int)) | 
|  | == -1) | 
|  | { | 
|  | ngx_log_error(NGX_LOG_ALERT, cycle->log, ngx_socket_errno, | 
|  | "setsockopt(TCP_KEEPCNT, %d) %V failed, ignored", | 
|  | ls[i].keepcnt, &ls[i].addr_text); | 
|  | } | 
|  | } | 
|  |  | 
|  | #endif | 
|  |  | 
|  | #if (NGX_HAVE_SETFIB) | 
|  | if (ls[i].setfib != -1) { | 
|  | if (setsockopt(ls[i].fd, SOL_SOCKET, SO_SETFIB, | 
|  | (const void *) &ls[i].setfib, sizeof(int)) | 
|  | == -1) | 
|  | { | 
|  | ngx_log_error(NGX_LOG_ALERT, cycle->log, ngx_socket_errno, | 
|  | "setsockopt(SO_SETFIB, %d) %V failed, ignored", | 
|  | ls[i].setfib, &ls[i].addr_text); | 
|  | } | 
|  | } | 
|  | #endif | 
|  |  | 
|  | #if (NGX_HAVE_TCP_FASTOPEN) | 
|  | if (ls[i].fastopen != -1) { | 
|  | if (setsockopt(ls[i].fd, IPPROTO_TCP, TCP_FASTOPEN, | 
|  | (const void *) &ls[i].fastopen, sizeof(int)) | 
|  | == -1) | 
|  | { | 
|  | ngx_log_error(NGX_LOG_ALERT, cycle->log, ngx_socket_errno, | 
|  | "setsockopt(TCP_FASTOPEN, %d) %V failed, ignored", | 
|  | ls[i].fastopen, &ls[i].addr_text); | 
|  | } | 
|  | } | 
|  | #endif | 
|  |  | 
|  | #if 0 | 
|  | if (1) { | 
|  | int tcp_nodelay = 1; | 
|  |  | 
|  | if (setsockopt(ls[i].fd, IPPROTO_TCP, TCP_NODELAY, | 
|  | (const void *) &tcp_nodelay, sizeof(int)) | 
|  | == -1) | 
|  | { | 
|  | ngx_log_error(NGX_LOG_ALERT, cycle->log, ngx_socket_errno, | 
|  | "setsockopt(TCP_NODELAY) %V failed, ignored", | 
|  | &ls[i].addr_text); | 
|  | } | 
|  | } | 
|  | #endif | 
|  |  | 
|  | if (ls[i].listen) { | 
|  |  | 
|  | /* change backlog via listen() */ | 
|  |  | 
|  | if (listen(ls[i].fd, ls[i].backlog) == -1) { | 
|  | ngx_log_error(NGX_LOG_ALERT, cycle->log, ngx_socket_errno, | 
|  | "listen() to %V, backlog %d failed, ignored", | 
|  | &ls[i].addr_text, ls[i].backlog); | 
|  | } | 
|  | } | 
|  |  | 
|  | /* | 
|  | * setting deferred mode should be last operation on socket, | 
|  | * because code may prematurely continue cycle on failure | 
|  | */ | 
|  |  | 
|  | #if (NGX_HAVE_DEFERRED_ACCEPT) | 
|  |  | 
|  | #ifdef SO_ACCEPTFILTER | 
|  |  | 
|  | if (ls[i].delete_deferred) { | 
|  | if (setsockopt(ls[i].fd, SOL_SOCKET, SO_ACCEPTFILTER, NULL, 0) | 
|  | == -1) | 
|  | { | 
|  | ngx_log_error(NGX_LOG_ALERT, cycle->log, ngx_socket_errno, | 
|  | "setsockopt(SO_ACCEPTFILTER, NULL) " | 
|  | "for %V failed, ignored", | 
|  | &ls[i].addr_text); | 
|  |  | 
|  | if (ls[i].accept_filter) { | 
|  | ngx_log_error(NGX_LOG_ALERT, cycle->log, 0, | 
|  | "could not change the accept filter " | 
|  | "to \"%s\" for %V, ignored", | 
|  | ls[i].accept_filter, &ls[i].addr_text); | 
|  | } | 
|  |  | 
|  | continue; | 
|  | } | 
|  |  | 
|  | ls[i].deferred_accept = 0; | 
|  | } | 
|  |  | 
|  | if (ls[i].add_deferred) { | 
|  | ngx_memzero(&af, sizeof(struct accept_filter_arg)); | 
|  | (void) ngx_cpystrn((u_char *) af.af_name, | 
|  | (u_char *) ls[i].accept_filter, 16); | 
|  |  | 
|  | if (setsockopt(ls[i].fd, SOL_SOCKET, SO_ACCEPTFILTER, | 
|  | &af, sizeof(struct accept_filter_arg)) | 
|  | == -1) | 
|  | { | 
|  | ngx_log_error(NGX_LOG_ALERT, cycle->log, ngx_socket_errno, | 
|  | "setsockopt(SO_ACCEPTFILTER, \"%s\") " | 
|  | "for %V failed, ignored", | 
|  | ls[i].accept_filter, &ls[i].addr_text); | 
|  | continue; | 
|  | } | 
|  |  | 
|  | ls[i].deferred_accept = 1; | 
|  | } | 
|  |  | 
|  | #endif | 
|  |  | 
|  | #ifdef TCP_DEFER_ACCEPT | 
|  |  | 
|  | if (ls[i].add_deferred || ls[i].delete_deferred) { | 
|  |  | 
|  | if (ls[i].add_deferred) { | 
|  | /* | 
|  | * There is no way to find out how long a connection was | 
|  | * in queue (and a connection may bypass deferred queue at all | 
|  | * if syncookies were used), hence we use 1 second timeout | 
|  | * here. | 
|  | */ | 
|  | value = 1; | 
|  |  | 
|  | } else { | 
|  | value = 0; | 
|  | } | 
|  |  | 
|  | if (setsockopt(ls[i].fd, IPPROTO_TCP, TCP_DEFER_ACCEPT, | 
|  | &value, sizeof(int)) | 
|  | == -1) | 
|  | { | 
|  | ngx_log_error(NGX_LOG_ALERT, cycle->log, ngx_socket_errno, | 
|  | "setsockopt(TCP_DEFER_ACCEPT, %d) for %V failed, " | 
|  | "ignored", | 
|  | value, &ls[i].addr_text); | 
|  |  | 
|  | continue; | 
|  | } | 
|  | } | 
|  |  | 
|  | if (ls[i].add_deferred) { | 
|  | ls[i].deferred_accept = 1; | 
|  | } | 
|  |  | 
|  | #endif | 
|  |  | 
|  | #endif /* NGX_HAVE_DEFERRED_ACCEPT */ | 
|  |  | 
|  | #if (NGX_HAVE_IP_RECVDSTADDR) | 
|  |  | 
|  | if (ls[i].wildcard | 
|  | && ls[i].type == SOCK_DGRAM | 
|  | && ls[i].sockaddr->sa_family == AF_INET) | 
|  | { | 
|  | value = 1; | 
|  |  | 
|  | if (setsockopt(ls[i].fd, IPPROTO_IP, IP_RECVDSTADDR, | 
|  | (const void *) &value, sizeof(int)) | 
|  | == -1) | 
|  | { | 
|  | ngx_log_error(NGX_LOG_ALERT, cycle->log, ngx_socket_errno, | 
|  | "setsockopt(IP_RECVDSTADDR) " | 
|  | "for %V failed, ignored", | 
|  | &ls[i].addr_text); | 
|  | } | 
|  | } | 
|  |  | 
|  | #elif (NGX_HAVE_IP_PKTINFO) | 
|  |  | 
|  | if (ls[i].wildcard | 
|  | && ls[i].type == SOCK_DGRAM | 
|  | && ls[i].sockaddr->sa_family == AF_INET) | 
|  | { | 
|  | value = 1; | 
|  |  | 
|  | if (setsockopt(ls[i].fd, IPPROTO_IP, IP_PKTINFO, | 
|  | (const void *) &value, sizeof(int)) | 
|  | == -1) | 
|  | { | 
|  | ngx_log_error(NGX_LOG_ALERT, cycle->log, ngx_socket_errno, | 
|  | "setsockopt(IP_PKTINFO) " | 
|  | "for %V failed, ignored", | 
|  | &ls[i].addr_text); | 
|  | } | 
|  | } | 
|  |  | 
|  | #endif | 
|  |  | 
|  | #if (NGX_HAVE_INET6 && NGX_HAVE_IPV6_RECVPKTINFO) | 
|  |  | 
|  | if (ls[i].wildcard | 
|  | && ls[i].type == SOCK_DGRAM | 
|  | && ls[i].sockaddr->sa_family == AF_INET6) | 
|  | { | 
|  | value = 1; | 
|  |  | 
|  | if (setsockopt(ls[i].fd, IPPROTO_IPV6, IPV6_RECVPKTINFO, | 
|  | (const void *) &value, sizeof(int)) | 
|  | == -1) | 
|  | { | 
|  | ngx_log_error(NGX_LOG_ALERT, cycle->log, ngx_socket_errno, | 
|  | "setsockopt(IPV6_RECVPKTINFO) " | 
|  | "for %V failed, ignored", | 
|  | &ls[i].addr_text); | 
|  | } | 
|  | } | 
|  |  | 
|  | #endif | 
|  | } | 
|  |  | 
|  | return; | 
|  | } | 
|  |  | 
|  |  | 
|  | void | 
|  | ngx_close_listening_sockets(ngx_cycle_t *cycle) | 
|  | { | 
|  | ngx_uint_t         i; | 
|  | ngx_listening_t   *ls; | 
|  | ngx_connection_t  *c; | 
|  |  | 
|  | if (ngx_event_flags & NGX_USE_IOCP_EVENT) { | 
|  | return; | 
|  | } | 
|  |  | 
|  | ngx_accept_mutex_held = 0; | 
|  | ngx_use_accept_mutex = 0; | 
|  |  | 
|  | ls = cycle->listening.elts; | 
|  | for (i = 0; i < cycle->listening.nelts; i++) { | 
|  |  | 
|  | c = ls[i].connection; | 
|  |  | 
|  | if (c) { | 
|  | if (c->read->active) { | 
|  | if (ngx_event_flags & NGX_USE_EPOLL_EVENT) { | 
|  |  | 
|  | /* | 
|  | * it seems that Linux-2.6.x OpenVZ sends events | 
|  | * for closed shared listening sockets unless | 
|  | * the events was explicitly deleted | 
|  | */ | 
|  |  | 
|  | ngx_del_event(c->read, NGX_READ_EVENT, 0); | 
|  |  | 
|  | } else { | 
|  | ngx_del_event(c->read, NGX_READ_EVENT, NGX_CLOSE_EVENT); | 
|  | } | 
|  | } | 
|  |  | 
|  | ngx_free_connection(c); | 
|  |  | 
|  | c->fd = (ngx_socket_t) -1; | 
|  | } | 
|  |  | 
|  | ngx_log_debug2(NGX_LOG_DEBUG_CORE, cycle->log, 0, | 
|  | "close listening %V #%d ", &ls[i].addr_text, ls[i].fd); | 
|  |  | 
|  | if (ngx_close_socket(ls[i].fd) == -1) { | 
|  | ngx_log_error(NGX_LOG_EMERG, cycle->log, ngx_socket_errno, | 
|  | ngx_close_socket_n " %V failed", &ls[i].addr_text); | 
|  | } | 
|  |  | 
|  | #if (NGX_HAVE_UNIX_DOMAIN) | 
|  |  | 
|  | if (ls[i].sockaddr->sa_family == AF_UNIX | 
|  | && ngx_process <= NGX_PROCESS_MASTER | 
|  | && ngx_new_binary == 0 | 
|  | && (!ls[i].inherited || ngx_getppid() != ngx_parent)) | 
|  | { | 
|  | u_char *name = ls[i].addr_text.data + sizeof("unix:") - 1; | 
|  |  | 
|  | if (ngx_delete_file(name) == NGX_FILE_ERROR) { | 
|  | ngx_log_error(NGX_LOG_EMERG, cycle->log, ngx_socket_errno, | 
|  | ngx_delete_file_n " %s failed", name); | 
|  | } | 
|  | } | 
|  |  | 
|  | #endif | 
|  |  | 
|  | ls[i].fd = (ngx_socket_t) -1; | 
|  | } | 
|  |  | 
|  | cycle->listening.nelts = 0; | 
|  | } | 
|  |  | 
|  |  | 
|  | ngx_connection_t * | 
|  | ngx_get_connection(ngx_socket_t s, ngx_log_t *log) | 
|  | { | 
|  | ngx_uint_t         instance; | 
|  | ngx_event_t       *rev, *wev; | 
|  | ngx_connection_t  *c; | 
|  |  | 
|  | /* disable warning: Win32 SOCKET is u_int while UNIX socket is int */ | 
|  |  | 
|  | if (ngx_cycle->files && (ngx_uint_t) s >= ngx_cycle->files_n) { | 
|  | ngx_log_error(NGX_LOG_ALERT, log, 0, | 
|  | "the new socket has number %d, " | 
|  | "but only %ui files are available", | 
|  | s, ngx_cycle->files_n); | 
|  | return NULL; | 
|  | } | 
|  |  | 
|  | ngx_drain_connections((ngx_cycle_t *) ngx_cycle); | 
|  |  | 
|  | c = ngx_cycle->free_connections; | 
|  |  | 
|  | if (c == NULL) { | 
|  | ngx_log_error(NGX_LOG_ALERT, log, 0, | 
|  | "%ui worker_connections are not enough", | 
|  | ngx_cycle->connection_n); | 
|  |  | 
|  | return NULL; | 
|  | } | 
|  |  | 
|  | ngx_cycle->free_connections = c->data; | 
|  | ngx_cycle->free_connection_n--; | 
|  |  | 
|  | if (ngx_cycle->files && ngx_cycle->files[s] == NULL) { | 
|  | ngx_cycle->files[s] = c; | 
|  | } | 
|  |  | 
|  | rev = c->read; | 
|  | wev = c->write; | 
|  |  | 
|  | ngx_memzero(c, sizeof(ngx_connection_t)); | 
|  |  | 
|  | c->read = rev; | 
|  | c->write = wev; | 
|  | c->fd = s; | 
|  | c->log = log; | 
|  |  | 
|  | instance = rev->instance; | 
|  |  | 
|  | ngx_memzero(rev, sizeof(ngx_event_t)); | 
|  | ngx_memzero(wev, sizeof(ngx_event_t)); | 
|  |  | 
|  | rev->instance = !instance; | 
|  | wev->instance = !instance; | 
|  |  | 
|  | rev->index = NGX_INVALID_INDEX; | 
|  | wev->index = NGX_INVALID_INDEX; | 
|  |  | 
|  | rev->data = c; | 
|  | wev->data = c; | 
|  |  | 
|  | wev->write = 1; | 
|  |  | 
|  | return c; | 
|  | } | 
|  |  | 
|  |  | 
|  | void | 
|  | ngx_free_connection(ngx_connection_t *c) | 
|  | { | 
|  | c->data = ngx_cycle->free_connections; | 
|  | ngx_cycle->free_connections = c; | 
|  | ngx_cycle->free_connection_n++; | 
|  |  | 
|  | if (ngx_cycle->files && ngx_cycle->files[c->fd] == c) { | 
|  | ngx_cycle->files[c->fd] = NULL; | 
|  | } | 
|  | } | 
|  |  | 
|  |  | 
|  | void | 
|  | ngx_close_connection(ngx_connection_t *c) | 
|  | { | 
|  | ngx_err_t     err; | 
|  | ngx_uint_t    log_error, level; | 
|  | ngx_socket_t  fd; | 
|  |  | 
|  | if (c->fd == (ngx_socket_t) -1) { | 
|  | ngx_log_error(NGX_LOG_ALERT, c->log, 0, "connection already closed"); | 
|  | return; | 
|  | } | 
|  |  | 
|  | if (c->read->timer_set) { | 
|  | ngx_del_timer(c->read); | 
|  | } | 
|  |  | 
|  | if (c->write->timer_set) { | 
|  | ngx_del_timer(c->write); | 
|  | } | 
|  |  | 
|  | if (!c->shared) { | 
|  | if (ngx_del_conn) { | 
|  | ngx_del_conn(c, NGX_CLOSE_EVENT); | 
|  |  | 
|  | } else { | 
|  | if (c->read->active || c->read->disabled) { | 
|  | ngx_del_event(c->read, NGX_READ_EVENT, NGX_CLOSE_EVENT); | 
|  | } | 
|  |  | 
|  | if (c->write->active || c->write->disabled) { | 
|  | ngx_del_event(c->write, NGX_WRITE_EVENT, NGX_CLOSE_EVENT); | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | if (c->read->posted) { | 
|  | ngx_delete_posted_event(c->read); | 
|  | } | 
|  |  | 
|  | if (c->write->posted) { | 
|  | ngx_delete_posted_event(c->write); | 
|  | } | 
|  |  | 
|  | c->read->closed = 1; | 
|  | c->write->closed = 1; | 
|  |  | 
|  | ngx_reusable_connection(c, 0); | 
|  |  | 
|  | log_error = c->log_error; | 
|  |  | 
|  | ngx_free_connection(c); | 
|  |  | 
|  | fd = c->fd; | 
|  | c->fd = (ngx_socket_t) -1; | 
|  |  | 
|  | if (c->shared) { | 
|  | return; | 
|  | } | 
|  |  | 
|  | if (ngx_close_socket(fd) == -1) { | 
|  |  | 
|  | err = ngx_socket_errno; | 
|  |  | 
|  | if (err == NGX_ECONNRESET || err == NGX_ENOTCONN) { | 
|  |  | 
|  | switch (log_error) { | 
|  |  | 
|  | case NGX_ERROR_INFO: | 
|  | level = NGX_LOG_INFO; | 
|  | break; | 
|  |  | 
|  | case NGX_ERROR_ERR: | 
|  | level = NGX_LOG_ERR; | 
|  | break; | 
|  |  | 
|  | default: | 
|  | level = NGX_LOG_CRIT; | 
|  | } | 
|  |  | 
|  | } else { | 
|  | level = NGX_LOG_CRIT; | 
|  | } | 
|  |  | 
|  | ngx_log_error(level, c->log, err, ngx_close_socket_n " %d failed", fd); | 
|  | } | 
|  | } | 
|  |  | 
|  |  | 
|  | void | 
|  | ngx_reusable_connection(ngx_connection_t *c, ngx_uint_t reusable) | 
|  | { | 
|  | ngx_log_debug1(NGX_LOG_DEBUG_CORE, c->log, 0, | 
|  | "reusable connection: %ui", reusable); | 
|  |  | 
|  | if (c->reusable) { | 
|  | ngx_queue_remove(&c->queue); | 
|  | ngx_cycle->reusable_connections_n--; | 
|  |  | 
|  | #if (NGX_STAT_STUB) | 
|  | (void) ngx_atomic_fetch_add(ngx_stat_waiting, -1); | 
|  | #endif | 
|  | } | 
|  |  | 
|  | c->reusable = reusable; | 
|  |  | 
|  | if (reusable) { | 
|  | /* need cast as ngx_cycle is volatile */ | 
|  |  | 
|  | ngx_queue_insert_head( | 
|  | (ngx_queue_t *) &ngx_cycle->reusable_connections_queue, &c->queue); | 
|  | ngx_cycle->reusable_connections_n++; | 
|  |  | 
|  | #if (NGX_STAT_STUB) | 
|  | (void) ngx_atomic_fetch_add(ngx_stat_waiting, 1); | 
|  | #endif | 
|  | } | 
|  | } | 
|  |  | 
|  |  | 
|  | static void | 
|  | ngx_drain_connections(ngx_cycle_t *cycle) | 
|  | { | 
|  | ngx_uint_t         i, n; | 
|  | ngx_queue_t       *q; | 
|  | ngx_connection_t  *c; | 
|  |  | 
|  | if (cycle->free_connection_n > cycle->connection_n / 16 | 
|  | || cycle->reusable_connections_n == 0) | 
|  | { | 
|  | return; | 
|  | } | 
|  |  | 
|  | if (cycle->connections_reuse_time != ngx_time()) { | 
|  | cycle->connections_reuse_time = ngx_time(); | 
|  |  | 
|  | ngx_log_error(NGX_LOG_WARN, cycle->log, 0, | 
|  | "%ui worker_connections are not enough, " | 
|  | "reusing connections", | 
|  | cycle->connection_n); | 
|  | } | 
|  |  | 
|  | c = NULL; | 
|  | n = ngx_max(ngx_min(32, cycle->reusable_connections_n / 8), 1); | 
|  |  | 
|  | for (i = 0; i < n; i++) { | 
|  | if (ngx_queue_empty(&cycle->reusable_connections_queue)) { | 
|  | break; | 
|  | } | 
|  |  | 
|  | q = ngx_queue_last(&cycle->reusable_connections_queue); | 
|  | c = ngx_queue_data(q, ngx_connection_t, queue); | 
|  |  | 
|  | ngx_log_debug0(NGX_LOG_DEBUG_CORE, c->log, 0, | 
|  | "reusing connection"); | 
|  |  | 
|  | c->close = 1; | 
|  | c->read->handler(c->read); | 
|  | } | 
|  |  | 
|  | if (cycle->free_connection_n == 0 && c && c->reusable) { | 
|  |  | 
|  | /* | 
|  | * if no connections were freed, try to reuse the last | 
|  | * connection again: this should free it as long as | 
|  | * previous reuse moved it to lingering close | 
|  | */ | 
|  |  | 
|  | ngx_log_debug0(NGX_LOG_DEBUG_CORE, c->log, 0, | 
|  | "reusing connection again"); | 
|  |  | 
|  | c->close = 1; | 
|  | c->read->handler(c->read); | 
|  | } | 
|  | } | 
|  |  | 
|  |  | 
|  | void | 
|  | ngx_close_idle_connections(ngx_cycle_t *cycle) | 
|  | { | 
|  | ngx_uint_t         i; | 
|  | ngx_connection_t  *c; | 
|  |  | 
|  | c = cycle->connections; | 
|  |  | 
|  | for (i = 0; i < cycle->connection_n; i++) { | 
|  |  | 
|  | /* THREAD: lock */ | 
|  |  | 
|  | if (c[i].fd != (ngx_socket_t) -1 && c[i].idle) { | 
|  | c[i].close = 1; | 
|  | c[i].read->handler(c[i].read); | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  |  | 
|  | ngx_int_t | 
|  | ngx_connection_local_sockaddr(ngx_connection_t *c, ngx_str_t *s, | 
|  | ngx_uint_t port) | 
|  | { | 
|  | socklen_t             len; | 
|  | ngx_uint_t            addr; | 
|  | ngx_sockaddr_t        sa; | 
|  | struct sockaddr_in   *sin; | 
|  | #if (NGX_HAVE_INET6) | 
|  | ngx_uint_t            i; | 
|  | struct sockaddr_in6  *sin6; | 
|  | #endif | 
|  |  | 
|  | addr = 0; | 
|  |  | 
|  | if (c->local_socklen) { | 
|  | switch (c->local_sockaddr->sa_family) { | 
|  |  | 
|  | #if (NGX_HAVE_INET6) | 
|  | case AF_INET6: | 
|  | sin6 = (struct sockaddr_in6 *) c->local_sockaddr; | 
|  |  | 
|  | for (i = 0; addr == 0 && i < 16; i++) { | 
|  | addr |= sin6->sin6_addr.s6_addr[i]; | 
|  | } | 
|  |  | 
|  | break; | 
|  | #endif | 
|  |  | 
|  | #if (NGX_HAVE_UNIX_DOMAIN) | 
|  | case AF_UNIX: | 
|  | addr = 1; | 
|  | break; | 
|  | #endif | 
|  |  | 
|  | default: /* AF_INET */ | 
|  | sin = (struct sockaddr_in *) c->local_sockaddr; | 
|  | addr = sin->sin_addr.s_addr; | 
|  | break; | 
|  | } | 
|  | } | 
|  |  | 
|  | if (addr == 0) { | 
|  |  | 
|  | len = sizeof(ngx_sockaddr_t); | 
|  |  | 
|  | if (getsockname(c->fd, &sa.sockaddr, &len) == -1) { | 
|  | ngx_connection_error(c, ngx_socket_errno, "getsockname() failed"); | 
|  | return NGX_ERROR; | 
|  | } | 
|  |  | 
|  | c->local_sockaddr = ngx_palloc(c->pool, len); | 
|  | if (c->local_sockaddr == NULL) { | 
|  | return NGX_ERROR; | 
|  | } | 
|  |  | 
|  | ngx_memcpy(c->local_sockaddr, &sa, len); | 
|  |  | 
|  | c->local_socklen = len; | 
|  | } | 
|  |  | 
|  | if (s == NULL) { | 
|  | return NGX_OK; | 
|  | } | 
|  |  | 
|  | s->len = ngx_sock_ntop(c->local_sockaddr, c->local_socklen, | 
|  | s->data, s->len, port); | 
|  |  | 
|  | return NGX_OK; | 
|  | } | 
|  |  | 
|  |  | 
|  | ngx_int_t | 
|  | ngx_tcp_nodelay(ngx_connection_t *c) | 
|  | { | 
|  | int  tcp_nodelay; | 
|  |  | 
|  | if (c->tcp_nodelay != NGX_TCP_NODELAY_UNSET) { | 
|  | return NGX_OK; | 
|  | } | 
|  |  | 
|  | ngx_log_debug0(NGX_LOG_DEBUG_CORE, c->log, 0, "tcp_nodelay"); | 
|  |  | 
|  | tcp_nodelay = 1; | 
|  |  | 
|  | if (setsockopt(c->fd, IPPROTO_TCP, TCP_NODELAY, | 
|  | (const void *) &tcp_nodelay, sizeof(int)) | 
|  | == -1) | 
|  | { | 
|  | #if (NGX_SOLARIS) | 
|  | if (c->log_error == NGX_ERROR_INFO) { | 
|  |  | 
|  | /* Solaris returns EINVAL if a socket has been shut down */ | 
|  | c->log_error = NGX_ERROR_IGNORE_EINVAL; | 
|  |  | 
|  | ngx_connection_error(c, ngx_socket_errno, | 
|  | "setsockopt(TCP_NODELAY) failed"); | 
|  |  | 
|  | c->log_error = NGX_ERROR_INFO; | 
|  |  | 
|  | return NGX_ERROR; | 
|  | } | 
|  | #endif | 
|  |  | 
|  | ngx_connection_error(c, ngx_socket_errno, | 
|  | "setsockopt(TCP_NODELAY) failed"); | 
|  | return NGX_ERROR; | 
|  | } | 
|  |  | 
|  | c->tcp_nodelay = NGX_TCP_NODELAY_SET; | 
|  |  | 
|  | return NGX_OK; | 
|  | } | 
|  |  | 
|  |  | 
|  | ngx_int_t | 
|  | ngx_connection_error(ngx_connection_t *c, ngx_err_t err, char *text) | 
|  | { | 
|  | ngx_uint_t  level; | 
|  |  | 
|  | /* Winsock may return NGX_ECONNABORTED instead of NGX_ECONNRESET */ | 
|  |  | 
|  | if ((err == NGX_ECONNRESET | 
|  | #if (NGX_WIN32) | 
|  | || err == NGX_ECONNABORTED | 
|  | #endif | 
|  | ) && c->log_error == NGX_ERROR_IGNORE_ECONNRESET) | 
|  | { | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | #if (NGX_SOLARIS) | 
|  | if (err == NGX_EINVAL && c->log_error == NGX_ERROR_IGNORE_EINVAL) { | 
|  | return 0; | 
|  | } | 
|  | #endif | 
|  |  | 
|  | if (err == 0 | 
|  | || err == NGX_ECONNRESET | 
|  | #if (NGX_WIN32) | 
|  | || err == NGX_ECONNABORTED | 
|  | #else | 
|  | || err == NGX_EPIPE | 
|  | #endif | 
|  | || err == NGX_ENOTCONN | 
|  | || err == NGX_ETIMEDOUT | 
|  | || err == NGX_ECONNREFUSED | 
|  | || err == NGX_ENETDOWN | 
|  | || err == NGX_ENETUNREACH | 
|  | || err == NGX_EHOSTDOWN | 
|  | || err == NGX_EHOSTUNREACH) | 
|  | { | 
|  | switch (c->log_error) { | 
|  |  | 
|  | case NGX_ERROR_IGNORE_EINVAL: | 
|  | case NGX_ERROR_IGNORE_ECONNRESET: | 
|  | case NGX_ERROR_INFO: | 
|  | level = NGX_LOG_INFO; | 
|  | break; | 
|  |  | 
|  | default: | 
|  | level = NGX_LOG_ERR; | 
|  | } | 
|  |  | 
|  | } else { | 
|  | level = NGX_LOG_ALERT; | 
|  | } | 
|  |  | 
|  | ngx_log_error(level, c->log, err, text); | 
|  |  | 
|  | return NGX_ERROR; | 
|  | } |