src/os/win32/ngx_wsasend_chain.c - nginx source code

Functions defined

Macros 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. #define NGX_WSABUFS  64


  9. ngx_chain_t *
  10. ngx_wsasend_chain(ngx_connection_t *c, ngx_chain_t *in, off_t limit)
  11. {
  12.     int           rc;
  13.     u_char       *prev;
  14.     u_long        size, sent, send, prev_send;
  15.     ngx_err_t     err;
  16.     ngx_event_t  *wev;
  17.     ngx_array_t   vec;
  18.     ngx_chain_t  *cl;
  19.     LPWSABUF      wsabuf;
  20.     WSABUF        wsabufs[NGX_WSABUFS];

  21.     wev = c->write;

  22.     if (!wev->ready) {
  23.         return in;
  24.     }

  25.     /* the maximum limit size is the maximum u_long value - the page size */

  26.     if (limit == 0 || limit > (off_t) (NGX_MAX_UINT32_VALUE - ngx_pagesize)) {
  27.         limit = NGX_MAX_UINT32_VALUE - ngx_pagesize;
  28.     }

  29.     send = 0;

  30.     /*
  31.      * WSABUFs must be 4-byte aligned otherwise
  32.      * WSASend() will return undocumented WSAEINVAL error.
  33.      */

  34.     vec.elts = wsabufs;
  35.     vec.size = sizeof(WSABUF);
  36.     vec.nalloc = ngx_min(NGX_WSABUFS, ngx_max_wsabufs);
  37.     vec.pool = c->pool;

  38.     for ( ;; ) {
  39.         prev = NULL;
  40.         wsabuf = NULL;
  41.         prev_send = send;

  42.         vec.nelts = 0;

  43.         /* create the WSABUF and coalesce the neighbouring bufs */

  44.         for (cl = in; cl && send < limit; cl = cl->next) {

  45.             if (ngx_buf_special(cl->buf)) {
  46.                 continue;
  47.             }

  48.             size = cl->buf->last - cl->buf->pos;

  49.             if (send + size > limit) {
  50.                 size = (u_long) (limit - send);
  51.             }

  52.             if (prev == cl->buf->pos) {
  53.                 wsabuf->len += cl->buf->last - cl->buf->pos;

  54.             } else {
  55.                 if (vec.nelts == vec.nalloc) {
  56.                     break;
  57.                 }

  58.                 wsabuf = ngx_array_push(&vec);
  59.                 if (wsabuf == NULL) {
  60.                     return NGX_CHAIN_ERROR;
  61.                 }

  62.                 wsabuf->buf = (char *) cl->buf->pos;
  63.                 wsabuf->len = cl->buf->last - cl->buf->pos;
  64.             }

  65.             prev = cl->buf->last;
  66.             send += size;
  67.         }

  68.         sent = 0;

  69.         rc = WSASend(c->fd, vec.elts, vec.nelts, &sent, 0, NULL, NULL);

  70.         if (rc == -1) {
  71.             err = ngx_errno;

  72.             if (err == WSAEWOULDBLOCK) {
  73.                 ngx_log_debug0(NGX_LOG_DEBUG_EVENT, c->log, err,
  74.                                "WSASend() not ready");

  75.             } else {
  76.                 wev->error = 1;
  77.                 ngx_connection_error(c, err, "WSASend() failed");
  78.                 return NGX_CHAIN_ERROR;
  79.             }
  80.         }

  81.         ngx_log_debug2(NGX_LOG_DEBUG_EVENT, c->log, 0,
  82.                        "WSASend: fd:%d, s:%ul", c->fd, sent);

  83.         c->sent += sent;

  84.         in = ngx_chain_update_sent(in, sent);

  85.         if (send - prev_send != sent) {
  86.             wev->ready = 0;
  87.             return in;
  88.         }

  89.         if (send >= limit || in == NULL) {
  90.             return in;
  91.         }
  92.     }
  93. }


  94. ngx_chain_t *
  95. ngx_overlapped_wsasend_chain(ngx_connection_t *c, ngx_chain_t *in, off_t limit)
  96. {
  97.     int               rc;
  98.     u_char           *prev;
  99.     u_long            size, send, sent;
  100.     ngx_err_t         err;
  101.     ngx_event_t      *wev;
  102.     ngx_array_t       vec;
  103.     ngx_chain_t      *cl;
  104.     LPWSAOVERLAPPED   ovlp;
  105.     LPWSABUF          wsabuf;
  106.     WSABUF            wsabufs[NGX_WSABUFS];

  107.     wev = c->write;

  108.     if (!wev->ready) {
  109.         return in;
  110.     }

  111.     ngx_log_debug1(NGX_LOG_DEBUG_EVENT, c->log, 0,
  112.                    "wev->complete: %d", wev->complete);

  113.     if (!wev->complete) {

  114.         /* post the overlapped WSASend() */

  115.         /* the maximum limit size is the maximum u_long value - the page size */

  116.         if (limit == 0 || limit > (off_t) (NGX_MAX_UINT32_VALUE - ngx_pagesize))
  117.         {
  118.             limit = NGX_MAX_UINT32_VALUE - ngx_pagesize;
  119.         }

  120.         /*
  121.          * WSABUFs must be 4-byte aligned otherwise
  122.          * WSASend() will return undocumented WSAEINVAL error.
  123.          */

  124.         vec.elts = wsabufs;
  125.         vec.nelts = 0;
  126.         vec.size = sizeof(WSABUF);
  127.         vec.nalloc = ngx_min(NGX_WSABUFS, ngx_max_wsabufs);
  128.         vec.pool = c->pool;

  129.         send = 0;
  130.         prev = NULL;
  131.         wsabuf = NULL;

  132.         /* create the WSABUF and coalesce the neighbouring bufs */

  133.         for (cl = in; cl && send < limit; cl = cl->next) {

  134.             if (ngx_buf_special(cl->buf)) {
  135.                 continue;
  136.             }

  137.             size = cl->buf->last - cl->buf->pos;

  138.             if (send + size > limit) {
  139.                 size = (u_long) (limit - send);
  140.             }

  141.             if (prev == cl->buf->pos) {
  142.                 wsabuf->len += cl->buf->last - cl->buf->pos;

  143.             } else {
  144.                 if (vec.nelts == vec.nalloc) {
  145.                     break;
  146.                 }

  147.                 wsabuf = ngx_array_push(&vec);
  148.                 if (wsabuf == NULL) {
  149.                     return NGX_CHAIN_ERROR;
  150.                 }

  151.                 wsabuf->buf = (char *) cl->buf->pos;
  152.                 wsabuf->len = cl->buf->last - cl->buf->pos;
  153.             }

  154.             prev = cl->buf->last;
  155.             send += size;
  156.         }

  157.         ovlp = (LPWSAOVERLAPPED) &c->write->ovlp;
  158.         ngx_memzero(ovlp, sizeof(WSAOVERLAPPED));

  159.         rc = WSASend(c->fd, vec.elts, vec.nelts, &sent, 0, ovlp, NULL);

  160.         wev->complete = 0;

  161.         if (rc == -1) {
  162.             err = ngx_errno;

  163.             if (err == WSA_IO_PENDING) {
  164.                 ngx_log_debug0(NGX_LOG_DEBUG_EVENT, c->log, err,
  165.                                "WSASend() posted");
  166.                 wev->active = 1;
  167.                 return in;

  168.             } else {
  169.                 wev->error = 1;
  170.                 ngx_connection_error(c, err, "WSASend() failed");
  171.                 return NGX_CHAIN_ERROR;
  172.             }

  173.         } else if (ngx_event_flags & NGX_USE_IOCP_EVENT) {

  174.             /*
  175.              * if a socket was bound with I/O completion port then
  176.              * GetQueuedCompletionStatus() would anyway return its status
  177.              * despite that WSASend() was already complete
  178.              */

  179.             wev->active = 1;
  180.             return in;
  181.         }

  182.         ngx_log_debug2(NGX_LOG_DEBUG_EVENT, c->log, 0,
  183.                        "WSASend: fd:%d, s:%ul", c->fd, sent);

  184.     } else {

  185.         /* the overlapped WSASend() complete */

  186.         wev->complete = 0;
  187.         wev->active = 0;

  188.         if (ngx_event_flags & NGX_USE_IOCP_EVENT) {
  189.             if (wev->ovlp.error) {
  190.                 ngx_connection_error(c, wev->ovlp.error, "WSASend() failed");
  191.                 return NGX_CHAIN_ERROR;
  192.             }

  193.             sent = wev->available;

  194.         } else {
  195.             if (WSAGetOverlappedResult(c->fd, (LPWSAOVERLAPPED) &wev->ovlp,
  196.                                        &sent, 0, NULL)
  197.                 == 0)
  198.             {
  199.                 ngx_connection_error(c, ngx_socket_errno,
  200.                                "WSASend() or WSAGetOverlappedResult() failed");

  201.                 return NGX_CHAIN_ERROR;
  202.             }
  203.         }
  204.     }

  205.     ngx_log_debug2(NGX_LOG_DEBUG_EVENT, c->log, 0,
  206.                    "WSASend ovlp: fd:%d, s:%ul", c->fd, sent);

  207.     c->sent += sent;

  208.     in = ngx_chain_update_sent(in, sent);

  209.     if (in) {
  210.         wev->ready = 0;

  211.     } else {
  212.         wev->ready = 1;
  213.     }

  214.     return in;
  215. }