Mercurial > hg > nginx
view src/http/ngx_http_filter.c @ 5:62b1a364857c
nginx-0.0.1-2002-08-23-20:14:30 import
author | Igor Sysoev <igor@sysoev.ru> |
---|---|
date | Fri, 23 Aug 2002 16:14:30 +0000 |
parents | c5f071d376e5 |
children | b5481d6fbbd4 |
line wrap: on
line source
#include <ngx_hunk.h> #include <ngx_http.h> #include <ngx_http_filter.h> ngx_http_module_t ngx_http_filter_module; /* STUB */ static ngx_http_filter_ctx_t module_ctx; void ngx_http_filter_init() { module_ctx.hunk_size = 32 * 1024; module_ctx.out.hunk = NULL; module_ctx.out.next = NULL; module_ctx.next_filter = ngx_http_write_filter; ngx_http_filter_module.ctx = &module_ctx; } /* */ /* int ngx_http_filter(ngx_http_request_t *r, ngx_chain_t *in) */ /* flags NGX_HUNK_RECYCLED, NGX_HUNK_FLUSH, NGX_HUNK_LAST */ int ngx_http_filter(ngx_http_request_t *r, ngx_hunk_t *hunk) { int rc; size_t size; ssize_t n; ngx_chain_t *ce; ngx_http_filter_ctx_t *ctx; ctx = (ngx_http_filter_ctx_t *) ngx_get_module_ctx(r->main ? r->main : r, &ngx_http_filter_module); if (hunk && (hunk->type & NGX_HUNK_LAST)) ctx->last = 1; /* input chain is not empty */ if (ctx->in) { while (ctx->in) { /* add hunk to input chain */ if (hunk) { for (ce = ctx->in; ce->next; ce = ce->next) /* void */ ; ngx_add_hunk_to_chain(ce->next, hunk, r->pool, NGX_ERROR); } /* our hunk is still busy */ if (ctx->hunk->pos.mem < ctx->hunk->last.mem) { rc = ctx->next_filter(r, NULL); /* our hunk is free */ } else { ctx->out.hunk = ctx->hunk; rc = ngx_http_filter_copy_hunk(ctx->hunk, ctx->in->hunk); #if (NGX_FILE_AIO) if (rc == NGX_AGAIN) return rc; #endif if (rc == NGX_ERROR) return rc; /* whole hunk is copied so we send to next filter chain part up to next hunk that need to be copied */ if (ctx->in->hunk->pos.mem == ctx->in->hunk->last.mem) { ctx->out.next = ctx->in->next; for (ce = ctx->in->next; ce; ce = ce->next) { if (ce->type & NGX_HUNK_FILE) break; if ((ce->type & NGX_HUNK_MEMORY|NGX_HUNK_MMAP) && (r->filter & NGX_HTTP_FILTER_NEED_TEMP)) break; } ctx->out.next = ce; } else { ctx->out.next = NULL; } rc = ctx->next_filter(r, &ctx->out); } if (rc == NGX_OK) ctx->hunk->pos.mem = ctx->hunk->last.mem = ctx->hunk->start; else return rc; } /* input chain is empty */ } else { if (hunk == NULL) { rc = ctx->next_filter(r, NULL); } else { /* we need to copy hunk to our hunk */ if (((r->filter & NGX_HTTP_FILTER_NEED_IN_MEMORY) && (hunk->type & NGX_HUNK_FILE)) || ((r->filter & NGX_HTTP_FILTER_NEED_TEMP) && (hunk->type & NGX_HUNK_MEMORY|NGX_HUNK_MMAP)) ) { /* out hunk is still busy */ if (ctx->hunk && ctx->hunk->pos.mem < ctx->hunk->last.mem) { ngx_add_hunk_to_chain(ctx->in, hunk, r->pool, NGX_ERROR); rc = ctx->next_filter(r, NULL); } else { if (ctx->hunk == NULL) { if (hunk->type & NGX_HUNK_LAST) { size = hunk->last.mem - hunk->pos.mem; if (size > ctx->hunk_size) size = ctx->hunk_size; } else { size = ctx->hunk_size; } ngx_test_null(ctx->hunk, ngx_create_temp_hunk(r->pool, size, 50, 50), NGX_ERROR); rc = ngx_http_filter_copy_hunk(ctx->hunk, ctx->in->hunk); #if (NGX_FILE_AIO) if (rc == NGX_AGAIN) { /* add hunk to input chain */ ngx_add_hunk_to_chain(ctx->in, hunk, r->pool, NGX_ERROR); return rc; } #endif if (rc == NGX_ERROR) return rc; if (ctx->in->hunk->pos.mem < ctx->in->hunk->last.mem) ngx_add_hunk_to_chain(ctx->in, hunk, r->pool, NGX_ERROR); ctx->out.hunk = ctx->hunk; ctx->out.next = NULL; rc = ctx->next_filter(r, &ctx->out); } } } else { ctx->out.hunk = hunk; ctx->out.next = NULL; rc = ctx->next_filter(r, &ctx->out); } } } if (rc == NGX_OK && ctx->last) { /* STUB */ return NGX_ERROR; } if (rc == NGX_OK) { ctx->hunk->pos.mem = ctx->hunk->last.mem = ctx->hunk->start; #if level_event ngx_del_event(r->connection->write, NGX_WRITE_EVENT); #endif } return rc; } int ngx_http_filter_copy_hunk(ngx_hunk_t *dst, ngx_hunk_t *src, ngx_log_t *log) { size_t size; size = hunk->last.mem - hunk->pos.mem; if (size > dst->end - dst->pos.mem) size = dst->end - dst->pos.mem; if (src->type & NGX_HUNK_FILE) { n = ngx_read_file(src->handle, dst->pos.mem, size); if (n == NGX_ERROR) { ngx_log_error(NGX_LOG_ERR, log, ngx_errno, ngx_read_file_n " failed for client"); return n; #if (NGX_FILE_AIO) } else if (n == NGX_AGAIN) { return n; #endif } else { ngx_assert((n == size), /* void */ ; , log, ngx_read_file_n " reads only %d of %d for client" _ n _ size); } src->pos.mem += n; dst->last.mem += n; } else { ngx_memcpy(src->pos.mem, dst->pos.mem, size); src->pos.mem += size; dst->last.mem += size; } return NGX_OK; }