blob: 5f6970f80808fccba40418c9948349c81e20f8d9 [file] [log] [blame]
Igor Sysoev10fc9ef2003-10-27 08:53:49 +00001
Igor Sysoevd90282d2004-09-28 08:34:51 +00002/*
Igor Sysoevff8da912004-09-29 16:00:49 +00003 * Copyright (C) Igor Sysoev
Igor Sysoevd90282d2004-09-28 08:34:51 +00004 */
5
6
Igor Sysoev10fc9ef2003-10-27 08:53:49 +00007#include <ngx_config.h>
8#include <ngx_core.h>
Igor Sysoev2b0c76c2003-10-27 21:01:00 +00009#include <ngx_event.h>
Igor Sysoev10fc9ef2003-10-27 08:53:49 +000010
11
Igor Sysoevf6906042004-11-25 16:17:31 +000012#if 0
13#define NGX_SENDFILE_LIMIT 4096
14#endif
15
16
17#define NGX_NONE 1
Igor Sysoev10fc9ef2003-10-27 08:53:49 +000018
19
Igor Sysoevd43bee82004-11-20 19:52:20 +000020static ngx_inline ngx_int_t
Igor Sysoev2f657222004-06-16 15:32:11 +000021 ngx_output_chain_need_to_copy(ngx_output_chain_ctx_t *ctx, ngx_buf_t *buf);
Igor Sysoevf6906042004-11-25 16:17:31 +000022static ngx_int_t ngx_output_chain_add_copy(ngx_pool_t *pool,
Igor Sysoev8184d1b2005-03-04 14:06:57 +000023 ngx_chain_t **chain, ngx_chain_t *in);
Igor Sysoev369145c2004-05-28 15:49:23 +000024static ngx_int_t ngx_output_chain_copy_buf(ngx_buf_t *dst, ngx_buf_t *src,
Igor Sysoev8184d1b2005-03-04 14:06:57 +000025 ngx_uint_t sendfile);
Igor Sysoev10fc9ef2003-10-27 08:53:49 +000026
27
Igor Sysoev8184d1b2005-03-04 14:06:57 +000028ngx_int_t
29ngx_output_chain(ngx_output_chain_ctx_t *ctx, ngx_chain_t *in)
Igor Sysoev10fc9ef2003-10-27 08:53:49 +000030{
31 int rc, last;
Igor Sysoev42b12b32004-12-02 18:40:46 +000032 off_t bsize;
33 size_t size;
Igor Sysoev10fc9ef2003-10-27 08:53:49 +000034 ngx_chain_t *cl, *out, **last_out;
35
Igor Sysoevc0edbcc2004-10-21 15:34:38 +000036 if (ctx->in == NULL && ctx->busy == NULL) {
Igor Sysoev10fc9ef2003-10-27 08:53:49 +000037
Igor Sysoevf6906042004-11-25 16:17:31 +000038 /*
39 * the short path for the case when the ctx->in and ctx->busy chains
40 * are empty, the incoming chain is empty too or has the single buf
41 * that does not require the copy
42 */
Igor Sysoev10fc9ef2003-10-27 08:53:49 +000043
44 if (in == NULL) {
Igor Sysoev89690bf2004-03-23 06:01:52 +000045 return ctx->output_filter(ctx->filter_ctx, in);
Igor Sysoev10fc9ef2003-10-27 08:53:49 +000046 }
47
Igor Sysoev2b0c76c2003-10-27 21:01:00 +000048 if (in->next == NULL
Igor Sysoevf6906042004-11-25 16:17:31 +000049#if (NGX_SENDFILE_LIMIT)
50 && !(in->buf->in_file && in->buf->file_last > NGX_SENDFILE_LIMIT)
51#endif
Igor Sysoev369145c2004-05-28 15:49:23 +000052 && (!ngx_output_chain_need_to_copy(ctx, in->buf)))
Igor Sysoev10fc9ef2003-10-27 08:53:49 +000053 {
Igor Sysoev89690bf2004-03-23 06:01:52 +000054 return ctx->output_filter(ctx->filter_ctx, in);
Igor Sysoev10fc9ef2003-10-27 08:53:49 +000055 }
56 }
57
Igor Sysoev67f450d2004-06-01 06:04:46 +000058 /* add the incoming buf to the chain ctx->in */
Igor Sysoev10fc9ef2003-10-27 08:53:49 +000059
60 if (in) {
Igor Sysoevf6906042004-11-25 16:17:31 +000061 if (ngx_output_chain_add_copy(ctx->pool, &ctx->in, in) == NGX_ERROR) {
Igor Sysoev10fc9ef2003-10-27 08:53:49 +000062 return NGX_ERROR;
63 }
64 }
65
Igor Sysoevb5faed22003-10-29 08:30:44 +000066 out = NULL;
Igor Sysoev10fc9ef2003-10-27 08:53:49 +000067 last_out = &out;
Igor Sysoev8184d1b2005-03-04 14:06:57 +000068 last = NGX_NONE;
Igor Sysoev10fc9ef2003-10-27 08:53:49 +000069
70 for ( ;; ) {
71
72 while (ctx->in) {
73
Igor Sysoev9bfb4342004-04-18 19:06:02 +000074 /*
Igor Sysoev369145c2004-05-28 15:49:23 +000075 * cycle while there are the ctx->in bufs
76 * or there are the free output bufs to copy in
Igor Sysoev9bfb4342004-04-18 19:06:02 +000077 */
78
Igor Sysoev00d433f2004-07-28 19:21:26 +000079 bsize = ngx_buf_size(ctx->in->buf);
80
81 if (bsize == 0 && !ngx_buf_special(ctx->in->buf)) {
82
83 ngx_log_error(NGX_LOG_ALERT, ctx->pool->log, 0,
Igor Sysoeve5a222c2005-01-25 12:27:35 +000084 "zero size buf in output "
85 "t:%d r:%d f:%d %p %p-%p %p %O-%O",
86 ctx->in->buf->temporary,
87 ctx->in->buf->recycled,
88 ctx->in->buf->in_file,
89 ctx->in->buf->start,
90 ctx->in->buf->pos,
91 ctx->in->buf->last,
92 ctx->in->buf->file,
93 ctx->in->buf->file_pos,
94 ctx->in->buf->file_last);
Igor Sysoev00d433f2004-07-28 19:21:26 +000095
Igor Sysoev42b12b32004-12-02 18:40:46 +000096 ngx_debug_point();
97
Igor Sysoev00d433f2004-07-28 19:21:26 +000098 ctx->in = ctx->in->next;
Igor Sysoeva2c81192004-09-19 18:27:00 +000099
Igor Sysoev00d433f2004-07-28 19:21:26 +0000100 continue;
101 }
102
Igor Sysoev369145c2004-05-28 15:49:23 +0000103 if (!ngx_output_chain_need_to_copy(ctx, ctx->in->buf)) {
Igor Sysoev10fc9ef2003-10-27 08:53:49 +0000104
Igor Sysoev9bfb4342004-04-18 19:06:02 +0000105 /* move the chain link to the output chain */
Igor Sysoev10fc9ef2003-10-27 08:53:49 +0000106
107 cl = ctx->in;
108 ctx->in = cl->next;
109
110 *last_out = cl;
111 last_out = &cl->next;
112 cl->next = NULL;
113
114 continue;
115 }
116
Igor Sysoev369145c2004-05-28 15:49:23 +0000117 if (ctx->buf == NULL) {
Igor Sysoev10fc9ef2003-10-27 08:53:49 +0000118
Igor Sysoev369145c2004-05-28 15:49:23 +0000119 /* get the free buf */
Igor Sysoev10fc9ef2003-10-27 08:53:49 +0000120
121 if (ctx->free) {
Igor Sysoev369145c2004-05-28 15:49:23 +0000122 ctx->buf = ctx->free->buf;
Igor Sysoev10fc9ef2003-10-27 08:53:49 +0000123 ctx->free = ctx->free->next;
124
Igor Sysoev369145c2004-05-28 15:49:23 +0000125 } else if (out || ctx->allocated == ctx->bufs.num) {
Igor Sysoev31f7f6a2004-05-12 05:37:55 +0000126
127 break;
128
129 } else {
Igor Sysoev10fc9ef2003-10-27 08:53:49 +0000130
131 size = ctx->bufs.size;
132
Igor Sysoev369145c2004-05-28 15:49:23 +0000133 if (ctx->in->buf->last_buf) {
Igor Sysoev10fc9ef2003-10-27 08:53:49 +0000134
Igor Sysoev42b12b32004-12-02 18:40:46 +0000135 if (bsize < (off_t) ctx->bufs.size) {
Igor Sysoev10fc9ef2003-10-27 08:53:49 +0000136
137 /*
Igor Sysoev369145c2004-05-28 15:49:23 +0000138 * allocate small temp buf for the small last buf
Igor Sysoev10fc9ef2003-10-27 08:53:49 +0000139 * or its small last part
140 */
141
Igor Sysoev42b12b32004-12-02 18:40:46 +0000142 size = (size_t) bsize;
Igor Sysoev10fc9ef2003-10-27 08:53:49 +0000143
144 } else if (ctx->bufs.num == 1
Igor Sysoev42b12b32004-12-02 18:40:46 +0000145 && (bsize < (off_t) (ctx->bufs.size
146 + (ctx->bufs.size >> 2))))
Igor Sysoev10fc9ef2003-10-27 08:53:49 +0000147 {
148 /*
Igor Sysoev369145c2004-05-28 15:49:23 +0000149 * allocate a temp buf that equals
150 * to the last buf if the last buf size is lesser
151 * than 1.25 of bufs.size and a temp buf is single
Igor Sysoev10fc9ef2003-10-27 08:53:49 +0000152 */
153
Igor Sysoev42b12b32004-12-02 18:40:46 +0000154 size = (size_t) bsize;
Igor Sysoev10fc9ef2003-10-27 08:53:49 +0000155 }
156 }
157
Igor Sysoev369145c2004-05-28 15:49:23 +0000158 if (!(ctx->buf = ngx_create_temp_buf(ctx->pool, size))) {
159 return NGX_ERROR;
160 }
161
162 ctx->buf->tag = ctx->tag;
163 ctx->buf->recycled = 1;
164 ctx->allocated++;
Igor Sysoev10fc9ef2003-10-27 08:53:49 +0000165 }
166 }
167
Igor Sysoev369145c2004-05-28 15:49:23 +0000168 rc = ngx_output_chain_copy_buf(ctx->buf, ctx->in->buf,
169 ctx->sendfile);
Igor Sysoev10fc9ef2003-10-27 08:53:49 +0000170
171 if (rc == NGX_ERROR) {
172 return rc;
173 }
174
175 if (rc == NGX_AGAIN) {
176 if (out) {
177 break;
178 }
Igor Sysoeva2c81192004-09-19 18:27:00 +0000179
Igor Sysoev10fc9ef2003-10-27 08:53:49 +0000180 return rc;
181 }
182
Igor Sysoev369145c2004-05-28 15:49:23 +0000183 /* delete the completed buf from the ctx->in chain */
Igor Sysoev10fc9ef2003-10-27 08:53:49 +0000184
Igor Sysoev369145c2004-05-28 15:49:23 +0000185 if (ngx_buf_size(ctx->in->buf) == 0) {
Igor Sysoev10fc9ef2003-10-27 08:53:49 +0000186 ctx->in = ctx->in->next;
187 }
188
Igor Sysoev1b735832004-11-11 14:07:14 +0000189 if (!(cl = ngx_alloc_chain_link(ctx->pool))) {
190 return NGX_ERROR;
191 }
192 cl->buf = ctx->buf;
193 cl->next = NULL;
Igor Sysoev10fc9ef2003-10-27 08:53:49 +0000194 *last_out = cl;
195 last_out = &cl->next;
Igor Sysoev369145c2004-05-28 15:49:23 +0000196 ctx->buf = NULL;
Igor Sysoev10fc9ef2003-10-27 08:53:49 +0000197 }
198
199 if (out == NULL && last != NGX_NONE) {
200 return last;
201 }
202
Igor Sysoev89690bf2004-03-23 06:01:52 +0000203 last = ctx->output_filter(ctx->filter_ctx, out);
Igor Sysoev10fc9ef2003-10-27 08:53:49 +0000204
205 ngx_chain_update_chains(&ctx->free, &ctx->busy, &out, ctx->tag);
206 last_out = &out;
Igor Sysoev9bfb4342004-04-18 19:06:02 +0000207
208 if (last == NGX_ERROR) {
209 return last;
210 }
Igor Sysoev10fc9ef2003-10-27 08:53:49 +0000211 }
212}
213
214
Igor Sysoevd43bee82004-11-20 19:52:20 +0000215static ngx_inline ngx_int_t
Igor Sysoev8184d1b2005-03-04 14:06:57 +0000216ngx_output_chain_need_to_copy(ngx_output_chain_ctx_t *ctx, ngx_buf_t *buf)
Igor Sysoev10fc9ef2003-10-27 08:53:49 +0000217{
Igor Sysoevf6906042004-11-25 16:17:31 +0000218 ngx_uint_t sendfile;
219
Igor Sysoev369145c2004-05-28 15:49:23 +0000220 if (ngx_buf_special(buf)) {
Igor Sysoev10fc9ef2003-10-27 08:53:49 +0000221 return 0;
222 }
223
Igor Sysoevf6906042004-11-25 16:17:31 +0000224 sendfile = ctx->sendfile;
225
226#if (NGX_SENDFILE_LIMIT)
227
228 if (buf->in_file && buf->file_pos >= NGX_SENDFILE_LIMIT) {
229 sendfile = 0;
230 }
231
232#endif
233
234 if (!sendfile) {
Igor Sysoevc0edbcc2004-10-21 15:34:38 +0000235
Igor Sysoev369145c2004-05-28 15:49:23 +0000236 if (!ngx_buf_in_memory(buf)) {
Igor Sysoev65977492003-11-02 22:56:18 +0000237 return 1;
238 }
239
Igor Sysoev369145c2004-05-28 15:49:23 +0000240 buf->in_file = 0;
Igor Sysoev10fc9ef2003-10-27 08:53:49 +0000241 }
242
Igor Sysoev369145c2004-05-28 15:49:23 +0000243 if (ctx->need_in_memory && !ngx_buf_in_memory(buf)) {
Igor Sysoev10fc9ef2003-10-27 08:53:49 +0000244 return 1;
245 }
246
Igor Sysoev369145c2004-05-28 15:49:23 +0000247 if (ctx->need_in_temp && (buf->memory || buf->mmap)) {
Igor Sysoev10fc9ef2003-10-27 08:53:49 +0000248 return 1;
249 }
250
251 return 0;
252}
253
254
Igor Sysoev8184d1b2005-03-04 14:06:57 +0000255static ngx_int_t
256ngx_output_chain_add_copy(ngx_pool_t *pool, ngx_chain_t **chain,
257 ngx_chain_t *in)
Igor Sysoevf6906042004-11-25 16:17:31 +0000258{
259 ngx_chain_t *cl, **ll;
260#if (NGX_SENDFILE_LIMIT)
261 ngx_buf_t *b, *buf;
262#endif
263
264 ll = chain;
265
266 for (cl = *chain; cl; cl = cl->next) {
267 ll = &cl->next;
268 }
269
270 while (in) {
271
272 if (!(cl = ngx_alloc_chain_link(pool))) {
273 return NGX_ERROR;
274 }
275
276#if (NGX_SENDFILE_LIMIT)
277
278 buf = in->buf;
279
280 if (buf->in_file
281 && buf->file_pos < NGX_SENDFILE_LIMIT
282 && buf->file_last > NGX_SENDFILE_LIMIT)
283 {
284 if (!(b = ngx_calloc_buf(pool))) {
285 return NGX_ERROR;
286 }
287
288 ngx_memcpy(b, buf, sizeof(ngx_buf_t));
289
290 if (ngx_buf_in_memory(buf)) {
291 buf->pos += (ssize_t) (NGX_SENDFILE_LIMIT - buf->file_pos);
292 b->last = buf->pos;
293 }
294
295 buf->file_pos = NGX_SENDFILE_LIMIT;
296 b->file_last = NGX_SENDFILE_LIMIT;
297
298 cl->buf = b;
299
300 } else {
301 cl->buf = buf;
302 in = in->next;
303 }
304
305#else
306 cl->buf = in->buf;
307 in = in->next;
308
309#endif
310
311 *ll = cl;
312 ll = &cl->next;
313 }
314
315 *ll = NULL;
316
317 return NGX_OK;
318}
319
320
Igor Sysoev8184d1b2005-03-04 14:06:57 +0000321static ngx_int_t
322ngx_output_chain_copy_buf(ngx_buf_t *dst, ngx_buf_t *src, ngx_uint_t sendfile)
Igor Sysoev10fc9ef2003-10-27 08:53:49 +0000323{
Igor Sysoev42b12b32004-12-02 18:40:46 +0000324 off_t size;
Igor Sysoevd9d0ca12003-11-21 06:30:49 +0000325 ssize_t n;
Igor Sysoev10fc9ef2003-10-27 08:53:49 +0000326
Igor Sysoev369145c2004-05-28 15:49:23 +0000327 size = ngx_buf_size(src);
Igor Sysoev10fc9ef2003-10-27 08:53:49 +0000328
Igor Sysoev42b12b32004-12-02 18:40:46 +0000329 if (size > dst->end - dst->pos) {
Igor Sysoev10fc9ef2003-10-27 08:53:49 +0000330 size = dst->end - dst->pos;
331 }
332
Igor Sysoevf6906042004-11-25 16:17:31 +0000333#if (NGX_SENDFILE_LIMIT)
334
335 if (src->in_file && src->file_pos >= NGX_SENDFILE_LIMIT) {
336 sendfile = 0;
337 }
338
339#endif
340
Igor Sysoev369145c2004-05-28 15:49:23 +0000341 if (ngx_buf_in_memory(src)) {
Igor Sysoev42b12b32004-12-02 18:40:46 +0000342 ngx_memcpy(dst->pos, src->pos, (size_t) size);
343 src->pos += (size_t) size;
344 dst->last += (size_t) size;
Igor Sysoev10fc9ef2003-10-27 08:53:49 +0000345
Igor Sysoevc0edbcc2004-10-21 15:34:38 +0000346 if (src->in_file) {
347
348 if (sendfile) {
349 dst->in_file = 1;
350 dst->file = src->file;
351 dst->file_pos = src->file_pos;
352 dst->file_last = src->file_pos + size;
353
354 } else {
355 dst->in_file = 0;
356 }
357
Igor Sysoev10fc9ef2003-10-27 08:53:49 +0000358 src->file_pos += size;
Igor Sysoev924bd792004-10-11 15:07:03 +0000359
360 } else {
361 dst->in_file = 0;
Igor Sysoev10fc9ef2003-10-27 08:53:49 +0000362 }
363
Igor Sysoev369145c2004-05-28 15:49:23 +0000364 if (src->last_buf && src->pos == src->last) {
365 dst->last_buf = 1;
Igor Sysoev10fc9ef2003-10-27 08:53:49 +0000366 }
367
368 } else {
Igor Sysoev42b12b32004-12-02 18:40:46 +0000369 n = ngx_read_file(src->file, dst->pos, (size_t) size, src->file_pos);
Igor Sysoev10fc9ef2003-10-27 08:53:49 +0000370
Igor Sysoev10fc9ef2003-10-27 08:53:49 +0000371 if (n == NGX_ERROR) {
372 return n;
373 }
374
375#if (NGX_FILE_AIO_READ)
376 if (n == NGX_AGAIN) {
377 return n;
378 }
379#endif
380
Igor Sysoev42b12b32004-12-02 18:40:46 +0000381 if (n != size) {
Igor Sysoev10fc9ef2003-10-27 08:53:49 +0000382 ngx_log_error(NGX_LOG_ALERT, src->file->log, 0,
Igor Sysoev42b12b32004-12-02 18:40:46 +0000383 ngx_read_file_n " reads only %z of %O from file",
Igor Sysoev10fc9ef2003-10-27 08:53:49 +0000384 n, size);
385 if (n == 0) {
386 return NGX_ERROR;
387 }
388 }
389
Igor Sysoev10fc9ef2003-10-27 08:53:49 +0000390 dst->last += n;
391
Igor Sysoev924bd792004-10-11 15:07:03 +0000392 if (sendfile) {
393 dst->in_file = 1;
394 dst->file = src->file;
395 dst->file_pos = src->file_pos;
Igor Sysoevc0edbcc2004-10-21 15:34:38 +0000396 dst->file_last = src->file_pos + n;
Igor Sysoev924bd792004-10-11 15:07:03 +0000397
398 } else {
Igor Sysoev369145c2004-05-28 15:49:23 +0000399 dst->in_file = 0;
Igor Sysoev10fc9ef2003-10-27 08:53:49 +0000400 }
401
Igor Sysoevc0edbcc2004-10-21 15:34:38 +0000402 src->file_pos += n;
403
Igor Sysoev369145c2004-05-28 15:49:23 +0000404 if (src->last_buf && src->file_pos == src->file_last) {
405 dst->last_buf = 1;
Igor Sysoev10fc9ef2003-10-27 08:53:49 +0000406 }
407 }
408
409 return NGX_OK;
410}
Igor Sysoev2b0c76c2003-10-27 21:01:00 +0000411
412
Igor Sysoev8184d1b2005-03-04 14:06:57 +0000413ngx_int_t
414ngx_chain_writer(void *data, ngx_chain_t *in)
Igor Sysoev2b0c76c2003-10-27 21:01:00 +0000415{
Igor Sysoev68ee8f12003-10-30 08:51:06 +0000416 ngx_chain_writer_ctx_t *ctx = data;
Igor Sysoev2b0c76c2003-10-27 21:01:00 +0000417
Igor Sysoev42b12b32004-12-02 18:40:46 +0000418 off_t size;
Igor Sysoev2b0c76c2003-10-27 21:01:00 +0000419 ngx_chain_t *cl;
420
Igor Sysoev42b12b32004-12-02 18:40:46 +0000421 for (size = 0; in; in = in->next) {
422
423#if 1
424 if (ngx_buf_size(in->buf) == 0 && !ngx_buf_special(in->buf)) {
425 ngx_debug_point();
426 }
427#endif
428
429 size += ngx_buf_size(in->buf);
Igor Sysoev967fd632004-08-27 15:40:59 +0000430
431 ngx_log_debug1(NGX_LOG_DEBUG_CORE, ctx->connection->log, 0,
Igor Sysoev1b735832004-11-11 14:07:14 +0000432 "chain writer buf size: %uz", ngx_buf_size(in->buf));
Igor Sysoev967fd632004-08-27 15:40:59 +0000433
Igor Sysoev1b735832004-11-11 14:07:14 +0000434 if (!(cl = ngx_alloc_chain_link(ctx->pool))) {
435 return NGX_ERROR;
436 }
437 cl->buf = in->buf;
438 cl->next = NULL;
Igor Sysoev2b0c76c2003-10-27 21:01:00 +0000439 *ctx->last = cl;
440 ctx->last = &cl->next;
441 }
442
Igor Sysoev9c610952004-03-16 13:35:20 +0000443 ngx_log_debug1(NGX_LOG_DEBUG_CORE, ctx->connection->log, 0,
Igor Sysoev1b735832004-11-11 14:07:14 +0000444 "chain writer in: %p", ctx->out);
Igor Sysoev9c610952004-03-16 13:35:20 +0000445
Igor Sysoev42b12b32004-12-02 18:40:46 +0000446 for (cl = ctx->out; cl; cl = cl->next) {
447
448#if 1
Igor Sysoev42b12b32004-12-02 18:40:46 +0000449 if (ngx_buf_size(cl->buf) == 0 && !ngx_buf_special(cl->buf)) {
450 ngx_debug_point();
451 }
452
453#endif
454
455 size += ngx_buf_size(cl->buf);
456 }
457
458 if (size == 0) {
459 return NGX_OK;
460 }
461
Igor Sysoevf38e0462004-07-16 17:11:43 +0000462 ctx->out = ngx_send_chain(ctx->connection, ctx->out, ctx->limit);
Igor Sysoev2b0c76c2003-10-27 21:01:00 +0000463
Igor Sysoev9c610952004-03-16 13:35:20 +0000464 ngx_log_debug1(NGX_LOG_DEBUG_CORE, ctx->connection->log, 0,
Igor Sysoev1b735832004-11-11 14:07:14 +0000465 "chain writer out: %p", ctx->out);
Igor Sysoev9c610952004-03-16 13:35:20 +0000466
Igor Sysoev2b0c76c2003-10-27 21:01:00 +0000467 if (ctx->out == NGX_CHAIN_ERROR) {
468 return NGX_ERROR;
469 }
470
471 if (ctx->out == NULL) {
472 ctx->last = &ctx->out;
473 return NGX_OK;
474 }
475
476 return NGX_AGAIN;
477}