Update and clean Tomato RAF files
[tomato.git] / release / src / router / nginx / src / os / unix / ngx_writev_chain.c
blob805982d6558404f5629219da0fbf07c11ce3b04e
2 /*
3 * Copyright (C) Igor Sysoev
4 * Copyright (C) Nginx, Inc.
5 */
8 #include <ngx_config.h>
9 #include <ngx_core.h>
10 #include <ngx_event.h>
13 #if (IOV_MAX > 64)
14 #define NGX_IOVS 64
15 #else
16 #define NGX_IOVS IOV_MAX
17 #endif
20 ngx_chain_t *
21 ngx_writev_chain(ngx_connection_t *c, ngx_chain_t *in, off_t limit)
23 u_char *prev;
24 ssize_t n, size, sent;
25 off_t send, prev_send;
26 ngx_uint_t eintr, complete;
27 ngx_err_t err;
28 ngx_array_t vec;
29 ngx_chain_t *cl;
30 ngx_event_t *wev;
31 struct iovec *iov, iovs[NGX_IOVS];
33 wev = c->write;
35 if (!wev->ready) {
36 return in;
39 #if (NGX_HAVE_KQUEUE)
41 if ((ngx_event_flags & NGX_USE_KQUEUE_EVENT) && wev->pending_eof) {
42 (void) ngx_connection_error(c, wev->kq_errno,
43 "kevent() reported about an closed connection");
44 wev->error = 1;
45 return NGX_CHAIN_ERROR;
48 #endif
50 /* the maximum limit size is the maximum size_t value - the page size */
52 if (limit == 0 || limit > (off_t) (NGX_MAX_SIZE_T_VALUE - ngx_pagesize)) {
53 limit = NGX_MAX_SIZE_T_VALUE - ngx_pagesize;
56 send = 0;
58 vec.elts = iovs;
59 vec.size = sizeof(struct iovec);
60 vec.nalloc = NGX_IOVS;
61 vec.pool = c->pool;
63 for ( ;; ) {
64 prev = NULL;
65 iov = NULL;
66 eintr = 0;
67 complete = 0;
68 prev_send = send;
70 vec.nelts = 0;
72 /* create the iovec and coalesce the neighbouring bufs */
74 for (cl = in; cl && send < limit; cl = cl->next) {
76 if (ngx_buf_special(cl->buf)) {
77 continue;
80 #if 1
81 if (!ngx_buf_in_memory(cl->buf)) {
82 ngx_debug_point();
84 #endif
86 size = cl->buf->last - cl->buf->pos;
88 if (send + size > limit) {
89 size = (ssize_t) (limit - send);
92 if (prev == cl->buf->pos) {
93 iov->iov_len += size;
95 } else {
96 if (vec.nelts >= IOV_MAX) {
97 break;
100 iov = ngx_array_push(&vec);
101 if (iov == NULL) {
102 return NGX_CHAIN_ERROR;
105 iov->iov_base = (void *) cl->buf->pos;
106 iov->iov_len = size;
109 prev = cl->buf->pos + size;
110 send += size;
113 n = writev(c->fd, vec.elts, vec.nelts);
115 if (n == -1) {
116 err = ngx_errno;
118 switch (err) {
119 case NGX_EAGAIN:
120 break;
122 case NGX_EINTR:
123 eintr = 1;
124 break;
126 default:
127 wev->error = 1;
128 (void) ngx_connection_error(c, err, "writev() failed");
129 return NGX_CHAIN_ERROR;
132 ngx_log_debug0(NGX_LOG_DEBUG_EVENT, c->log, err,
133 "writev() not ready");
136 sent = n > 0 ? n : 0;
138 ngx_log_debug1(NGX_LOG_DEBUG_EVENT, c->log, 0, "writev: %z", sent);
140 if (send - prev_send == sent) {
141 complete = 1;
144 c->sent += sent;
146 for (cl = in; cl; cl = cl->next) {
148 if (ngx_buf_special(cl->buf)) {
149 continue;
152 if (sent == 0) {
153 break;
156 size = cl->buf->last - cl->buf->pos;
158 if (sent >= size) {
159 sent -= size;
160 cl->buf->pos = cl->buf->last;
162 continue;
165 cl->buf->pos += sent;
167 break;
170 if (eintr) {
171 continue;
174 if (!complete) {
175 wev->ready = 0;
176 return cl;
179 if (send >= limit || cl == NULL) {
180 return cl;
183 in = cl;