|  |  | 
|  | /* | 
|  | * Copyright (C) Igor Sysoev | 
|  | */ | 
|  |  | 
|  |  | 
|  | #include <ngx_config.h> | 
|  | #include <ngx_core.h> | 
|  | #include <ngx_event.h> | 
|  | #include <ngx_aio.h> | 
|  |  | 
|  |  | 
|  | ngx_chain_t *ngx_aio_write_chain(ngx_connection_t *c, ngx_chain_t *in, | 
|  | off_t limit) | 
|  | { | 
|  | u_char       *buf, *prev; | 
|  | off_t         send, sent; | 
|  | size_t        len; | 
|  | ssize_t       n, size; | 
|  | ngx_chain_t  *cl; | 
|  |  | 
|  | /* the maximum limit size is the maximum size_t value - the page size */ | 
|  |  | 
|  | if (limit == 0 || limit > NGX_MAX_SIZE_T_VALUE - ngx_pagesize) { | 
|  | limit = NGX_MAX_SIZE_T_VALUE - ngx_pagesize; | 
|  | } | 
|  |  | 
|  | send = 0; | 
|  | sent = 0; | 
|  | cl = in; | 
|  |  | 
|  | while (cl) { | 
|  |  | 
|  | if (cl->buf->pos == cl->buf->last) { | 
|  | cl = cl->next; | 
|  | continue; | 
|  | } | 
|  |  | 
|  | /* we can post the single aio operation only */ | 
|  |  | 
|  | if (!c->write->ready) { | 
|  | return cl; | 
|  | } | 
|  |  | 
|  | buf = cl->buf->pos; | 
|  | prev = buf; | 
|  | len = 0; | 
|  |  | 
|  | /* coalesce the neighbouring bufs */ | 
|  |  | 
|  | while (cl && prev == cl->buf->pos && send < limit) { | 
|  | if (ngx_buf_special(cl->buf)) { | 
|  | continue; | 
|  | } | 
|  |  | 
|  | size = cl->buf->last - cl->buf->pos; | 
|  |  | 
|  | if (send + size > limit) { | 
|  | size = limit - send; | 
|  | } | 
|  |  | 
|  | len += size; | 
|  | prev = cl->buf->pos + size; | 
|  | send += size; | 
|  | cl = cl->next; | 
|  | } | 
|  |  | 
|  | n = ngx_aio_write(c, buf, len); | 
|  |  | 
|  | ngx_log_debug1(NGX_LOG_DEBUG_EVENT, c->log, 0, "aio_write: %z", n); | 
|  |  | 
|  | if (n == NGX_ERROR) { | 
|  | return NGX_CHAIN_ERROR; | 
|  | } | 
|  |  | 
|  | if (n > 0) { | 
|  | sent += n; | 
|  | c->sent += n; | 
|  | } | 
|  |  | 
|  | ngx_log_debug1(NGX_LOG_DEBUG_EVENT, c->log, 0, | 
|  | "aio_write sent: %O", c->sent); | 
|  |  | 
|  | for (cl = in; cl; cl = cl->next) { | 
|  |  | 
|  | if (sent >= cl->buf->last - cl->buf->pos) { | 
|  | sent -= cl->buf->last - cl->buf->pos; | 
|  | cl->buf->pos = cl->buf->last; | 
|  |  | 
|  | continue; | 
|  | } | 
|  |  | 
|  | cl->buf->pos += sent; | 
|  |  | 
|  | break; | 
|  | } | 
|  | } | 
|  |  | 
|  | return cl; | 
|  | } |