src/event/ngx_event_accept.c - nginx source code

Functions defined

Source code


  1. /*
  2. * Copyright (C) Igor Sysoev
  3. * Copyright (C) Nginx, Inc.
  4. */


  5. #include <ngx_config.h>
  6. #include <ngx_core.h>
  7. #include <ngx_event.h>


  8. static ngx_int_t ngx_disable_accept_events(ngx_cycle_t *cycle, ngx_uint_t all);
  9. #if (NGX_HAVE_EPOLLEXCLUSIVE)
  10. static void ngx_reorder_accept_events(ngx_listening_t *ls);
  11. #endif
  12. static void ngx_close_accepted_connection(ngx_connection_t *c);


  13. void
  14. ngx_event_accept(ngx_event_t *ev)
  15. {
  16.     socklen_t          socklen;
  17.     ngx_err_t          err;
  18.     ngx_log_t         *log;
  19.     ngx_uint_t         level;
  20.     ngx_socket_t       s;
  21.     ngx_event_t       *rev, *wev;
  22.     ngx_sockaddr_t     sa;
  23.     ngx_listening_t   *ls;
  24.     ngx_connection_t  *c, *lc;
  25.     ngx_event_conf_t  *ecf;
  26. #if (NGX_HAVE_ACCEPT4)
  27.     static ngx_uint_t  use_accept4 = 1;
  28. #endif

  29.     if (ev->timedout) {
  30.         if (ngx_enable_accept_events((ngx_cycle_t *) ngx_cycle) != NGX_OK) {
  31.             return;
  32.         }

  33.         ev->timedout = 0;
  34.     }

  35.     ecf = ngx_event_get_conf(ngx_cycle->conf_ctx, ngx_event_core_module);

  36.     if (!(ngx_event_flags & NGX_USE_KQUEUE_EVENT)) {
  37.         ev->available = ecf->multi_accept;
  38.     }

  39.     lc = ev->data;
  40.     ls = lc->listening;
  41.     ev->ready = 0;

  42.     ngx_log_debug2(NGX_LOG_DEBUG_EVENT, ev->log, 0,
  43.                    "accept on %V, ready: %d", &ls->addr_text, ev->available);

  44.     do {
  45.         socklen = sizeof(ngx_sockaddr_t);

  46. #if (NGX_HAVE_ACCEPT4)
  47.         if (use_accept4) {
  48.             s = accept4(lc->fd, &sa.sockaddr, &socklen, SOCK_NONBLOCK);
  49.         } else {
  50.             s = accept(lc->fd, &sa.sockaddr, &socklen);
  51.         }
  52. #else
  53.         s = accept(lc->fd, &sa.sockaddr, &socklen);
  54. #endif

  55.         if (s == (ngx_socket_t) -1) {
  56.             err = ngx_socket_errno;

  57.             if (err == NGX_EAGAIN) {
  58.                 ngx_log_debug0(NGX_LOG_DEBUG_EVENT, ev->log, err,
  59.                                "accept() not ready");
  60.                 return;
  61.             }

  62.             level = NGX_LOG_ALERT;

  63.             if (err == NGX_ECONNABORTED) {
  64.                 level = NGX_LOG_ERR;

  65.             } else if (err == NGX_EMFILE || err == NGX_ENFILE) {
  66.                 level = NGX_LOG_CRIT;
  67.             }

  68. #if (NGX_HAVE_ACCEPT4)
  69.             ngx_log_error(level, ev->log, err,
  70.                           use_accept4 ? "accept4() failed" : "accept() failed");

  71.             if (use_accept4 && err == NGX_ENOSYS) {
  72.                 use_accept4 = 0;
  73.                 ngx_inherited_nonblocking = 0;
  74.                 continue;
  75.             }
  76. #else
  77.             ngx_log_error(level, ev->log, err, "accept() failed");
  78. #endif

  79.             if (err == NGX_ECONNABORTED) {
  80.                 if (ngx_event_flags & NGX_USE_KQUEUE_EVENT) {
  81.                     ev->available--;
  82.                 }

  83.                 if (ev->available) {
  84.                     continue;
  85.                 }
  86.             }

  87.             if (err == NGX_EMFILE || err == NGX_ENFILE) {
  88.                 if (ngx_disable_accept_events((ngx_cycle_t *) ngx_cycle, 1)
  89.                     != NGX_OK)
  90.                 {
  91.                     return;
  92.                 }

  93.                 if (ngx_use_accept_mutex) {
  94.                     if (ngx_accept_mutex_held) {
  95.                         ngx_shmtx_unlock(&ngx_accept_mutex);
  96.                         ngx_accept_mutex_held = 0;
  97.                     }

  98.                     ngx_accept_disabled = 1;

  99.                 } else {
  100.                     ngx_add_timer(ev, ecf->accept_mutex_delay);
  101.                 }
  102.             }

  103.             return;
  104.         }

  105. #if (NGX_STAT_STUB)
  106.         (void) ngx_atomic_fetch_add(ngx_stat_accepted, 1);
  107. #endif

  108.         ngx_accept_disabled = ngx_cycle->connection_n / 8
  109.                               - ngx_cycle->free_connection_n;

  110.         c = ngx_get_connection(s, ev->log);

  111.         if (c == NULL) {
  112.             if (ngx_close_socket(s) == -1) {
  113.                 ngx_log_error(NGX_LOG_ALERT, ev->log, ngx_socket_errno,
  114.                               ngx_close_socket_n " failed");
  115.             }

  116.             return;
  117.         }

  118.         c->type = SOCK_STREAM;

  119. #if (NGX_STAT_STUB)
  120.         (void) ngx_atomic_fetch_add(ngx_stat_active, 1);
  121. #endif

  122.         c->pool = ngx_create_pool(ls->pool_size, ev->log);
  123.         if (c->pool == NULL) {
  124.             ngx_close_accepted_connection(c);
  125.             return;
  126.         }

  127.         if (socklen > (socklen_t) sizeof(ngx_sockaddr_t)) {
  128.             socklen = sizeof(ngx_sockaddr_t);
  129.         }

  130.         c->sockaddr = ngx_palloc(c->pool, socklen);
  131.         if (c->sockaddr == NULL) {
  132.             ngx_close_accepted_connection(c);
  133.             return;
  134.         }

  135.         ngx_memcpy(c->sockaddr, &sa, socklen);

  136.         log = ngx_palloc(c->pool, sizeof(ngx_log_t));
  137.         if (log == NULL) {
  138.             ngx_close_accepted_connection(c);
  139.             return;
  140.         }

  141.         /* set a blocking mode for iocp and non-blocking mode for others */

  142.         if (ngx_inherited_nonblocking) {
  143.             if (ngx_event_flags & NGX_USE_IOCP_EVENT) {
  144.                 if (ngx_blocking(s) == -1) {
  145.                     ngx_log_error(NGX_LOG_ALERT, ev->log, ngx_socket_errno,
  146.                                   ngx_blocking_n " failed");
  147.                     ngx_close_accepted_connection(c);
  148.                     return;
  149.                 }
  150.             }

  151.         } else {
  152.             if (!(ngx_event_flags & NGX_USE_IOCP_EVENT)) {
  153.                 if (ngx_nonblocking(s) == -1) {
  154.                     ngx_log_error(NGX_LOG_ALERT, ev->log, ngx_socket_errno,
  155.                                   ngx_nonblocking_n " failed");
  156.                     ngx_close_accepted_connection(c);
  157.                     return;
  158.                 }
  159.             }
  160.         }

  161.         *log = ls->log;

  162.         c->recv = ngx_recv;
  163.         c->send = ngx_send;
  164.         c->recv_chain = ngx_recv_chain;
  165.         c->send_chain = ngx_send_chain;

  166.         c->log = log;
  167.         c->pool->log = log;

  168.         c->socklen = socklen;
  169.         c->listening = ls;
  170.         c->local_sockaddr = ls->sockaddr;
  171.         c->local_socklen = ls->socklen;

  172. #if (NGX_HAVE_UNIX_DOMAIN)
  173.         if (c->sockaddr->sa_family == AF_UNIX) {
  174.             c->tcp_nopush = NGX_TCP_NOPUSH_DISABLED;
  175.             c->tcp_nodelay = NGX_TCP_NODELAY_DISABLED;
  176. #if (NGX_SOLARIS)
  177.             /* Solaris's sendfilev() supports AF_NCA, AF_INET, and AF_INET6 */
  178.             c->sendfile = 0;
  179. #endif
  180.         }
  181. #endif

  182.         rev = c->read;
  183.         wev = c->write;

  184.         wev->ready = 1;

  185.         if (ngx_event_flags & NGX_USE_IOCP_EVENT) {
  186.             rev->ready = 1;
  187.         }

  188.         if (ev->deferred_accept) {
  189.             rev->ready = 1;
  190. #if (NGX_HAVE_KQUEUE || NGX_HAVE_EPOLLRDHUP)
  191.             rev->available = 1;
  192. #endif
  193.         }

  194.         rev->log = log;
  195.         wev->log = log;

  196.         /*
  197.          * TODO: MT: - ngx_atomic_fetch_add()
  198.          *             or protection by critical section or light mutex
  199.          *
  200.          * TODO: MP: - allocated in a shared memory
  201.          *           - ngx_atomic_fetch_add()
  202.          *             or protection by critical section or light mutex
  203.          */

  204.         c->number = ngx_atomic_fetch_add(ngx_connection_counter, 1);

  205.         c->start_time = ngx_current_msec;

  206. #if (NGX_STAT_STUB)
  207.         (void) ngx_atomic_fetch_add(ngx_stat_handled, 1);
  208. #endif

  209.         if (ls->addr_ntop) {
  210.             c->addr_text.data = ngx_pnalloc(c->pool, ls->addr_text_max_len);
  211.             if (c->addr_text.data == NULL) {
  212.                 ngx_close_accepted_connection(c);
  213.                 return;
  214.             }

  215.             c->addr_text.len = ngx_sock_ntop(c->sockaddr, c->socklen,
  216.                                              c->addr_text.data,
  217.                                              ls->addr_text_max_len, 0);
  218.             if (c->addr_text.len == 0) {
  219.                 ngx_close_accepted_connection(c);
  220.                 return;
  221.             }
  222.         }

  223. #if (NGX_DEBUG)
  224.         {
  225.         ngx_str_t  addr;
  226.         u_char     text[NGX_SOCKADDR_STRLEN];

  227.         ngx_debug_accepted_connection(ecf, c);

  228.         if (log->log_level & NGX_LOG_DEBUG_EVENT) {
  229.             addr.data = text;
  230.             addr.len = ngx_sock_ntop(c->sockaddr, c->socklen, text,
  231.                                      NGX_SOCKADDR_STRLEN, 1);

  232.             ngx_log_debug3(NGX_LOG_DEBUG_EVENT, log, 0,
  233.                            "*%uA accept: %V fd:%d", c->number, &addr, s);
  234.         }

  235.         }
  236. #endif

  237.         if (ngx_add_conn && (ngx_event_flags & NGX_USE_EPOLL_EVENT) == 0) {
  238.             if (ngx_add_conn(c) == NGX_ERROR) {
  239.                 ngx_close_accepted_connection(c);
  240.                 return;
  241.             }
  242.         }

  243.         log->data = NULL;
  244.         log->handler = NULL;

  245.         ls->handler(c);

  246.         if (ngx_event_flags & NGX_USE_KQUEUE_EVENT) {
  247.             ev->available--;
  248.         }

  249.     } while (ev->available);

  250. #if (NGX_HAVE_EPOLLEXCLUSIVE)
  251.     ngx_reorder_accept_events(ls);
  252. #endif
  253. }


  254. ngx_int_t
  255. ngx_trylock_accept_mutex(ngx_cycle_t *cycle)
  256. {
  257.     if (ngx_shmtx_trylock(&ngx_accept_mutex)) {

  258.         ngx_log_debug0(NGX_LOG_DEBUG_EVENT, cycle->log, 0,
  259.                        "accept mutex locked");

  260.         if (ngx_accept_mutex_held && ngx_accept_events == 0) {
  261.             return NGX_OK;
  262.         }

  263.         if (ngx_enable_accept_events(cycle) == NGX_ERROR) {
  264.             ngx_shmtx_unlock(&ngx_accept_mutex);
  265.             return NGX_ERROR;
  266.         }

  267.         ngx_accept_events = 0;
  268.         ngx_accept_mutex_held = 1;

  269.         return NGX_OK;
  270.     }

  271.     ngx_log_debug1(NGX_LOG_DEBUG_EVENT, cycle->log, 0,
  272.                    "accept mutex lock failed: %ui", ngx_accept_mutex_held);

  273.     if (ngx_accept_mutex_held) {
  274.         if (ngx_disable_accept_events(cycle, 0) == NGX_ERROR) {
  275.             return NGX_ERROR;
  276.         }

  277.         ngx_accept_mutex_held = 0;
  278.     }

  279.     return NGX_OK;
  280. }


  281. ngx_int_t
  282. ngx_enable_accept_events(ngx_cycle_t *cycle)
  283. {
  284.     ngx_uint_t         i;
  285.     ngx_listening_t   *ls;
  286.     ngx_connection_t  *c;

  287.     ls = cycle->listening.elts;
  288.     for (i = 0; i < cycle->listening.nelts; i++) {

  289.         c = ls[i].connection;

  290.         if (c == NULL || c->read->active) {
  291.             continue;
  292.         }

  293.         if (ngx_add_event(c->read, NGX_READ_EVENT, 0) == NGX_ERROR) {
  294.             return NGX_ERROR;
  295.         }
  296.     }

  297.     return NGX_OK;
  298. }


  299. static ngx_int_t
  300. ngx_disable_accept_events(ngx_cycle_t *cycle, ngx_uint_t all)
  301. {
  302.     ngx_uint_t         i;
  303.     ngx_listening_t   *ls;
  304.     ngx_connection_t  *c;

  305.     ls = cycle->listening.elts;
  306.     for (i = 0; i < cycle->listening.nelts; i++) {

  307.         c = ls[i].connection;

  308.         if (c == NULL || !c->read->active) {
  309.             continue;
  310.         }

  311. #if (NGX_HAVE_REUSEPORT)

  312.         /*
  313.          * do not disable accept on worker's own sockets
  314.          * when disabling accept events due to accept mutex
  315.          */

  316.         if (ls[i].reuseport && !all) {
  317.             continue;
  318.         }

  319. #endif

  320.         if (ngx_del_event(c->read, NGX_READ_EVENT, NGX_DISABLE_EVENT)
  321.             == NGX_ERROR)
  322.         {
  323.             return NGX_ERROR;
  324.         }
  325.     }

  326.     return NGX_OK;
  327. }


  328. #if (NGX_HAVE_EPOLLEXCLUSIVE)

  329. static void
  330. ngx_reorder_accept_events(ngx_listening_t *ls)
  331. {
  332.     ngx_connection_t  *c;

  333.     /*
  334.      * Linux with EPOLLEXCLUSIVE usually notifies only the process which
  335.      * was first to add the listening socket to the epoll instance.  As
  336.      * a result most of the connections are handled by the first worker
  337.      * process.  To fix this, we re-add the socket periodically, so other
  338.      * workers will get a chance to accept connections.
  339.      */

  340.     if (!ngx_use_exclusive_accept) {
  341.         return;
  342.     }

  343. #if (NGX_HAVE_REUSEPORT)

  344.     if (ls->reuseport) {
  345.         return;
  346.     }

  347. #endif

  348.     c = ls->connection;

  349.     if (c->requests++ % 16 != 0
  350.         && ngx_accept_disabled <= 0)
  351.     {
  352.         return;
  353.     }

  354.     if (ngx_del_event(c->read, NGX_READ_EVENT, NGX_DISABLE_EVENT)
  355.         == NGX_ERROR)
  356.     {
  357.         return;
  358.     }

  359.     if (ngx_add_event(c->read, NGX_READ_EVENT, NGX_EXCLUSIVE_EVENT)
  360.         == NGX_ERROR)
  361.     {
  362.         return;
  363.     }
  364. }

  365. #endif


  366. static void
  367. ngx_close_accepted_connection(ngx_connection_t *c)
  368. {
  369.     ngx_socket_t  fd;

  370.     ngx_free_connection(c);

  371.     fd = c->fd;
  372.     c->fd = (ngx_socket_t) -1;

  373.     if (ngx_close_socket(fd) == -1) {
  374.         ngx_log_error(NGX_LOG_ALERT, c->log, ngx_socket_errno,
  375.                       ngx_close_socket_n " failed");
  376.     }

  377.     if (c->pool) {
  378.         ngx_destroy_pool(c->pool);
  379.     }

  380. #if (NGX_STAT_STUB)
  381.     (void) ngx_atomic_fetch_add(ngx_stat_active, -1);
  382. #endif
  383. }


  384. u_char *
  385. ngx_accept_log_error(ngx_log_t *log, u_char *buf, size_t len)
  386. {
  387.     return ngx_snprintf(buf, len, " while accepting new connection on %V",
  388.                         log->data);
  389. }


  390. #if (NGX_DEBUG)

  391. void
  392. ngx_debug_accepted_connection(ngx_event_conf_t *ecf, ngx_connection_t *c)
  393. {
  394.     struct sockaddr_in   *sin;
  395.     ngx_cidr_t           *cidr;
  396.     ngx_uint_t            i;
  397. #if (NGX_HAVE_INET6)
  398.     struct sockaddr_in6  *sin6;
  399.     ngx_uint_t            n;
  400. #endif

  401.     cidr = ecf->debug_connection.elts;
  402.     for (i = 0; i < ecf->debug_connection.nelts; i++) {
  403.         if (cidr[i].family != (ngx_uint_t) c->sockaddr->sa_family) {
  404.             goto next;
  405.         }

  406.         switch (cidr[i].family) {

  407. #if (NGX_HAVE_INET6)
  408.         case AF_INET6:
  409.             sin6 = (struct sockaddr_in6 *) c->sockaddr;
  410.             for (n = 0; n < 16; n++) {
  411.                 if ((sin6->sin6_addr.s6_addr[n]
  412.                     & cidr[i].u.in6.mask.s6_addr[n])
  413.                     != cidr[i].u.in6.addr.s6_addr[n])
  414.                 {
  415.                     goto next;
  416.                 }
  417.             }
  418.             break;
  419. #endif

  420. #if (NGX_HAVE_UNIX_DOMAIN)
  421.         case AF_UNIX:
  422.             break;
  423. #endif

  424.         default: /* AF_INET */
  425.             sin = (struct sockaddr_in *) c->sockaddr;
  426.             if ((sin->sin_addr.s_addr & cidr[i].u.in.mask)
  427.                 != cidr[i].u.in.addr)
  428.             {
  429.                 goto next;
  430.             }
  431.             break;
  432.         }

  433.         c->log->log_level = NGX_LOG_DEBUG_CONNECTION|NGX_LOG_DEBUG_ALL;
  434.         break;

  435.     next:
  436.         continue;
  437.     }
  438. }

  439. #endif