blob: a1ac1847931f08d60a9a99a43f26e701de41851c [file] [log] [blame]
Valentin Bartenev2686cb42013-03-20 10:36:57 +00001
2/*
3 * Copyright (C) Nginx, Inc.
4 * Copyright (C) Valentin V. Bartenev
5 */
6
7
8#include <ngx_config.h>
9#include <ngx_core.h>
10#include <ngx_http.h>
11#include <ngx_http_spdy_module.h>
12
13#include <zlib.h>
14
15
16#if (NGX_HAVE_LITTLE_ENDIAN && NGX_HAVE_NONALIGNED)
17
18#define ngx_str5cmp(m, c0, c1, c2, c3, c4) \
19 *(uint32_t *) m == (c3 << 24 | c2 << 16 | c1 << 8 | c0) \
20 && m[4] == c4
21
22#else
23
24#define ngx_str5cmp(m, c0, c1, c2, c3, c4) \
25 m[0] == c0 && m[1] == c1 && m[2] == c2 && m[3] == c3 && m[4] == c4
26
27#endif
28
29
30#if (NGX_HAVE_NONALIGNED)
31
32#define ngx_spdy_frame_parse_uint16(p) ntohs(*(uint16_t *) (p))
33#define ngx_spdy_frame_parse_uint32(p) ntohl(*(uint32_t *) (p))
34
35#else
36
37#define ngx_spdy_frame_parse_uint16(p) ((p)[0] << 8 | (p)[1])
38#define ngx_spdy_frame_parse_uint32(p) \
39 ((p)[0] << 24 | (p)[1] << 16 | (p)[2] << 8 | (p)[3])
40
41#endif
42
43#define ngx_spdy_frame_parse_sid(p) \
44 (ngx_spdy_frame_parse_uint32(p) & 0x7fffffff)
45
46
47#define ngx_spdy_ctl_frame_check(h) \
48 (((h) & 0xffffff00) == ngx_spdy_ctl_frame_head(0))
49#define ngx_spdy_data_frame_check(h) \
50 (!((h) & (uint32_t) NGX_SPDY_CTL_BIT << 31))
51
52#define ngx_spdy_ctl_frame_type(h) ((h) & 0x000000ff)
53#define ngx_spdy_frame_flags(p) ((p) >> 24)
54#define ngx_spdy_frame_length(p) ((p) & 0x00ffffff)
55
56
57#define NGX_SPDY_SKIP_HEADERS_BUFFER_SIZE 4096
58#define NGX_SPDY_CTL_FRAME_BUFFER_SIZE 16
59
60#define NGX_SPDY_PROTOCOL_ERROR 1
61#define NGX_SPDY_INVALID_STREAM 2
62#define NGX_SPDY_REFUSED_STREAM 3
63#define NGX_SPDY_UNSUPPORTED_VERSION 4
64#define NGX_SPDY_CANCEL 5
65#define NGX_SPDY_INTERNAL_ERROR 6
66#define NGX_SPDY_FLOW_CONTROL_ERROR 7
67
68#define NGX_SPDY_SETTINGS_MAX_STREAMS 4
69
70#define NGX_SPDY_SETTINGS_FLAG_PERSIST 0x01
71
72typedef struct {
73 ngx_uint_t hash;
74 u_char len;
75 u_char header[7];
76 ngx_int_t (*handler)(ngx_http_request_t *r);
77} ngx_http_spdy_request_header_t;
78
79
80static void ngx_http_spdy_read_handler(ngx_event_t *rev);
81static void ngx_http_spdy_write_handler(ngx_event_t *wev);
82static void ngx_http_spdy_handle_connection(ngx_http_spdy_connection_t *sc);
83
84static u_char *ngx_http_spdy_state_detect_settings(
85 ngx_http_spdy_connection_t *sc, u_char *pos, u_char *end);
86static u_char *ngx_http_spdy_state_head(ngx_http_spdy_connection_t *sc,
87 u_char *pos, u_char *end);
88static u_char *ngx_http_spdy_state_syn_stream(ngx_http_spdy_connection_t *sc,
89 u_char *pos, u_char *end);
90static u_char *ngx_http_spdy_state_headers(ngx_http_spdy_connection_t *sc,
91 u_char *pos, u_char *end);
92static u_char *ngx_http_spdy_state_headers_error(ngx_http_spdy_connection_t *sc,
93 u_char *pos, u_char *end);
94static u_char *ngx_http_spdy_state_headers_skip(ngx_http_spdy_connection_t *sc,
95 u_char *pos, u_char *end);
96static u_char *ngx_http_spdy_state_data(ngx_http_spdy_connection_t *sc,
97 u_char *pos, u_char *end);
98static u_char *ngx_http_spdy_state_rst_stream(ngx_http_spdy_connection_t *sc,
99 u_char *pos, u_char *end);
100static u_char *ngx_http_spdy_state_ping(ngx_http_spdy_connection_t *sc,
101 u_char *pos, u_char *end);
102static u_char *ngx_http_spdy_state_skip(ngx_http_spdy_connection_t *sc,
103 u_char *pos, u_char *end);
104static u_char *ngx_http_spdy_state_settings(ngx_http_spdy_connection_t *sc,
105 u_char *pos, u_char *end);
106static u_char *ngx_http_spdy_state_noop(ngx_http_spdy_connection_t *sc,
107 u_char *pos, u_char *end);
108static u_char *ngx_http_spdy_state_complete(ngx_http_spdy_connection_t *sc,
109 u_char *pos, u_char *end);
110static u_char *ngx_http_spdy_state_save(ngx_http_spdy_connection_t *sc,
111 u_char *pos, u_char *end, ngx_http_spdy_handler_pt handler);
112static u_char *ngx_http_spdy_state_protocol_error(
113 ngx_http_spdy_connection_t *sc);
114static u_char *ngx_http_spdy_state_internal_error(
115 ngx_http_spdy_connection_t *sc);
116
117static ngx_int_t ngx_http_spdy_send_rst_stream(ngx_http_spdy_connection_t *sc,
118 ngx_uint_t sid, ngx_uint_t status, ngx_uint_t priority);
119static ngx_int_t ngx_http_spdy_send_settings(ngx_http_spdy_connection_t *sc);
120static ngx_int_t ngx_http_spdy_settings_frame_handler(
121 ngx_http_spdy_connection_t *sc, ngx_http_spdy_out_frame_t *frame);
122static ngx_http_spdy_out_frame_t *ngx_http_spdy_get_ctl_frame(
123 ngx_http_spdy_connection_t *sc, size_t size, ngx_uint_t priority);
124static ngx_int_t ngx_http_spdy_ctl_frame_handler(
125 ngx_http_spdy_connection_t *sc, ngx_http_spdy_out_frame_t *frame);
126
127static ngx_http_spdy_stream_t *ngx_http_spdy_create_stream(
128 ngx_http_spdy_connection_t *sc, ngx_uint_t id, ngx_uint_t priority);
129static ngx_http_spdy_stream_t *ngx_http_spdy_get_stream_by_id(
130 ngx_http_spdy_connection_t *sc, ngx_uint_t sid);
131#define ngx_http_spdy_streams_index_size(sscf) (sscf->streams_index_mask + 1)
132#define ngx_http_spdy_stream_index(sscf, sid) \
133 ((sid >> 1) & sscf->streams_index_mask)
134
135static ngx_int_t ngx_http_spdy_parse_header(ngx_http_request_t *r);
136static ngx_int_t ngx_http_spdy_alloc_large_header_buffer(ngx_http_request_t *r);
137
138static ngx_int_t ngx_http_spdy_handle_request_header(ngx_http_request_t *r);
139static ngx_int_t ngx_http_spdy_parse_method(ngx_http_request_t *r);
140static ngx_int_t ngx_http_spdy_parse_scheme(ngx_http_request_t *r);
141static ngx_int_t ngx_http_spdy_parse_url(ngx_http_request_t *r);
142static ngx_int_t ngx_http_spdy_parse_version(ngx_http_request_t *r);
143
144static ngx_int_t ngx_http_spdy_construct_request_line(ngx_http_request_t *r);
145static void ngx_http_spdy_run_request(ngx_http_request_t *r);
146static ngx_int_t ngx_http_spdy_init_request_body(ngx_http_request_t *r);
147
148static void ngx_http_spdy_handle_connection_handler(ngx_event_t *rev);
149static void ngx_http_spdy_keepalive_handler(ngx_event_t *rev);
150static void ngx_http_spdy_finalize_connection(ngx_http_spdy_connection_t *sc,
151 ngx_int_t rc);
152
153static void ngx_http_spdy_pool_cleanup(void *data);
154
155static void *ngx_http_spdy_zalloc(void *opaque, u_int items, u_int size);
156static void ngx_http_spdy_zfree(void *opaque, void *address);
157
158
159static const u_char ngx_http_spdy_dict[] =
160 "options" "get" "head" "post" "put" "delete" "trace"
161 "accept" "accept-charset" "accept-encoding" "accept-language"
162 "authorization" "expect" "from" "host"
163 "if-modified-since" "if-match" "if-none-match" "if-range"
164 "if-unmodifiedsince" "max-forwards" "proxy-authorization"
165 "range" "referer" "te" "user-agent"
166 "100" "101" "200" "201" "202" "203" "204" "205" "206"
167 "300" "301" "302" "303" "304" "305" "306" "307"
168 "400" "401" "402" "403" "404" "405" "406" "407" "408" "409" "410"
169 "411" "412" "413" "414" "415" "416" "417"
170 "500" "501" "502" "503" "504" "505"
171 "accept-ranges" "age" "etag" "location" "proxy-authenticate" "public"
172 "retry-after" "server" "vary" "warning" "www-authenticate" "allow"
173 "content-base" "content-encoding" "cache-control" "connection" "date"
174 "trailer" "transfer-encoding" "upgrade" "via" "warning"
175 "content-language" "content-length" "content-location"
176 "content-md5" "content-range" "content-type" "etag" "expires"
177 "last-modified" "set-cookie"
178 "Monday" "Tuesday" "Wednesday" "Thursday" "Friday" "Saturday" "Sunday"
179 "Jan" "Feb" "Mar" "Apr" "May" "Jun" "Jul" "Aug" "Sep" "Oct" "Nov" "Dec"
180 "chunked" "text/html" "image/png" "image/jpg" "image/gif"
181 "application/xml" "application/xhtml" "text/plain" "public" "max-age"
182 "charset=iso-8859-1" "utf-8" "gzip" "deflate" "HTTP/1.1" "status"
183 "version" "url";
184
185
186static ngx_http_spdy_request_header_t ngx_http_spdy_request_headers[] = {
187 { 0, 6, "method", ngx_http_spdy_parse_method },
188 { 0, 6, "scheme", ngx_http_spdy_parse_scheme },
189 { 0, 3, "url", ngx_http_spdy_parse_url },
190 { 0, 7, "version", ngx_http_spdy_parse_version },
191};
192
193#define NGX_SPDY_REQUEST_HEADERS \
194 (sizeof(ngx_http_spdy_request_headers) \
195 / sizeof(ngx_http_spdy_request_header_t))
196
197
198void
199ngx_http_spdy_init(ngx_event_t *rev)
200{
201 int rc;
202 ngx_connection_t *c;
203 ngx_pool_cleanup_t *cln;
204 ngx_http_connection_t *hc;
205 ngx_http_spdy_srv_conf_t *sscf;
206 ngx_http_spdy_main_conf_t *smcf;
207 ngx_http_spdy_connection_t *sc;
208
209 c = rev->data;
210 hc = c->data;
211
212 ngx_log_debug0(NGX_LOG_DEBUG_HTTP, c->log, 0,
213 "init spdy request");
214
215 c->log->action = "processing SPDY";
216
217 smcf = ngx_http_get_module_main_conf(hc->conf_ctx, ngx_http_spdy_module);
218
219 if (smcf->recv_buffer == NULL) {
220 smcf->recv_buffer = ngx_palloc(ngx_cycle->pool, smcf->recv_buffer_size);
221 if (smcf->recv_buffer == NULL) {
222 ngx_http_close_connection(c);
223 return;
224 }
225 }
226
227 sc = ngx_pcalloc(c->pool, sizeof(ngx_http_spdy_connection_t));
228 if (sc == NULL) {
229 ngx_http_close_connection(c);
230 return;
231 }
232
233 sc->connection = c;
234 sc->http_connection = hc;
235
236 sc->handler = ngx_http_spdy_state_detect_settings;
237
238 sc->zstream_in.zalloc = ngx_http_spdy_zalloc;
239 sc->zstream_in.zfree = ngx_http_spdy_zfree;
240 sc->zstream_in.opaque = sc;
241
242 rc = inflateInit(&sc->zstream_in);
243 if (rc != Z_OK) {
244 ngx_log_error(NGX_LOG_ALERT, c->log, 0,
245 "inflateInit() failed: %d", rc);
246 ngx_http_close_connection(c);
247 return;
248 }
249
250 sc->zstream_out.zalloc = ngx_http_spdy_zalloc;
251 sc->zstream_out.zfree = ngx_http_spdy_zfree;
252 sc->zstream_out.opaque = sc;
253
254 sscf = ngx_http_get_module_srv_conf(hc->conf_ctx, ngx_http_spdy_module);
255
256 rc = deflateInit2(&sc->zstream_out, (int) sscf->headers_comp,
257 Z_DEFLATED, 11, 4, Z_DEFAULT_STRATEGY);
258
259 if (rc != Z_OK) {
260 ngx_log_error(NGX_LOG_ALERT, c->log, 0,
261 "deflateInit2() failed: %d", rc);
262 ngx_http_close_connection(c);
263 return;
264 }
265
266 rc = deflateSetDictionary(&sc->zstream_out, ngx_http_spdy_dict,
267 sizeof(ngx_http_spdy_dict));
268 if (rc != Z_OK) {
269 ngx_log_error(NGX_LOG_ALERT, c->log, 0,
270 "deflateSetDictionary() failed: %d", rc);
271 ngx_http_close_connection(c);
272 return;
273 }
274
275 sc->pool = ngx_create_pool(sscf->pool_size, sc->connection->log);
276 if (sc->pool == NULL) {
277 ngx_http_close_connection(c);
278 return;
279 }
280
281 cln = ngx_pool_cleanup_add(c->pool, sizeof(ngx_pool_cleanup_file_t));
282 if (cln == NULL) {
283 ngx_http_close_connection(c);
284 return;
285 }
286
287 cln->handler = ngx_http_spdy_pool_cleanup;
288 cln->data = sc;
289
290 sc->streams_index = ngx_pcalloc(sc->pool,
291 ngx_http_spdy_streams_index_size(sscf)
292 * sizeof(ngx_http_spdy_stream_t *));
293 if (sc->streams_index == NULL) {
294 ngx_http_close_connection(c);
295 return;
296 }
297
298 c->data = sc;
299
300 rev->handler = ngx_http_spdy_read_handler;
301 c->write->handler = ngx_http_spdy_write_handler;
302
303 ngx_http_spdy_read_handler(rev);
304}
305
306
307static void
308ngx_http_spdy_read_handler(ngx_event_t *rev)
309{
310 u_char *p, *end;
311 size_t available;
312 ssize_t n;
313 ngx_connection_t *c;
314 ngx_http_spdy_main_conf_t *smcf;
315 ngx_http_spdy_connection_t *sc;
316
317 c = rev->data;
318 sc = c->data;
319
320 if (rev->timedout) {
321 ngx_log_error(NGX_LOG_INFO, c->log, NGX_ETIMEDOUT, "client timed out");
322 ngx_http_spdy_finalize_connection(sc, NGX_HTTP_REQUEST_TIME_OUT);
323 return;
324 }
325
326 ngx_log_debug0(NGX_LOG_DEBUG_HTTP, c->log, 0, "spdy read handler");
327
328 sc->blocked = 1;
329
330 smcf = ngx_http_get_module_main_conf(sc->http_connection->conf_ctx,
331 ngx_http_spdy_module);
332
333 available = smcf->recv_buffer_size - 2 * NGX_SPDY_STATE_BUFFER_SIZE;
334
335 do {
336 p = smcf->recv_buffer;
337
338 ngx_memcpy(p, sc->buffer, NGX_SPDY_STATE_BUFFER_SIZE);
339 end = p + sc->buffer_used;
340
341 n = c->recv(c, end, available);
342
343 if (n == NGX_AGAIN) {
344 break;
345 }
346
347 if (n == 0 && (sc->waiting || sc->processing)) {
348 ngx_log_error(NGX_LOG_INFO, c->log, 0,
349 "client closed prematurely connection");
350 }
351
352 if (n == 0 || n == NGX_ERROR) {
353 ngx_http_spdy_finalize_connection(sc,
354 NGX_HTTP_CLIENT_CLOSED_REQUEST);
355 return;
356 }
357
358 end += n;
359
360 sc->buffer_used = 0;
361 sc->waiting = 0;
362
363 do {
364 p = sc->handler(sc, p, end);
365
366 if (p == NULL) {
367 return;
368 }
369
370 } while (p != end);
371
372 } while (rev->ready);
373
374 if (ngx_handle_read_event(rev, 0) != NGX_OK) {
375 ngx_http_spdy_finalize_connection(sc, NGX_HTTP_INTERNAL_SERVER_ERROR);
376 return;
377 }
378
379 sc->blocked = 0;
380
381 if (sc->processing) {
382 if (rev->timer_set) {
383 ngx_del_timer(rev);
384 }
385 return;
386 }
387
388 ngx_http_spdy_handle_connection(sc);
389}
390
391
392static void
393ngx_http_spdy_write_handler(ngx_event_t *wev)
394{
395 ngx_int_t rc;
396 ngx_connection_t *c;
397 ngx_http_spdy_stream_t *stream, *s, *sn;
398 ngx_http_spdy_connection_t *sc;
399
400 c = wev->data;
401 sc = c->data;
402
403 if (wev->timedout) {
404 ngx_log_debug0(NGX_LOG_DEBUG_HTTP, c->log, 0,
405 "spdy write event timed out");
406 ngx_http_spdy_finalize_connection(sc, NGX_HTTP_CLIENT_CLOSED_REQUEST);
407 return;
408 }
409
410 ngx_log_debug0(NGX_LOG_DEBUG_HTTP, c->log, 0, "spdy write handler");
411
412 sc->blocked = 2;
413
414 rc = ngx_http_spdy_send_output_queue(sc);
415
416 if (rc == NGX_ERROR) {
417 ngx_http_spdy_finalize_connection(sc, NGX_HTTP_CLIENT_CLOSED_REQUEST);
418 return;
419 }
420
421 stream = NULL;
422
423 for (s = sc->last_stream; s; s = sn) {
424 sn = s->next;
425 s->next = stream;
426 stream = s;
427 }
428
429 sc->last_stream = NULL;
430
431 sc->blocked = 1;
432
433 for ( /* void */ ; stream; stream = sn) {
434 sn = stream->next;
435 stream->handled = 0;
436
437 ngx_log_debug1(NGX_LOG_DEBUG_HTTP, c->log, 0,
438 "spdy run stream %ui", stream->id);
439
440 wev = stream->request->connection->write;
441 wev->handler(wev);
442 }
443
444 sc->blocked = 0;
445
446 if (rc == NGX_AGAIN) {
447 return;
448 }
449
450 ngx_http_spdy_handle_connection(sc);
451}
452
453
454ngx_int_t
455ngx_http_spdy_send_output_queue(ngx_http_spdy_connection_t *sc)
456{
457 ngx_chain_t *cl;
458 ngx_event_t *wev;
459 ngx_connection_t *c;
460 ngx_http_core_loc_conf_t *clcf;
461 ngx_http_spdy_out_frame_t *out, *frame, *fn;
462
463 c = sc->connection;
464
465 if (c->error) {
466 return NGX_ERROR;
467 }
468
469 wev = c->write;
470
471 if (!wev->ready) {
472 return NGX_OK;
473 }
474
475 cl = NULL;
476 out = NULL;
477
478 for (frame = sc->last_out; frame; frame = fn) {
479 frame->last->next = cl;
480 cl = frame->first;
481
482 fn = frame->next;
483 frame->next = out;
484 out = frame;
485
486 ngx_log_debug5(NGX_LOG_DEBUG_HTTP, c->log, 0,
487 "spdy frame out: %p sid:%ui prio:%ui bl:%ui size:%uz",
488 out, out->stream ? out->stream->id : 0, out->priority,
489 out->blocked, out->size);
490 }
491
492 cl = c->send_chain(c, cl, 0);
493
494 if (cl == NGX_CHAIN_ERROR) {
495 c->error = 1;
496
497 if (!sc->blocked) {
498 ngx_post_event(wev, &ngx_posted_events);
499 }
500
501 return NGX_ERROR;
502 }
503
504 clcf = ngx_http_get_module_loc_conf(sc->http_connection->conf_ctx,
505 ngx_http_core_module);
506
507 if (ngx_handle_write_event(wev, clcf->send_lowat) != NGX_OK) {
508 return NGX_ERROR; /* FIXME */
509 }
510
511 if (cl) {
512 ngx_add_timer(wev, clcf->send_timeout);
513
514 } else {
515 if (wev->timer_set) {
516 ngx_del_timer(wev);
517 }
518 }
519
520 for ( /* void */ ; out; out = out->next) {
521 if (out->handler(sc, out) != NGX_OK) {
522 out->blocked = 1;
523 out->priority = NGX_SPDY_HIGHEST_PRIORITY;
524 break;
525 }
526
527 ngx_log_debug4(NGX_LOG_DEBUG_HTTP, c->log, 0,
528 "spdy frame sent: %p sid:%ui bl:%ui size:%uz",
529 out, out->stream ? out->stream->id : 0,
530 out->blocked, out->size);
531 }
532
533 frame = NULL;
534
535 for ( /* void */ ; out; out = fn) {
536 fn = out->next;
537 out->next = frame;
538 frame = out;
539 }
540
541 sc->last_out = frame;
542
543 return NGX_OK;
544}
545
546
547static void
548ngx_http_spdy_handle_connection(ngx_http_spdy_connection_t *sc)
549{
550 ngx_connection_t *c;
551 ngx_http_spdy_srv_conf_t *sscf;
552
553 if (sc->last_out || sc->processing) {
554 return;
555 }
556
557 c = sc->connection;
558
559 if (c->error) {
560 ngx_http_close_connection(c);
561 return;
562 }
563
564 if (c->buffered) {
565 return;
566 }
567
568 sscf = ngx_http_get_module_srv_conf(sc->http_connection->conf_ctx,
569 ngx_http_spdy_module);
570 if (sc->waiting) {
571 ngx_add_timer(c->read, sscf->recv_timeout);
572 return;
573 }
574
575 if (ngx_terminate || ngx_exiting) {
576 ngx_http_close_connection(c);
577 return;
578 }
579
580 ngx_destroy_pool(sc->pool);
581
582 sc->pool = NULL;
583 sc->free_ctl_frames = NULL;
584 sc->free_fake_connections = NULL;
585
586#if (NGX_HTTP_SSL)
587 if (c->ssl) {
588 ngx_ssl_free_buffer(c);
589 }
590#endif
591
592 c->destroyed = 1;
593 c->idle = 1;
594 ngx_reusable_connection(c, 1);
595
596 c->write->handler = ngx_http_empty_handler;
597 c->read->handler = ngx_http_spdy_keepalive_handler;
598
599 if (c->write->timer_set) {
600 ngx_del_timer(c->write);
601 }
602
603 ngx_add_timer(c->read, sscf->keepalive_timeout);
604}
605
606
607static u_char *
608ngx_http_spdy_state_detect_settings(ngx_http_spdy_connection_t *sc,
609 u_char *pos, u_char *end)
610{
611 if (end - pos < NGX_SPDY_FRAME_HEADER_SIZE) {
612 return ngx_http_spdy_state_save(sc, pos, end,
613 ngx_http_spdy_state_detect_settings);
614 }
615
616 /*
617 * Since this is the first frame in a buffer,
618 * then it is properly aligned
619 */
620
621 if (*(uint32_t *) pos == htonl(ngx_spdy_ctl_frame_head(NGX_SPDY_SETTINGS)))
622 {
623 sc->length = ngx_spdy_frame_length(htonl(((uint32_t *) pos)[1]));
624
625 ngx_log_debug1(NGX_LOG_DEBUG_HTTP, sc->connection->log, 0,
626 "spdy SETTINGS frame received, size: %uz", sc->length);
627
628 pos += NGX_SPDY_FRAME_HEADER_SIZE;
629
630 return ngx_http_spdy_state_settings(sc, pos, end);
631 }
632
633 ngx_http_spdy_send_settings(sc);
634
635 return ngx_http_spdy_state_head(sc, pos, end);
636}
637
638
639static u_char *
640ngx_http_spdy_state_head(ngx_http_spdy_connection_t *sc, u_char *pos,
641 u_char *end)
642{
643 uint32_t head, flen;
644
645 if (end - pos < NGX_SPDY_FRAME_HEADER_SIZE) {
646 return ngx_http_spdy_state_save(sc, pos, end,
647 ngx_http_spdy_state_head);
648 }
649
650 head = ngx_spdy_frame_parse_uint32(pos);
651
652 pos += sizeof(uint32_t);
653
654 flen = ngx_spdy_frame_parse_uint32(pos);
655
656 sc->flags = ngx_spdy_frame_flags(flen);
657 sc->length = ngx_spdy_frame_length(flen);
658
659 pos += sizeof(uint32_t);
660
661 ngx_log_debug3(NGX_LOG_DEBUG_HTTP, sc->connection->log, 0,
662 "spdy process frame head:%08Xd f:%ui l:%ui",
663 head, sc->flags, sc->length);
664
665 if (ngx_spdy_ctl_frame_check(head)) {
666 switch (ngx_spdy_ctl_frame_type(head)) {
667
668 case NGX_SPDY_SYN_STREAM:
669 return ngx_http_spdy_state_syn_stream(sc, pos, end);
670
671 case NGX_SPDY_SYN_REPLY:
672 return ngx_http_spdy_state_protocol_error(sc);
673
674 case NGX_SPDY_RST_STREAM:
675 return ngx_http_spdy_state_rst_stream(sc, pos, end);
676
677 case NGX_SPDY_SETTINGS:
678 return ngx_http_spdy_state_skip(sc, pos, end);
679
680 case NGX_SPDY_NOOP:
681 return ngx_http_spdy_state_noop(sc, pos, end);
682
683 case NGX_SPDY_PING:
684 return ngx_http_spdy_state_ping(sc, pos, end);
685
686 case NGX_SPDY_GOAWAY:
687 return ngx_http_spdy_state_skip(sc, pos, end); /* TODO */
688
689 case NGX_SPDY_HEADERS:
690 return ngx_http_spdy_state_protocol_error(sc);
691
692 default: /* TODO logging */
693 return ngx_http_spdy_state_skip(sc, pos, end);
694 }
695 }
696
697 if (ngx_spdy_data_frame_check(head)) {
698 sc->stream = ngx_http_spdy_get_stream_by_id(sc, head);
699 return ngx_http_spdy_state_data(sc, pos, end);
700 }
701
702
703 /* TODO version & type check */
704 ngx_log_debug0(NGX_LOG_DEBUG_HTTP, sc->connection->log, 0,
705 "spdy unknown frame");
706
707 return ngx_http_spdy_state_protocol_error(sc);
708}
709
710
711static u_char *
712ngx_http_spdy_state_syn_stream(ngx_http_spdy_connection_t *sc, u_char *pos,
713 u_char *end)
714{
715 ngx_uint_t sid, prio;
716 ngx_http_spdy_stream_t *stream;
717 ngx_http_spdy_srv_conf_t *sscf;
718
719 if (end - pos < NGX_SPDY_SYN_STREAM_SIZE) {
720 return ngx_http_spdy_state_save(sc, pos, end,
721 ngx_http_spdy_state_syn_stream);
722 }
723
724 if (sc->length <= NGX_SPDY_SYN_STREAM_SIZE) {
725 /* TODO logging */
726 return ngx_http_spdy_state_protocol_error(sc);
727 }
728
729 sc->length -= NGX_SPDY_SYN_STREAM_SIZE;
730
731 sid = ngx_spdy_frame_parse_sid(pos);
732 prio = pos[8] >> 6;
733
734 pos += NGX_SPDY_SYN_STREAM_SIZE;
735
736 ngx_log_debug2(NGX_LOG_DEBUG_HTTP, sc->connection->log, 0,
737 "spdy SYN_STREAM frame sid:%ui prio:%ui", sid, prio);
738
739 sscf = ngx_http_get_module_srv_conf(sc->http_connection->conf_ctx,
740 ngx_http_spdy_module);
741
742 if (sc->processing >= sscf->concurrent_streams) {
743
744 ngx_log_error(NGX_LOG_INFO, sc->connection->log, 0,
745 "spdy concurrent streams excessed %ui", sc->processing);
746
747 if (ngx_http_spdy_send_rst_stream(sc, sid, NGX_SPDY_REFUSED_STREAM,
748 prio)
749 != NGX_OK)
750 {
751 return ngx_http_spdy_state_internal_error(sc);
752 }
753
754 return ngx_http_spdy_state_headers_skip(sc, pos, end);
755 }
756
757 stream = ngx_http_spdy_create_stream(sc, sid, prio);
758 if (stream == NULL) {
759 return ngx_http_spdy_state_internal_error(sc);
760 }
761
762 stream->in_closed = (sc->flags & NGX_SPDY_FLAG_FIN) ? 1 : 0;
763
764 stream->request->request_length = NGX_SPDY_FRAME_HEADER_SIZE
765 + NGX_SPDY_SYN_STREAM_SIZE
766 + sc->length;
767
768 sc->stream = stream;
769
770 sc->last_sid = sid;
771
772 return ngx_http_spdy_state_headers(sc, pos, end);
773}
774
775
776static u_char *
777ngx_http_spdy_state_headers(ngx_http_spdy_connection_t *sc, u_char *pos,
778 u_char *end)
779{
780 int z;
781 size_t size;
782 ngx_buf_t *buf;
783 ngx_int_t rc;
784 ngx_uint_t complete;
785 ngx_http_request_t *r;
786
787 size = end - pos;
788
789 if (size == 0) {
790 return ngx_http_spdy_state_save(sc, pos, end,
791 ngx_http_spdy_state_headers);
792 }
793
794 if (size >= sc->length) {
795 size = sc->length;
796 complete = 1;
797
798 } else {
799 complete = 0;
800 }
801
802 r = sc->stream->request;
803
804 ngx_log_debug2(NGX_LOG_DEBUG_HTTP, r->connection->log, 0,
805 "spdy process HEADERS %uz of %uz", size, sc->length);
806
807 buf = r->header_in;
808
809 sc->zstream_in.next_in = pos;
810 sc->zstream_in.avail_in = size;
811 sc->zstream_in.next_out = buf->last;
Valentin Bartenev3be925b2013-08-15 19:14:58 +0400812
813 /* one byte is reserved for null-termination of the last header value */
Valentin Bartenev2686cb42013-03-20 10:36:57 +0000814 sc->zstream_in.avail_out = buf->end - buf->last - 1;
815
816 z = inflate(&sc->zstream_in, Z_NO_FLUSH);
817
818 if (z == Z_NEED_DICT) {
819 z = inflateSetDictionary(&sc->zstream_in, ngx_http_spdy_dict,
820 sizeof(ngx_http_spdy_dict));
821 if (z != Z_OK) {
822 ngx_log_error(NGX_LOG_INFO, r->connection->log, 0,
823 "spdy inflateSetDictionary() failed: %d", z);
824 ngx_http_spdy_close_stream(sc->stream, 0);
825 return ngx_http_spdy_state_protocol_error(sc);
826 }
827
828 ngx_log_debug1(NGX_LOG_DEBUG_HTTP, r->connection->log, 0,
829 "spdy inflateSetDictionary(): %d", z);
830
831 z = sc->zstream_in.avail_in ? inflate(&sc->zstream_in, Z_NO_FLUSH)
832 : Z_OK;
833 }
834
835 if (z != Z_OK) {
836 ngx_log_error(NGX_LOG_INFO, r->connection->log, 0,
837 "spdy inflate() failed: %d", z);
838 ngx_http_spdy_close_stream(sc->stream, 0);
839 return ngx_http_spdy_state_protocol_error(sc);
840 }
841
842 ngx_log_debug5(NGX_LOG_DEBUG_HTTP, r->connection->log, 0,
843 "spdy inflate out: ni:%p no:%p ai:%ud ao:%ud rc:%d",
844 sc->zstream_in.next_in, sc->zstream_in.next_out,
845 sc->zstream_in.avail_in, sc->zstream_in.avail_out,
846 z);
847
848 sc->length -= sc->zstream_in.next_in - pos;
849 pos = sc->zstream_in.next_in;
850
851 buf->last = sc->zstream_in.next_out;
852
853 if (r->headers_in.headers.part.elts == NULL) {
854
855 if (buf->last - buf->pos < NGX_SPDY_NV_NUM_SIZE) {
856 return ngx_http_spdy_state_save(sc, pos, end,
857 ngx_http_spdy_state_headers);
858 }
859
860 sc->headers = ngx_spdy_frame_parse_uint16(buf->pos);
861
862 buf->pos += NGX_SPDY_NV_NUM_SIZE;
863
864 ngx_log_debug1(NGX_LOG_DEBUG_HTTP, r->connection->log, 0,
865 "spdy headers count: %ui", sc->headers);
866
867 if (ngx_list_init(&r->headers_in.headers, r->pool, sc->headers + 3,
868 sizeof(ngx_table_elt_t))
869 != NGX_OK)
870 {
871 ngx_http_spdy_close_stream(sc->stream,
872 NGX_HTTP_INTERNAL_SERVER_ERROR);
873 return ngx_http_spdy_state_headers_error(sc, pos, end);
874 }
875
876 if (ngx_array_init(&r->headers_in.cookies, r->pool, 2,
877 sizeof(ngx_table_elt_t *))
878 != NGX_OK)
879 {
880 ngx_http_spdy_close_stream(sc->stream,
881 NGX_HTTP_INTERNAL_SERVER_ERROR);
882 return ngx_http_spdy_state_headers_error(sc, pos, end);
883 }
884 }
885
886 while (sc->headers) {
887
888 rc = ngx_http_spdy_parse_header(r);
889
890 switch (rc) {
891
892 case NGX_DONE:
893 sc->headers--;
894
895 case NGX_OK:
896 break;
897
898 case NGX_AGAIN:
899
900 if (sc->zstream_in.avail_in) {
901
902 rc = ngx_http_spdy_alloc_large_header_buffer(r);
903
904 if (rc == NGX_DECLINED) {
905 /* TODO logging */
906 ngx_http_finalize_request(r,
907 NGX_HTTP_REQUEST_HEADER_TOO_LARGE);
908 return ngx_http_spdy_state_headers_error(sc, pos, end);
909 }
910
911 if (rc != NGX_OK) {
912 ngx_http_spdy_close_stream(sc->stream,
913 NGX_HTTP_INTERNAL_SERVER_ERROR);
914 return ngx_http_spdy_state_headers_error(sc, pos, end);
915 }
916
Valentin Bartenev3be925b2013-08-15 19:14:58 +0400917 /* null-terminate the last processed header name or value */
918 *buf->pos = '\0';
919
Valentin Bartenev2686cb42013-03-20 10:36:57 +0000920 buf = r->header_in;
921
922 sc->zstream_in.next_out = buf->last;
Valentin Bartenev3be925b2013-08-15 19:14:58 +0400923
924 /* one byte is reserved for null-termination */
Valentin Bartenev2686cb42013-03-20 10:36:57 +0000925 sc->zstream_in.avail_out = buf->end - buf->last - 1;
926
927 z = inflate(&sc->zstream_in, Z_NO_FLUSH);
928
929 if (z != Z_OK) {
930 ngx_log_error(NGX_LOG_INFO, r->connection->log, 0,
931 "spdy inflate() failed: %d", z);
932 ngx_http_spdy_close_stream(sc->stream, 0);
933 return ngx_http_spdy_state_protocol_error(sc);
934 }
935
936 sc->length -= sc->zstream_in.next_in - pos;
937 pos = sc->zstream_in.next_in;
938
939 buf->last = sc->zstream_in.next_out;
940
941 continue;
942 }
943
944 if (complete) {
945 /* TODO: improve error message */
946 ngx_log_debug0(NGX_LOG_DEBUG_HTTP, r->connection->log, 0,
947 "spdy again while last chunk");
948 ngx_http_spdy_close_stream(sc->stream, 0);
949 return ngx_http_spdy_state_protocol_error(sc);
950 }
951
952 return ngx_http_spdy_state_save(sc, pos, end,
953 ngx_http_spdy_state_headers);
954
955 case NGX_HTTP_PARSE_INVALID_REQUEST:
956
957 /* TODO: improve error message */
958 ngx_log_error(NGX_LOG_INFO, r->connection->log, 0,
959 "client sent invalid header line");
960
961 ngx_http_finalize_request(r, NGX_HTTP_BAD_REQUEST);
962
963 return ngx_http_spdy_state_headers_error(sc, pos, end);
964
965 default: /* NGX_HTTP_PARSE_INVALID_HEADER */
966
967 ngx_log_error(NGX_LOG_INFO, r->connection->log, 0,
968 "client sent invalid HEADERS spdy frame");
969 ngx_http_spdy_close_stream(sc->stream, NGX_HTTP_BAD_REQUEST);
970 return ngx_http_spdy_state_protocol_error(sc);
971 }
972
973 /* a header line has been parsed successfully */
974
975 rc = ngx_http_spdy_handle_request_header(r);
976
977 if (rc != NGX_OK) {
978 if (rc == NGX_HTTP_PARSE_INVALID_HEADER) {
979 ngx_log_error(NGX_LOG_INFO, r->connection->log, 0,
980 "client sent invalid HEADERS spdy frame");
981 ngx_http_spdy_close_stream(sc->stream, NGX_HTTP_BAD_REQUEST);
982 return ngx_http_spdy_state_protocol_error(sc);
983 }
984
985 if (rc == NGX_HTTP_PARSE_INVALID_REQUEST) {
986 ngx_http_finalize_request(r, NGX_HTTP_BAD_REQUEST);
987 }
988
989 return ngx_http_spdy_state_headers_error(sc, pos, end);
990 }
991 }
992
993 if (buf->pos != buf->last) {
994 /* TODO: improve error message */
995 ngx_log_debug3(NGX_LOG_DEBUG_HTTP, r->connection->log, 0,
996 "end %ui %p %p", complete, buf->pos, buf->last);
997 ngx_http_spdy_close_stream(sc->stream, NGX_HTTP_BAD_REQUEST);
998 return ngx_http_spdy_state_protocol_error(sc);
999 }
1000
1001 if (!complete) {
1002 return ngx_http_spdy_state_save(sc, pos, end,
1003 ngx_http_spdy_state_headers);
1004 }
1005
Valentin Bartenev3be925b2013-08-15 19:14:58 +04001006 /* null-terminate the last header value */
1007 *buf->pos = '\0';
1008
Valentin Bartenev2686cb42013-03-20 10:36:57 +00001009 ngx_http_spdy_run_request(r);
1010
1011 return ngx_http_spdy_state_complete(sc, pos, end);
1012}
1013
1014
1015static u_char *
1016ngx_http_spdy_state_headers_error(ngx_http_spdy_connection_t *sc, u_char *pos,
1017 u_char *end)
1018{
1019 if (sc->connection->error) {
1020 return ngx_http_spdy_state_internal_error(sc);
1021 }
1022
1023 return ngx_http_spdy_state_headers_skip(sc, pos, end);
1024}
1025
1026
1027static u_char *
1028ngx_http_spdy_state_headers_skip(ngx_http_spdy_connection_t *sc, u_char *pos,
1029 u_char *end)
1030{
1031 int n;
1032 size_t size;
1033 u_char buffer[NGX_SPDY_SKIP_HEADERS_BUFFER_SIZE];
1034
1035 if (sc->length == 0) {
1036 return ngx_http_spdy_state_complete(sc, pos, end);
1037 }
1038
1039 size = end - pos;
1040
1041 if (size == 0) {
1042 return ngx_http_spdy_state_save(sc, pos, end,
1043 ngx_http_spdy_state_headers_skip);
1044 }
1045
1046 sc->zstream_in.next_in = pos;
1047 sc->zstream_in.avail_in = (size < sc->length) ? size : sc->length;
1048
1049 while (sc->zstream_in.avail_in) {
1050 sc->zstream_in.next_out = buffer;
1051 sc->zstream_in.avail_out = NGX_SPDY_SKIP_HEADERS_BUFFER_SIZE;
1052
1053 n = inflate(&sc->zstream_in, Z_NO_FLUSH);
1054
1055 ngx_log_debug1(NGX_LOG_DEBUG_HTTP, sc->connection->log, 0,
1056 "spdy inflate(): %d", n);
1057
1058 if (n != Z_OK) {
1059 /* TODO: logging */
1060 return ngx_http_spdy_state_protocol_error(sc);
1061 }
1062 }
1063
1064 pos = sc->zstream_in.next_in;
1065
1066 if (size < sc->length) {
1067 sc->length -= size;
1068 return ngx_http_spdy_state_save(sc, pos, end,
1069 ngx_http_spdy_state_headers_skip);
1070 }
1071
1072 return ngx_http_spdy_state_complete(sc, pos, end);
1073}
1074
1075
1076static u_char *
1077ngx_http_spdy_state_data(ngx_http_spdy_connection_t *sc, u_char *pos,
1078 u_char *end)
1079{
1080 size_t size;
1081 ssize_t n;
1082 ngx_buf_t *buf;
1083 ngx_int_t rc;
1084 ngx_uint_t complete;
1085 ngx_temp_file_t *tf;
1086 ngx_http_request_t *r;
1087 ngx_http_spdy_stream_t *stream;
1088 ngx_http_request_body_t *rb;
1089 ngx_http_core_loc_conf_t *clcf;
1090
1091 stream = sc->stream;
1092
1093 ngx_log_debug0(NGX_LOG_DEBUG_HTTP, sc->connection->log, 0,
1094 "spdy DATA frame");
1095
1096 if (stream == NULL) {
1097 return ngx_http_spdy_state_skip(sc, pos, end);
1098 }
1099
1100 if (stream->in_closed) {
1101 /* TODO log */
1102 return ngx_http_spdy_state_protocol_error(sc);
1103 }
1104
1105 if (stream->skip_data) {
1106
1107 if (sc->flags & NGX_SPDY_FLAG_FIN) {
1108 stream->in_closed = 1;
1109 }
1110
1111 /* TODO log and accounting */
1112 return ngx_http_spdy_state_skip(sc, pos, end);
1113 }
1114
1115 size = end - pos;
1116
1117 if (size >= sc->length) {
1118 size = sc->length;
1119 complete = 1;
1120
1121 } else {
1122 sc->length -= size;
1123 complete = 0;
1124 }
1125
1126 r = stream->request;
1127
1128 if (r->request_body == NULL
1129 && ngx_http_spdy_init_request_body(r) != NGX_OK)
1130 {
1131 stream->skip_data = NGX_SPDY_DATA_INTERNAL_ERROR;
1132 return ngx_http_spdy_state_skip(sc, pos, end);
1133 }
1134
1135 rb = r->request_body;
1136 tf = rb->temp_file;
1137 buf = rb->buf;
1138
1139 if (size) {
1140 rb->rest += size;
1141
1142 if (r->headers_in.content_length_n != -1
1143 && r->headers_in.content_length_n < rb->rest)
1144 {
1145 /* TODO logging */
1146 stream->skip_data = NGX_SPDY_DATA_ERROR;
1147 goto error;
1148
1149 } else {
1150 clcf = ngx_http_get_module_loc_conf(r, ngx_http_core_module);
1151
1152 if (clcf->client_max_body_size
1153 && clcf->client_max_body_size < rb->rest)
1154 {
1155 ngx_log_error(NGX_LOG_ERR, r->connection->log, 0,
1156 "client intended to send too large chunked "
1157 "body: %O bytes",
1158 rb->rest);
1159
1160 stream->skip_data = NGX_SPDY_DATA_ERROR;
1161 goto error;
1162 }
1163 }
1164
1165 if (tf) {
1166 buf->start = pos;
1167 buf->pos = pos;
1168
1169 pos += size;
1170
1171 buf->end = pos;
1172 buf->last = pos;
1173
1174 n = ngx_write_chain_to_temp_file(tf, rb->bufs);
1175
1176 /* TODO: n == 0 or not complete and level event */
1177
1178 if (n == NGX_ERROR) {
1179 stream->skip_data = NGX_SPDY_DATA_INTERNAL_ERROR;
1180 goto error;
1181 }
1182
1183 tf->offset += n;
1184
1185 } else {
1186 buf->last = ngx_cpymem(buf->last, pos, size);
1187 pos += size;
1188 }
1189
1190 r->request_length += size;
1191 }
1192
1193 if (!complete) {
1194 return ngx_http_spdy_state_save(sc, pos, end,
1195 ngx_http_spdy_state_data);
1196 }
1197
1198 if (sc->flags & NGX_SPDY_FLAG_FIN) {
1199
1200 stream->in_closed = 1;
1201
1202 if (tf) {
1203 ngx_memzero(buf, sizeof(ngx_buf_t));
1204
1205 buf->in_file = 1;
1206 buf->file_last = tf->file.offset;
1207 buf->file = &tf->file;
1208
1209 rb->buf = NULL;
1210 }
1211
1212 if (r->headers_in.content_length_n < 0) {
1213 r->headers_in.content_length_n = rb->rest;
1214 }
1215
1216 if (rb->post_handler) {
Valentin Bartenev6ba03092013-10-01 00:00:57 +04001217 r->read_event_handler = ngx_http_block_reading;
Valentin Bartenev2686cb42013-03-20 10:36:57 +00001218 rb->post_handler(r);
1219 }
1220 }
1221
1222 return ngx_http_spdy_state_complete(sc, pos, end);
1223
1224error:
1225
1226 if (rb->post_handler) {
1227
1228 if (stream->skip_data == NGX_SPDY_DATA_ERROR) {
1229 rc = (r->headers_in.content_length_n == -1)
1230 ? NGX_HTTP_REQUEST_ENTITY_TOO_LARGE
1231 : NGX_HTTP_BAD_REQUEST;
1232
1233 } else {
1234 rc = NGX_HTTP_INTERNAL_SERVER_ERROR;
1235 }
1236
1237 ngx_http_finalize_request(r, rc);
1238 }
1239
1240 return ngx_http_spdy_state_skip(sc, pos, end);
1241}
1242
1243
1244static u_char *
1245ngx_http_spdy_state_rst_stream(ngx_http_spdy_connection_t *sc, u_char *pos,
1246 u_char *end)
1247{
1248 ngx_uint_t sid, status;
1249 ngx_event_t *ev;
1250 ngx_connection_t *fc;
1251 ngx_http_request_t *r;
1252 ngx_http_spdy_stream_t *stream;
1253
1254 if (end - pos < NGX_SPDY_RST_STREAM_SIZE) {
1255 return ngx_http_spdy_state_save(sc, pos, end,
1256 ngx_http_spdy_state_rst_stream);
1257 }
1258
1259 if (sc->length != NGX_SPDY_RST_STREAM_SIZE) {
1260 /* TODO logging */
1261 return ngx_http_spdy_state_protocol_error(sc);
1262 }
1263
1264 sid = ngx_spdy_frame_parse_sid(pos);
1265
1266 pos += NGX_SPDY_SID_SIZE;
1267
1268 status = ngx_spdy_frame_parse_uint32(pos);
1269
1270 pos += sizeof(uint32_t);
1271
1272 ngx_log_debug2(NGX_LOG_DEBUG_HTTP, sc->connection->log, 0,
1273 "spdy RST_STREAM sid:%ui st:%ui", sid, status);
1274
1275
1276 switch (status) {
1277
1278 case NGX_SPDY_PROTOCOL_ERROR:
1279 /* TODO logging */
1280 return ngx_http_spdy_state_protocol_error(sc);
1281
1282 case NGX_SPDY_INVALID_STREAM:
1283 /* TODO */
1284 break;
1285
1286 case NGX_SPDY_REFUSED_STREAM:
1287 /* TODO */
1288 break;
1289
1290 case NGX_SPDY_UNSUPPORTED_VERSION:
1291 /* TODO logging */
1292 return ngx_http_spdy_state_protocol_error(sc);
1293
1294 case NGX_SPDY_CANCEL:
1295 case NGX_SPDY_INTERNAL_ERROR:
1296 stream = ngx_http_spdy_get_stream_by_id(sc, sid);
1297 if (stream == NULL) {
1298 /* TODO false cancel */
1299 break;
1300 }
1301
1302 stream->in_closed = 1;
1303 stream->out_closed = 1;
1304
1305 r = stream->request;
1306
1307 fc = r->connection;
1308 fc->error = 1;
1309
1310 ev = fc->read;
1311 ev->handler(ev);
1312
1313 break;
1314
1315 case NGX_SPDY_FLOW_CONTROL_ERROR:
1316 /* TODO logging */
1317 return ngx_http_spdy_state_protocol_error(sc);
1318
1319 default:
1320 /* TODO */
1321 return ngx_http_spdy_state_protocol_error(sc);
1322 }
1323
1324 return ngx_http_spdy_state_complete(sc, pos, end);
1325}
1326
1327
1328static u_char *
1329ngx_http_spdy_state_ping(ngx_http_spdy_connection_t *sc, u_char *pos,
1330 u_char *end)
1331{
1332 u_char *p;
1333 ngx_buf_t *buf;
1334 ngx_http_spdy_out_frame_t *frame;
1335
1336 if (end - pos < NGX_SPDY_PING_SIZE) {
1337 return ngx_http_spdy_state_save(sc, pos, end,
1338 ngx_http_spdy_state_ping);
1339 }
1340
1341 if (sc->length != NGX_SPDY_PING_SIZE) {
1342 /* TODO logging */
1343 return ngx_http_spdy_state_protocol_error(sc);
1344 }
1345
1346 ngx_log_debug0(NGX_LOG_DEBUG_HTTP, sc->connection->log, 0,
1347 "spdy PING frame");
1348
1349 frame = ngx_http_spdy_get_ctl_frame(sc, NGX_SPDY_PING_SIZE,
1350 NGX_SPDY_HIGHEST_PRIORITY);
1351 if (frame == NULL) {
1352 return ngx_http_spdy_state_internal_error(sc);
1353 }
1354
1355 buf = frame->first->buf;
1356
1357 p = buf->pos;
1358
1359 p = ngx_spdy_frame_write_head(p, NGX_SPDY_PING);
1360 p = ngx_spdy_frame_write_flags_and_len(p, 0, NGX_SPDY_PING_SIZE);
1361
1362 p = ngx_cpymem(p, pos, NGX_SPDY_PING_SIZE);
1363
1364 buf->last = p;
1365
1366 ngx_http_spdy_queue_frame(sc, frame);
1367
1368 pos += NGX_SPDY_PING_SIZE;
1369
1370 return ngx_http_spdy_state_complete(sc, pos, end);
1371}
1372
1373
1374static u_char *
1375ngx_http_spdy_state_skip(ngx_http_spdy_connection_t *sc, u_char *pos,
1376 u_char *end)
1377{
1378 size_t size;
1379
1380 size = end - pos;
1381
1382 if (size < sc->length) {
1383 sc->length -= size;
1384 return ngx_http_spdy_state_save(sc, end, end,
1385 ngx_http_spdy_state_skip);
1386 }
1387
1388 return ngx_http_spdy_state_complete(sc, pos + sc->length, end);
1389}
1390
1391
1392static u_char *
1393ngx_http_spdy_state_settings(ngx_http_spdy_connection_t *sc, u_char *pos,
1394 u_char *end)
1395{
1396 ngx_uint_t v;
1397 ngx_http_spdy_srv_conf_t *sscf;
1398
1399 if (sc->headers == 0) {
1400
1401 if (end - pos < NGX_SPDY_SETTINGS_NUM_SIZE) {
1402 return ngx_http_spdy_state_save(sc, pos, end,
1403 ngx_http_spdy_state_settings);
1404 }
1405
1406 sc->headers = ngx_spdy_frame_parse_uint32(pos);
1407
1408 pos += NGX_SPDY_SETTINGS_NUM_SIZE;
1409 sc->length -= NGX_SPDY_SETTINGS_NUM_SIZE;
1410
1411 if (sc->length < sc->headers * NGX_SPDY_SETTINGS_PAIR_SIZE) {
1412 /* TODO logging */
1413 return ngx_http_spdy_state_protocol_error(sc);
1414 }
1415
1416 ngx_log_debug1(NGX_LOG_DEBUG_HTTP, sc->connection->log, 0,
1417 "spdy SETTINGS frame consists of %ui entries",
1418 sc->headers);
1419 }
1420
1421 while (sc->headers) {
1422 if (end - pos < NGX_SPDY_SETTINGS_PAIR_SIZE) {
1423 return ngx_http_spdy_state_save(sc, pos, end,
1424 ngx_http_spdy_state_settings);
1425 }
1426
1427 sc->headers--;
1428
1429 if (pos[0] != NGX_SPDY_SETTINGS_MAX_STREAMS) {
1430 pos += NGX_SPDY_SETTINGS_PAIR_SIZE;
1431 sc->length -= NGX_SPDY_SETTINGS_PAIR_SIZE;
1432 continue;
1433 }
1434
1435 v = ngx_spdy_frame_parse_uint32(pos + NGX_SPDY_SETTINGS_IDF_SIZE);
1436
1437 sscf = ngx_http_get_module_srv_conf(sc->http_connection->conf_ctx,
1438 ngx_http_spdy_module);
1439
1440 if (v != sscf->concurrent_streams) {
1441 ngx_http_spdy_send_settings(sc);
1442 }
1443
1444 return ngx_http_spdy_state_skip(sc, pos, end);
1445 }
1446
1447 ngx_http_spdy_send_settings(sc);
1448
1449 return ngx_http_spdy_state_complete(sc, pos, end);
1450}
1451
1452
1453static u_char *
1454ngx_http_spdy_state_noop(ngx_http_spdy_connection_t *sc, u_char *pos,
1455 u_char *end)
1456{
1457 if (sc->length) {
1458 /* TODO logging */
1459 return ngx_http_spdy_state_protocol_error(sc);
1460 }
1461
1462 return ngx_http_spdy_state_complete(sc, pos, end);
1463}
1464
1465
1466static u_char *
1467ngx_http_spdy_state_complete(ngx_http_spdy_connection_t *sc, u_char *pos,
1468 u_char *end)
1469{
1470 sc->handler = ngx_http_spdy_state_head;
1471 return pos;
1472}
1473
1474
1475static u_char *
1476ngx_http_spdy_state_save(ngx_http_spdy_connection_t *sc,
1477 u_char *pos, u_char *end, ngx_http_spdy_handler_pt handler)
1478{
1479#if (NGX_DEBUG)
1480 if (end - pos > NGX_SPDY_STATE_BUFFER_SIZE) {
1481 ngx_log_error(NGX_LOG_ALERT, sc->connection->log, 0,
1482 "spdy state buffer overflow: "
1483 "%i bytes required", end - pos);
1484 return ngx_http_spdy_state_internal_error(sc);
1485 }
1486#endif
1487
1488 ngx_memcpy(sc->buffer, pos, NGX_SPDY_STATE_BUFFER_SIZE);
1489
1490 sc->buffer_used = end - pos;
1491 sc->handler = handler;
1492 sc->waiting = 1;
1493
1494 return end;
1495}
1496
1497
1498static u_char *
1499ngx_http_spdy_state_protocol_error(ngx_http_spdy_connection_t *sc)
1500{
1501 ngx_log_debug0(NGX_LOG_DEBUG_HTTP, sc->connection->log, 0,
1502 "spdy state protocol error");
1503
1504 /* TODO */
1505 ngx_http_spdy_finalize_connection(sc, NGX_HTTP_CLIENT_CLOSED_REQUEST);
1506 return NULL;
1507}
1508
1509
1510static u_char *
1511ngx_http_spdy_state_internal_error(ngx_http_spdy_connection_t *sc)
1512{
1513 ngx_log_debug0(NGX_LOG_DEBUG_HTTP, sc->connection->log, 0,
1514 "spdy state internal error");
1515
1516 /* TODO */
1517 ngx_http_spdy_finalize_connection(sc, NGX_HTTP_INTERNAL_SERVER_ERROR);
1518 return NULL;
1519}
1520
1521
1522static ngx_int_t
1523ngx_http_spdy_send_rst_stream(ngx_http_spdy_connection_t *sc, ngx_uint_t sid,
1524 ngx_uint_t status, ngx_uint_t priority)
1525{
1526 u_char *p;
1527 ngx_buf_t *buf;
1528 ngx_http_spdy_out_frame_t *frame;
1529
1530 if (sc->connection->error) {
1531 return NGX_OK;
1532 }
1533
1534 ngx_log_debug2(NGX_LOG_DEBUG_HTTP, sc->connection->log, 0,
1535 "spdy write RST_STREAM sid:%ui st:%ui", sid, status);
1536
1537 frame = ngx_http_spdy_get_ctl_frame(sc, NGX_SPDY_RST_STREAM_SIZE,
1538 priority);
1539 if (frame == NULL) {
1540 return NGX_ERROR;
1541 }
1542
1543 buf = frame->first->buf;
1544
1545 p = buf->pos;
1546
1547 p = ngx_spdy_frame_write_head(p, NGX_SPDY_RST_STREAM);
1548 p = ngx_spdy_frame_write_flags_and_len(p, 0, NGX_SPDY_RST_STREAM_SIZE);
1549
1550 p = ngx_spdy_frame_write_sid(p, sid);
1551 p = ngx_spdy_frame_aligned_write_uint32(p, status);
1552
1553 buf->last = p;
1554
1555 ngx_http_spdy_queue_frame(sc, frame);
1556
1557 return NGX_OK;
1558}
1559
1560
1561#if 0
1562static ngx_int_t
1563ngx_http_spdy_send_goaway(ngx_http_spdy_connection_t *sc)
1564{
1565 u_char *p;
1566 ngx_buf_t *buf;
1567 ngx_http_spdy_out_frame_t *frame;
1568
1569 ngx_log_debug1(NGX_LOG_DEBUG_HTTP, sc->connection->log, 0,
1570 "spdy create GOAWAY sid:%ui", sc->last_sid);
1571
1572 frame = ngx_http_spdy_get_ctl_frame(sc, NGX_SPDY_GOAWAY_SIZE,
1573 NGX_SPDY_HIGHEST_PRIORITY);
1574 if (frame == NULL) {
1575 return NGX_ERROR;
1576 }
1577
1578 buf = frame->first->buf;
1579
1580 p = buf->pos;
1581
1582 p = ngx_spdy_frame_write_head(p, NGX_SPDY_GOAWAY);
1583 p = ngx_spdy_frame_write_flags_and_len(p, 0, NGX_SPDY_GOAWAY_SIZE);
1584
1585 p = ngx_spdy_frame_write_sid(p, sc->last_sid);
1586
1587 buf->last = p;
1588
1589 ngx_http_spdy_queue_frame(sc, frame);
1590
1591 return NGX_OK;
1592}
1593#endif
1594
1595
1596static ngx_int_t
1597ngx_http_spdy_send_settings(ngx_http_spdy_connection_t *sc)
1598{
1599 u_char *p;
1600 ngx_buf_t *buf;
1601 ngx_pool_t *pool;
1602 ngx_chain_t *cl;
1603 ngx_http_spdy_srv_conf_t *sscf;
1604 ngx_http_spdy_out_frame_t *frame;
1605
1606 ngx_log_debug0(NGX_LOG_DEBUG_HTTP, sc->connection->log, 0,
1607 "spdy create SETTINGS frame");
1608
1609 pool = sc->connection->pool;
1610
1611 frame = ngx_palloc(pool, sizeof(ngx_http_spdy_out_frame_t));
1612 if (frame == NULL) {
1613 return NGX_ERROR;
1614 }
1615
1616 cl = ngx_alloc_chain_link(pool);
1617 if (cl == NULL) {
1618 return NGX_ERROR;
1619 }
1620
1621 buf = ngx_create_temp_buf(pool, NGX_SPDY_FRAME_HEADER_SIZE
1622 + NGX_SPDY_SETTINGS_NUM_SIZE
1623 + NGX_SPDY_SETTINGS_PAIR_SIZE);
1624 if (buf == NULL) {
1625 return NGX_ERROR;
1626 }
1627
1628 buf->last_buf = 1;
1629
1630 cl->buf = buf;
1631 cl->next = NULL;
1632
1633 frame->first = cl;
1634 frame->last = cl;
1635 frame->handler = ngx_http_spdy_settings_frame_handler;
1636#if (NGX_DEBUG)
1637 frame->stream = NULL;
1638 frame->size = NGX_SPDY_FRAME_HEADER_SIZE
1639 + NGX_SPDY_SETTINGS_NUM_SIZE
1640 + NGX_SPDY_SETTINGS_PAIR_SIZE;
1641#endif
1642 frame->priority = NGX_SPDY_HIGHEST_PRIORITY;
1643 frame->blocked = 0;
1644
1645 p = buf->pos;
1646
1647 p = ngx_spdy_frame_write_head(p, NGX_SPDY_SETTINGS);
1648 p = ngx_spdy_frame_write_flags_and_len(p, NGX_SPDY_FLAG_CLEAR_SETTINGS,
1649 NGX_SPDY_SETTINGS_NUM_SIZE
1650 + NGX_SPDY_SETTINGS_PAIR_SIZE);
1651
1652 p = ngx_spdy_frame_aligned_write_uint32(p, 1);
1653 p = ngx_spdy_frame_aligned_write_uint32(p,
1654 NGX_SPDY_SETTINGS_MAX_STREAMS << 24
1655 | NGX_SPDY_SETTINGS_FLAG_PERSIST);
1656
1657 sscf = ngx_http_get_module_srv_conf(sc->http_connection->conf_ctx,
1658 ngx_http_spdy_module);
1659
1660 p = ngx_spdy_frame_aligned_write_uint32(p, sscf->concurrent_streams);
1661
1662 buf->last = p;
1663
1664 ngx_http_spdy_queue_frame(sc, frame);
1665
1666 return NGX_OK;
1667}
1668
1669
1670ngx_int_t
1671ngx_http_spdy_settings_frame_handler(ngx_http_spdy_connection_t *sc,
1672 ngx_http_spdy_out_frame_t *frame)
1673{
1674 ngx_buf_t *buf;
1675
1676 buf = frame->first->buf;
1677
1678 if (buf->pos != buf->last) {
1679 return NGX_AGAIN;
1680 }
1681
1682 ngx_free_chain(sc->pool, frame->first);
1683
1684 return NGX_OK;
1685}
1686
1687
1688static ngx_http_spdy_out_frame_t *
1689ngx_http_spdy_get_ctl_frame(ngx_http_spdy_connection_t *sc, size_t size,
1690 ngx_uint_t priority)
1691{
1692 ngx_chain_t *cl;
1693 ngx_http_spdy_out_frame_t *frame;
1694
1695 frame = sc->free_ctl_frames;
1696
1697 if (frame) {
1698 sc->free_ctl_frames = frame->free;
1699
1700 cl = frame->first;
1701 cl->buf->pos = cl->buf->start;
1702
1703 } else {
1704 frame = ngx_palloc(sc->pool, sizeof(ngx_http_spdy_out_frame_t));
1705 if (frame == NULL) {
1706 return NULL;
1707 }
1708
1709 cl = ngx_alloc_chain_link(sc->pool);
1710 if (cl == NULL) {
1711 return NULL;
1712 }
1713
1714 cl->buf = ngx_create_temp_buf(sc->pool,
1715 NGX_SPDY_CTL_FRAME_BUFFER_SIZE);
1716 if (cl->buf == NULL) {
1717 return NULL;
1718 }
1719
1720 cl->buf->last_buf = 1;
1721
1722 frame->first = cl;
1723 frame->last = cl;
1724 frame->handler = ngx_http_spdy_ctl_frame_handler;
1725 }
1726
1727 frame->free = NULL;
1728
1729#if (NGX_DEBUG)
1730 if (size > NGX_SPDY_CTL_FRAME_BUFFER_SIZE - NGX_SPDY_FRAME_HEADER_SIZE) {
1731 ngx_log_error(NGX_LOG_ALERT, sc->pool->log, 0,
1732 "requested control frame is too big: %z", size);
1733 return NULL;
1734 }
1735
1736 frame->stream = NULL;
1737 frame->size = size;
1738#endif
1739
1740 frame->priority = priority;
1741 frame->blocked = 0;
1742
1743 return frame;
1744}
1745
1746
1747static ngx_int_t
1748ngx_http_spdy_ctl_frame_handler(ngx_http_spdy_connection_t *sc,
1749 ngx_http_spdy_out_frame_t *frame)
1750{
1751 ngx_buf_t *buf;
1752
1753 buf = frame->first->buf;
1754
1755 if (buf->pos != buf->last) {
1756 return NGX_AGAIN;
1757 }
1758
1759 frame->free = sc->free_ctl_frames;
1760 sc->free_ctl_frames = frame;
1761
1762 return NGX_OK;
1763}
1764
1765
1766static ngx_http_spdy_stream_t *
1767ngx_http_spdy_create_stream(ngx_http_spdy_connection_t *sc, ngx_uint_t id,
1768 ngx_uint_t priority)
1769{
1770 ngx_log_t *log;
1771 ngx_uint_t index;
1772 ngx_event_t *rev, *wev;
1773 ngx_connection_t *fc;
1774 ngx_http_log_ctx_t *ctx;
1775 ngx_http_request_t *r;
1776 ngx_http_spdy_stream_t *stream;
1777 ngx_http_core_srv_conf_t *cscf;
1778 ngx_http_spdy_srv_conf_t *sscf;
1779
1780 fc = sc->free_fake_connections;
1781
1782 if (fc) {
1783 sc->free_fake_connections = fc->data;
1784
1785 rev = fc->read;
1786 wev = fc->write;
1787 log = fc->log;
1788 ctx = log->data;
1789
1790 } else {
1791 fc = ngx_palloc(sc->pool, sizeof(ngx_connection_t));
1792 if (fc == NULL) {
1793 return NULL;
1794 }
1795
1796 rev = ngx_palloc(sc->pool, sizeof(ngx_event_t));
1797 if (rev == NULL) {
1798 return NULL;
1799 }
1800
1801 wev = ngx_palloc(sc->pool, sizeof(ngx_event_t));
1802 if (wev == NULL) {
1803 return NULL;
1804 }
1805
1806 log = ngx_palloc(sc->pool, sizeof(ngx_log_t));
1807 if (log == NULL) {
1808 return NULL;
1809 }
1810
1811 ctx = ngx_palloc(sc->pool, sizeof(ngx_http_log_ctx_t));
1812 if (ctx == NULL) {
1813 return NULL;
1814 }
1815
1816 ctx->connection = fc;
1817 ctx->request = NULL;
1818 }
1819
1820 ngx_memcpy(log, sc->connection->log, sizeof(ngx_log_t));
1821
1822 log->data = ctx;
1823
1824 ngx_memzero(rev, sizeof(ngx_event_t));
1825
1826 rev->data = fc;
1827 rev->ready = 1;
1828 rev->handler = ngx_http_empty_handler;
1829 rev->log = log;
1830
1831 ngx_memcpy(wev, rev, sizeof(ngx_event_t));
1832
1833 wev->write = 1;
1834
1835 ngx_memcpy(fc, sc->connection, sizeof(ngx_connection_t));
1836
1837 fc->data = sc->http_connection;
1838 fc->read = rev;
1839 fc->write = wev;
1840 fc->sent = 0;
1841 fc->log = log;
1842 fc->buffered = 0;
1843 fc->sndlowat = 1;
Valentin Bartenev670d4282013-04-23 10:15:49 +00001844 fc->tcp_nodelay = NGX_TCP_NODELAY_DISABLED;
Valentin Bartenev2686cb42013-03-20 10:36:57 +00001845
1846 r = ngx_http_create_request(fc);
1847 if (r == NULL) {
1848 return NULL;
1849 }
1850
1851 r->valid_location = 1;
1852
1853 fc->data = r;
1854 sc->connection->requests++;
1855
1856 cscf = ngx_http_get_module_srv_conf(r, ngx_http_core_module);
1857
1858 r->header_in = ngx_create_temp_buf(r->pool,
1859 cscf->client_header_buffer_size);
1860 if (r->header_in == NULL) {
1861 ngx_http_free_request(r, NGX_HTTP_INTERNAL_SERVER_ERROR);
1862 return NULL;
1863 }
1864
1865 r->headers_in.connection_type = NGX_HTTP_CONNECTION_CLOSE;
1866
1867 stream = ngx_pcalloc(r->pool, sizeof(ngx_http_spdy_stream_t));
1868 if (stream == NULL) {
1869 ngx_http_free_request(r, NGX_HTTP_INTERNAL_SERVER_ERROR);
1870 return NULL;
1871 }
1872
1873 r->spdy_stream = stream;
1874
1875 stream->id = id;
1876 stream->request = r;
1877 stream->connection = sc;
1878 stream->priority = priority;
1879
1880 sscf = ngx_http_get_module_srv_conf(r, ngx_http_spdy_module);
1881
1882 index = ngx_http_spdy_stream_index(sscf, id);
1883
1884 stream->index = sc->streams_index[index];
1885 sc->streams_index[index] = stream;
1886
1887 sc->processing++;
1888
1889 return stream;
1890}
1891
1892
1893static ngx_http_spdy_stream_t *
1894ngx_http_spdy_get_stream_by_id(ngx_http_spdy_connection_t *sc,
1895 ngx_uint_t sid)
1896{
1897 ngx_http_spdy_stream_t *stream;
1898 ngx_http_spdy_srv_conf_t *sscf;
1899
1900 sscf = ngx_http_get_module_srv_conf(sc->http_connection->conf_ctx,
1901 ngx_http_spdy_module);
1902
1903 stream = sc->streams_index[ngx_http_spdy_stream_index(sscf, sid)];
1904
1905 while (stream) {
1906 if (stream->id == sid) {
1907 return stream;
1908 }
1909
1910 stream = stream->index;
1911 }
1912
1913 return NULL;
1914}
1915
1916
1917static ngx_int_t
1918ngx_http_spdy_parse_header(ngx_http_request_t *r)
1919{
1920 u_char *p, *end, ch;
1921 ngx_uint_t len, hash;
1922 ngx_http_core_srv_conf_t *cscf;
1923
1924 enum {
1925 sw_name_len = 0,
1926 sw_name,
1927 sw_value_len,
1928 sw_value
1929 } state;
1930
1931 state = r->state;
1932
1933 p = r->header_in->pos;
1934 end = r->header_in->last;
1935
1936 switch (state) {
1937
1938 case sw_name_len:
1939
1940 if (end - p < NGX_SPDY_NV_NLEN_SIZE) {
1941 return NGX_AGAIN;
1942 }
1943
1944 len = ngx_spdy_frame_parse_uint16(p);
1945
1946 if (!len) {
1947 return NGX_HTTP_PARSE_INVALID_HEADER;
1948 }
1949
Valentin Bartenev3be925b2013-08-15 19:14:58 +04001950 /* null-terminate the previous header value */
1951 *p = '\0';
1952
Valentin Bartenev2686cb42013-03-20 10:36:57 +00001953 p += NGX_SPDY_NV_NLEN_SIZE;
1954
1955 r->header_name_end = p + len;
1956 r->lowcase_index = len;
1957 r->invalid_header = 0;
1958
1959 state = sw_name;
1960
1961 /* fall through */
1962
1963 case sw_name:
1964
1965 if (r->header_name_end > end) {
1966 break;
1967 }
1968
1969 cscf = ngx_http_get_module_srv_conf(r, ngx_http_core_module);
1970
1971 r->header_name_start = p;
1972
1973 hash = 0;
1974
1975 for ( /* void */ ; p != r->header_name_end; p++) {
1976
1977 ch = *p;
1978
1979 hash = ngx_hash(hash, ch);
1980
1981 if ((ch >= 'a' && ch <= 'z')
1982 || (ch == '-')
1983 || (ch >= '0' && ch <= '9')
1984 || (ch == '_' && cscf->underscores_in_headers))
1985 {
1986 continue;
1987 }
1988
1989 switch (ch) {
1990 case '\0':
1991 case LF:
1992 case CR:
1993 case ':':
1994 return NGX_HTTP_PARSE_INVALID_REQUEST;
1995 }
1996
1997 if (ch >= 'A' && ch <= 'Z') {
1998 return NGX_HTTP_PARSE_INVALID_HEADER;
1999 }
2000
2001 r->invalid_header = 1;
2002 }
2003
2004 r->header_hash = hash;
2005
2006 state = sw_value_len;
2007
2008 /* fall through */
2009
2010 case sw_value_len:
2011
2012 if (end - p < NGX_SPDY_NV_VLEN_SIZE) {
2013 break;
2014 }
2015
2016 len = ngx_spdy_frame_parse_uint16(p);
2017
Valentin Bartenev3be925b2013-08-15 19:14:58 +04002018 /* null-terminate header name */
2019 *p = '\0';
2020
Valentin Bartenev2686cb42013-03-20 10:36:57 +00002021 p += NGX_SPDY_NV_VLEN_SIZE;
2022
2023 r->header_end = p + len;
2024
2025 state = sw_value;
2026
2027 /* fall through */
2028
2029 case sw_value:
2030
2031 if (r->header_end > end) {
2032 break;
2033 }
2034
2035 r->header_start = p;
2036
2037 for ( /* void */ ; p != r->header_end; p++) {
2038
2039 ch = *p;
2040
2041 if (ch == '\0') {
2042
2043 if (p == r->header_start) {
2044 return NGX_ERROR;
2045 }
2046
2047 r->header_size = p - r->header_start;
2048 r->header_in->pos = p + 1;
2049
2050 return NGX_OK;
2051 }
2052
2053 if (ch == CR || ch == LF) {
2054 return NGX_HTTP_PARSE_INVALID_HEADER;
2055 }
2056 }
2057
2058 r->header_size = p - r->header_start;
2059 r->header_in->pos = p;
2060
2061 r->state = 0;
2062
2063 return NGX_DONE;
2064 }
2065
2066 r->header_in->pos = p;
2067 r->state = state;
2068
2069 return NGX_AGAIN;
2070}
2071
2072
2073static ngx_int_t
2074ngx_http_spdy_alloc_large_header_buffer(ngx_http_request_t *r)
2075{
2076 u_char *old, *new;
2077 size_t rest;
2078 ngx_buf_t *buf;
2079 ngx_http_spdy_stream_t *stream;
2080 ngx_http_core_srv_conf_t *cscf;
2081
2082 ngx_log_debug0(NGX_LOG_DEBUG_HTTP, r->connection->log, 0,
2083 "spdy alloc large header buffer");
2084
2085 stream = r->spdy_stream;
2086
2087 cscf = ngx_http_get_module_srv_conf(r, ngx_http_core_module);
2088
2089 if (stream->header_buffers
2090 == (ngx_uint_t) cscf->large_client_header_buffers.num)
2091 {
2092 return NGX_DECLINED;
2093 }
2094
2095 rest = r->header_in->last - r->header_in->pos;
2096
2097 if (rest >= cscf->large_client_header_buffers.size) {
2098 return NGX_DECLINED;
2099 }
2100
2101 buf = ngx_create_temp_buf(r->pool, cscf->large_client_header_buffers.size);
2102 if (buf == NULL) {
2103 return NGX_ERROR;
2104 }
2105
2106 ngx_log_debug2(NGX_LOG_DEBUG_HTTP, r->connection->log, 0,
2107 "spdy large header alloc: %p %uz",
2108 buf->pos, buf->end - buf->last);
2109
2110 old = r->header_in->pos;
2111 new = buf->pos;
2112
2113 if (rest) {
2114 buf->last = ngx_cpymem(new, old, rest);
2115 }
2116
2117 if (r->header_name_end > old) {
2118 r->header_name_end = new + (r->header_name_end - old);
2119
2120 } else if (r->header_end > old) {
2121 r->header_end = new + (r->header_end - old);
2122 }
2123
2124 r->header_in = buf;
2125
2126 stream->header_buffers++;
2127
2128 return NGX_OK;
2129}
2130
2131
2132static ngx_int_t
2133ngx_http_spdy_handle_request_header(ngx_http_request_t *r)
2134{
2135 ngx_uint_t i;
2136 ngx_table_elt_t *h;
2137 ngx_http_core_srv_conf_t *cscf;
2138 ngx_http_spdy_request_header_t *sh;
2139
2140 if (r->invalid_header) {
2141 cscf = ngx_http_get_module_srv_conf(r, ngx_http_core_module);
2142
2143 if (cscf->ignore_invalid_headers) {
2144 ngx_log_error(NGX_LOG_INFO, r->connection->log, 0,
2145 "client sent invalid header: \"%*s\"",
2146 r->header_end - r->header_name_start,
2147 r->header_name_start);
2148 return NGX_OK;
2149 }
2150
2151 } else {
2152 for (i = 0; i < NGX_SPDY_REQUEST_HEADERS; i++) {
2153 sh = &ngx_http_spdy_request_headers[i];
2154
2155 if (sh->hash != r->header_hash
2156 || sh->len != r->lowcase_index
2157 || ngx_strncmp(sh->header, r->header_name_start,
2158 r->lowcase_index)
2159 != 0)
2160 {
2161 continue;
2162 }
2163
2164 return sh->handler(r);
2165 }
2166 }
2167
2168 h = ngx_list_push(&r->headers_in.headers);
2169 if (h == NULL) {
2170 ngx_http_spdy_close_stream(r->spdy_stream,
2171 NGX_HTTP_INTERNAL_SERVER_ERROR);
2172 return NGX_ERROR;
2173 }
2174
2175 h->hash = r->header_hash;
2176
2177 h->key.len = r->lowcase_index;
2178 h->key.data = r->header_name_start;
Valentin Bartenev2686cb42013-03-20 10:36:57 +00002179
2180 h->value.len = r->header_size;
2181 h->value.data = r->header_start;
Valentin Bartenev2686cb42013-03-20 10:36:57 +00002182
2183 h->lowcase_key = h->key.data;
2184
2185 return NGX_OK;
2186}
2187
2188
2189void
Sergey Kandaurov3be6cc92013-05-23 15:47:58 +04002190ngx_http_spdy_request_headers_init(void)
Valentin Bartenev2686cb42013-03-20 10:36:57 +00002191{
2192 ngx_uint_t i;
2193 ngx_http_spdy_request_header_t *h;
2194
2195 for (i = 0; i < NGX_SPDY_REQUEST_HEADERS; i++) {
2196 h = &ngx_http_spdy_request_headers[i];
2197 h->hash = ngx_hash_key(h->header, h->len);
2198 }
2199}
2200
2201
2202static ngx_int_t
2203ngx_http_spdy_parse_method(ngx_http_request_t *r)
2204{
2205 size_t k, len;
2206 ngx_uint_t n;
2207 const u_char *p, *m;
2208
2209 /*
2210 * This array takes less than 256 sequential bytes,
2211 * and if typical CPU cache line size is 64 bytes,
2212 * it is prefetched for 4 load operations.
2213 */
2214 static const struct {
2215 u_char len;
2216 const u_char method[11];
2217 uint32_t value;
2218 } tests[] = {
2219 { 3, "GET", NGX_HTTP_GET },
2220 { 4, "POST", NGX_HTTP_POST },
2221 { 4, "HEAD", NGX_HTTP_HEAD },
2222 { 7, "OPTIONS", NGX_HTTP_OPTIONS },
2223 { 8, "PROPFIND", NGX_HTTP_PROPFIND },
2224 { 3, "PUT", NGX_HTTP_PUT },
2225 { 5, "MKCOL", NGX_HTTP_MKCOL },
2226 { 6, "DELETE", NGX_HTTP_DELETE },
2227 { 4, "COPY", NGX_HTTP_COPY },
2228 { 4, "MOVE", NGX_HTTP_MOVE },
2229 { 9, "PROPPATCH", NGX_HTTP_PROPPATCH },
2230 { 4, "LOCK", NGX_HTTP_LOCK },
2231 { 6, "UNLOCK", NGX_HTTP_UNLOCK },
2232 { 5, "PATCH", NGX_HTTP_PATCH },
2233 { 5, "TRACE", NGX_HTTP_TRACE }
2234 }, *test;
2235
2236 if (r->method_name.len) {
2237 return NGX_HTTP_PARSE_INVALID_HEADER;
2238 }
2239
2240 len = r->header_size;
2241
2242 r->method_name.len = len;
2243 r->method_name.data = r->header_start;
2244
2245 test = tests;
2246 n = sizeof(tests) / sizeof(tests[0]);
2247
2248 do {
2249 if (len == test->len) {
2250 p = r->method_name.data;
2251 m = test->method;
2252 k = len;
2253
2254 do {
2255 if (*p++ != *m++) {
2256 goto next;
2257 }
2258 } while (--k);
2259
2260 r->method = test->value;
2261 return NGX_OK;
2262 }
2263
2264 next:
2265 test++;
2266
2267 } while (--n);
2268
2269 p = r->method_name.data;
2270
2271 do {
2272 if ((*p < 'A' || *p > 'Z') && *p != '_') {
2273 ngx_log_error(NGX_LOG_INFO, r->connection->log, 0,
2274 "client sent invalid method");
2275 return NGX_HTTP_PARSE_INVALID_REQUEST;
2276 }
2277
2278 p++;
2279
2280 } while (--len);
2281
2282 return NGX_OK;
2283}
2284
2285
2286static ngx_int_t
2287ngx_http_spdy_parse_scheme(ngx_http_request_t *r)
2288{
2289 if (r->schema_start) {
2290 return NGX_HTTP_PARSE_INVALID_HEADER;
2291 }
2292
2293 r->schema_start = r->header_start;
2294 r->schema_end = r->header_end;
2295
2296 return NGX_OK;
2297}
2298
2299
2300static ngx_int_t
2301ngx_http_spdy_parse_url(ngx_http_request_t *r)
2302{
2303 if (r->unparsed_uri.len) {
2304 return NGX_HTTP_PARSE_INVALID_HEADER;
2305 }
2306
2307 r->uri_start = r->header_start;
2308 r->uri_end = r->header_end;
2309
2310 if (ngx_http_parse_uri(r) != NGX_OK) {
2311 return NGX_HTTP_PARSE_INVALID_REQUEST;
2312 }
2313
2314 if (ngx_http_process_request_uri(r) != NGX_OK) {
2315 return NGX_ERROR;
2316 }
2317
2318 return NGX_OK;
2319}
2320
2321
2322static ngx_int_t
2323ngx_http_spdy_parse_version(ngx_http_request_t *r)
2324{
2325 u_char *p, ch;
2326
2327 if (r->http_protocol.len) {
2328 return NGX_HTTP_PARSE_INVALID_HEADER;
2329 }
2330
2331 p = r->header_start;
2332
2333 if (r->header_size < 8 || !(ngx_str5cmp(p, 'H', 'T', 'T', 'P', '/'))) {
2334 return NGX_HTTP_PARSE_INVALID_REQUEST;
2335 }
2336
2337 ch = *(p + 5);
2338
2339 if (ch < '1' || ch > '9') {
2340 return NGX_HTTP_PARSE_INVALID_REQUEST;
2341 }
2342
2343 r->http_major = ch - '0';
2344
2345 for (p += 6; p != r->header_end - 2; p++) {
2346
2347 ch = *p;
2348
2349 if (ch < '0' || ch > '9') {
2350 return NGX_HTTP_PARSE_INVALID_REQUEST;
2351 }
2352
2353 r->http_major = r->http_major * 10 + ch - '0';
2354 }
2355
2356 if (*p != '.') {
2357 return NGX_HTTP_PARSE_INVALID_REQUEST;
2358 }
2359
2360 ch = *(p + 1);
2361
2362 if (ch < '0' || ch > '9') {
2363 return NGX_HTTP_PARSE_INVALID_REQUEST;
2364 }
2365
2366 r->http_minor = ch - '0';
2367
2368 for (p += 2; p != r->header_end; p++) {
2369
2370 ch = *p;
2371
2372 if (ch < '0' || ch > '9') {
2373 return NGX_HTTP_PARSE_INVALID_REQUEST;
2374 }
2375
2376 r->http_minor = r->http_minor * 10 + ch - '0';
2377 }
2378
2379 r->http_protocol.len = r->header_size;
2380 r->http_protocol.data = r->header_start;
2381 r->http_version = r->http_major * 1000 + r->http_minor;
2382
2383 return NGX_OK;
2384}
2385
2386
2387static ngx_int_t
2388ngx_http_spdy_construct_request_line(ngx_http_request_t *r)
2389{
2390 u_char *p;
2391
2392 if (r->method_name.len == 0
2393 || r->unparsed_uri.len == 0
2394 || r->http_protocol.len == 0)
2395 {
2396 ngx_http_finalize_request(r, NGX_HTTP_BAD_REQUEST);
2397 return NGX_ERROR;
2398 }
2399
2400 r->request_line.len = r->method_name.len + 1
2401 + r->unparsed_uri.len + 1
2402 + r->http_protocol.len;
2403
2404 p = ngx_pnalloc(r->pool, r->request_line.len + 1);
2405 if (p == NULL) {
2406 ngx_http_finalize_request(r, NGX_HTTP_INTERNAL_SERVER_ERROR);
2407 return NGX_ERROR;
2408 }
2409
2410 r->request_line.data = p;
2411
2412 p = ngx_cpymem(p, r->method_name.data, r->method_name.len);
2413
2414 *p++ = ' ';
2415
2416 p = ngx_cpymem(p, r->unparsed_uri.data, r->unparsed_uri.len);
2417
2418 *p++ = ' ';
2419
2420 ngx_memcpy(p, r->http_protocol.data, r->http_protocol.len + 1);
2421
2422 /* some modules expect the space character after method name */
2423 r->method_name.data = r->request_line.data;
2424
2425 return NGX_OK;
2426}
2427
2428
2429static void
2430ngx_http_spdy_run_request(ngx_http_request_t *r)
2431{
2432 ngx_uint_t i;
2433 ngx_list_part_t *part;
2434 ngx_table_elt_t *h;
2435 ngx_http_header_t *hh;
2436 ngx_http_core_main_conf_t *cmcf;
2437
2438 if (ngx_http_spdy_construct_request_line(r) != NGX_OK) {
2439 return;
2440 }
2441
2442 ngx_log_debug1(NGX_LOG_DEBUG_HTTP, r->connection->log, 0,
2443 "spdy http request line: \"%V\"", &r->request_line);
2444
2445 cmcf = ngx_http_get_module_main_conf(r, ngx_http_core_module);
2446
2447 part = &r->headers_in.headers.part;
2448 h = part->elts;
2449
2450 for (i = 0 ;; i++) {
2451
2452 if (i >= part->nelts) {
2453 if (part->next == NULL) {
2454 break;
2455 }
2456
2457 part = part->next;
2458 h = part->elts;
2459 i = 0;
2460 }
2461
2462 hh = ngx_hash_find(&cmcf->headers_in_hash, h[i].hash,
2463 h[i].lowcase_key, h[i].key.len);
2464
2465 if (hh && hh->handler(r, &h[i], hh->offset) != NGX_OK) {
2466 return;
2467 }
2468
2469 ngx_log_debug2(NGX_LOG_DEBUG_HTTP, r->connection->log, 0,
2470 "http header: \"%V: %V\"", &h[i].key, &h[i].value);
2471 }
2472
2473 r->http_state = NGX_HTTP_PROCESS_REQUEST_STATE;
2474
2475 if (ngx_http_process_request_header(r) != NGX_OK) {
2476 return;
2477 }
2478
2479 ngx_http_process_request(r);
2480}
2481
2482
2483static ngx_int_t
2484ngx_http_spdy_init_request_body(ngx_http_request_t *r)
2485{
2486 ngx_buf_t *buf;
2487 ngx_temp_file_t *tf;
2488 ngx_http_request_body_t *rb;
2489 ngx_http_core_loc_conf_t *clcf;
2490
2491 rb = ngx_pcalloc(r->pool, sizeof(ngx_http_request_body_t));
2492 if (rb == NULL) {
2493 return NGX_ERROR;
2494 }
2495
2496 r->request_body = rb;
2497
2498 if (r->spdy_stream->in_closed) {
2499 return NGX_OK;
2500 }
2501
2502 rb->rest = r->headers_in.content_length_n;
2503
2504 clcf = ngx_http_get_module_loc_conf(r, ngx_http_core_module);
2505
2506 if (r->request_body_in_file_only
2507 || rb->rest > (off_t) clcf->client_body_buffer_size
2508 || rb->rest < 0)
2509 {
2510 tf = ngx_pcalloc(r->pool, sizeof(ngx_temp_file_t));
2511 if (tf == NULL) {
2512 return NGX_ERROR;
2513 }
2514
2515 tf->file.fd = NGX_INVALID_FILE;
2516 tf->file.log = r->connection->log;
2517 tf->path = clcf->client_body_temp_path;
2518 tf->pool = r->pool;
2519 tf->warn = "a client request body is buffered to a temporary file";
2520 tf->log_level = r->request_body_file_log_level;
2521 tf->persistent = r->request_body_in_persistent_file;
2522 tf->clean = r->request_body_in_clean_file;
2523
2524 if (r->request_body_file_group_access) {
2525 tf->access = 0660;
2526 }
2527
2528 rb->temp_file = tf;
2529
2530 if (r->spdy_stream->in_closed
2531 && ngx_create_temp_file(&tf->file, tf->path, tf->pool,
2532 tf->persistent, tf->clean, tf->access)
2533 != NGX_OK)
2534 {
2535 return NGX_ERROR;
2536 }
2537
2538 buf = ngx_calloc_buf(r->pool);
2539 if (buf == NULL) {
2540 return NGX_ERROR;
2541 }
2542
Valentin Bartenev2686cb42013-03-20 10:36:57 +00002543 } else {
2544
2545 if (rb->rest == 0) {
2546 return NGX_OK;
2547 }
2548
2549 buf = ngx_create_temp_buf(r->pool, (size_t) rb->rest);
2550 if (buf == NULL) {
2551 return NGX_ERROR;
2552 }
Valentin Bartenev2686cb42013-03-20 10:36:57 +00002553 }
2554
Valentin Bartenev32e167e2013-07-24 22:24:25 +04002555 rb->buf = buf;
2556
Valentin Bartenev2686cb42013-03-20 10:36:57 +00002557 rb->bufs = ngx_alloc_chain_link(r->pool);
2558 if (rb->bufs == NULL) {
2559 return NGX_ERROR;
2560 }
2561
2562 rb->bufs->buf = buf;
2563 rb->bufs->next = NULL;
2564
2565 rb->rest = 0;
2566
2567 return NGX_OK;
2568}
2569
2570
2571ngx_int_t
2572ngx_http_spdy_read_request_body(ngx_http_request_t *r,
2573 ngx_http_client_body_handler_pt post_handler)
2574{
2575 ngx_http_spdy_stream_t *stream;
2576
2577 ngx_log_debug0(NGX_LOG_DEBUG_HTTP, r->connection->log, 0,
2578 "spdy read request body");
2579
2580 stream = r->spdy_stream;
2581
2582 switch (stream->skip_data) {
2583
2584 case NGX_SPDY_DATA_DISCARD:
2585 post_handler(r);
2586 return NGX_OK;
2587
2588 case NGX_SPDY_DATA_ERROR:
2589 if (r->headers_in.content_length_n == -1) {
2590 return NGX_HTTP_REQUEST_ENTITY_TOO_LARGE;
2591 } else {
2592 return NGX_HTTP_BAD_REQUEST;
2593 }
2594
2595 case NGX_SPDY_DATA_INTERNAL_ERROR:
2596 return NGX_HTTP_INTERNAL_SERVER_ERROR;
2597 }
2598
2599 if (!r->request_body && ngx_http_spdy_init_request_body(r) != NGX_OK) {
2600 stream->skip_data = NGX_SPDY_DATA_INTERNAL_ERROR;
2601 return NGX_HTTP_INTERNAL_SERVER_ERROR;
2602 }
2603
2604 if (stream->in_closed) {
2605 post_handler(r);
2606 return NGX_OK;
2607 }
2608
2609 r->request_body->post_handler = post_handler;
2610
Valentin Bartenev6ba03092013-10-01 00:00:57 +04002611 r->read_event_handler = ngx_http_test_reading;
2612 r->write_event_handler = ngx_http_request_empty_handler;
2613
Valentin Bartenev2686cb42013-03-20 10:36:57 +00002614 return NGX_AGAIN;
2615}
2616
2617
2618void
2619ngx_http_spdy_close_stream(ngx_http_spdy_stream_t *stream, ngx_int_t rc)
2620{
2621 ngx_event_t *ev;
2622 ngx_connection_t *fc;
2623 ngx_http_spdy_stream_t **index, *s;
2624 ngx_http_spdy_srv_conf_t *sscf;
2625 ngx_http_spdy_connection_t *sc;
2626
2627 sc = stream->connection;
2628
2629 ngx_log_debug2(NGX_LOG_DEBUG_HTTP, sc->connection->log, 0,
2630 "spdy close stream %ui, processing %ui",
2631 stream->id, sc->processing);
2632
2633 if (!stream->out_closed) {
2634 if (ngx_http_spdy_send_rst_stream(sc, stream->id,
2635 NGX_SPDY_INTERNAL_ERROR,
2636 stream->priority)
2637 != NGX_OK)
2638 {
2639 sc->connection->error = 1;
2640 }
2641 }
2642
2643 sscf = ngx_http_get_module_srv_conf(sc->http_connection->conf_ctx,
2644 ngx_http_spdy_module);
2645
2646 index = sc->streams_index + ngx_http_spdy_stream_index(sscf, stream->id);
2647
2648 for ( ;; ) {
2649 s = *index;
2650
2651 if (s == NULL) {
2652 break;
2653 }
2654
2655 if (s == stream) {
2656 *index = s->index;
2657 break;
2658 }
2659
2660 index = &s->index;
2661 }
2662
2663 fc = stream->request->connection;
2664
2665 ngx_http_free_request(stream->request, rc);
2666
2667 ev = fc->read;
2668
2669 if (ev->active || ev->disabled) {
Valentin Bartenevc189eda2013-08-15 19:16:12 +04002670 ngx_log_error(NGX_LOG_ALERT, sc->connection->log, 0,
2671 "spdy fake read event was activated");
Valentin Bartenev2686cb42013-03-20 10:36:57 +00002672 }
2673
2674 if (ev->timer_set) {
2675 ngx_del_timer(ev);
2676 }
2677
2678 if (ev->prev) {
2679 ngx_delete_posted_event(ev);
2680 }
2681
2682 ev = fc->write;
2683
2684 if (ev->active || ev->disabled) {
Valentin Bartenevc189eda2013-08-15 19:16:12 +04002685 ngx_log_error(NGX_LOG_ALERT, sc->connection->log, 0,
2686 "spdy fake write event was activated");
Valentin Bartenev2686cb42013-03-20 10:36:57 +00002687 }
2688
2689 if (ev->timer_set) {
2690 ngx_del_timer(ev);
2691 }
2692
2693 if (ev->prev) {
2694 ngx_delete_posted_event(ev);
2695 }
2696
2697 fc->data = sc->free_fake_connections;
2698 sc->free_fake_connections = fc;
2699
2700 sc->processing--;
2701
2702 if (sc->processing || sc->blocked) {
2703 return;
2704 }
2705
2706 ev = sc->connection->read;
2707
2708 ev->handler = ngx_http_spdy_handle_connection_handler;
2709 ngx_post_event(ev, &ngx_posted_events);
2710}
2711
2712
2713static void
2714ngx_http_spdy_handle_connection_handler(ngx_event_t *rev)
2715{
2716 ngx_connection_t *c;
2717
2718 rev->handler = ngx_http_spdy_read_handler;
2719
2720 if (rev->ready) {
2721 ngx_http_spdy_read_handler(rev);
2722 return;
2723 }
2724
2725 c = rev->data;
2726
2727 ngx_http_spdy_handle_connection(c->data);
2728}
2729
2730
2731static void
2732ngx_http_spdy_keepalive_handler(ngx_event_t *rev)
2733{
2734 ngx_connection_t *c;
2735 ngx_http_spdy_srv_conf_t *sscf;
2736 ngx_http_spdy_connection_t *sc;
2737
2738 c = rev->data;
2739
2740 ngx_log_debug0(NGX_LOG_DEBUG_HTTP, c->log, 0, "spdy keepalive handler");
2741
2742 if (rev->timedout || c->close) {
2743 ngx_http_close_connection(c);
2744 return;
2745 }
2746
2747#if (NGX_HAVE_KQUEUE)
2748
2749 if (ngx_event_flags & NGX_USE_KQUEUE_EVENT) {
2750 if (rev->pending_eof) {
2751 c->log->handler = NULL;
2752 ngx_log_error(NGX_LOG_INFO, c->log, rev->kq_errno,
2753 "kevent() reported that client %V closed "
2754 "keepalive connection", &c->addr_text);
2755#if (NGX_HTTP_SSL)
2756 if (c->ssl) {
2757 c->ssl->no_send_shutdown = 1;
2758 }
2759#endif
2760 ngx_http_close_connection(c);
2761 return;
2762 }
2763 }
2764
2765#endif
2766
2767 c->destroyed = 0;
2768 c->idle = 0;
2769 ngx_reusable_connection(c, 0);
2770
2771 sc = c->data;
2772
2773 sscf = ngx_http_get_module_srv_conf(sc->http_connection->conf_ctx,
2774 ngx_http_spdy_module);
2775
2776 sc->pool = ngx_create_pool(sscf->pool_size, sc->connection->log);
2777 if (sc->pool == NULL) {
2778 ngx_http_close_connection(c);
2779 return;
2780 }
2781
2782 sc->streams_index = ngx_pcalloc(sc->pool,
2783 ngx_http_spdy_streams_index_size(sscf)
2784 * sizeof(ngx_http_spdy_stream_t *));
2785 if (sc->streams_index == NULL) {
2786 ngx_http_close_connection(c);
2787 return;
2788 }
2789
2790 c->write->handler = ngx_http_spdy_write_handler;
2791
2792 rev->handler = ngx_http_spdy_read_handler;
2793 ngx_http_spdy_read_handler(rev);
2794}
2795
2796
2797static void
2798ngx_http_spdy_finalize_connection(ngx_http_spdy_connection_t *sc,
2799 ngx_int_t rc)
2800{
2801 ngx_uint_t i, size;
2802 ngx_event_t *ev;
2803 ngx_connection_t *c, *fc;
2804 ngx_http_request_t *r;
2805 ngx_http_spdy_stream_t *stream;
2806 ngx_http_spdy_srv_conf_t *sscf;
2807
2808 c = sc->connection;
2809
2810 if (!sc->processing) {
2811 ngx_http_close_connection(c);
2812 return;
2813 }
2814
2815 c->error = 1;
2816 c->read->handler = ngx_http_empty_handler;
2817
2818 sc->last_out = NULL;
2819
2820 sc->blocked = 1;
2821
2822 sscf = ngx_http_get_module_srv_conf(sc->http_connection->conf_ctx,
2823 ngx_http_spdy_module);
2824
2825 size = ngx_http_spdy_streams_index_size(sscf);
2826
2827 for (i = 0; i < size; i++) {
2828 stream = sc->streams_index[i];
2829
2830 while (stream) {
2831 r = stream->request;
2832
2833 fc = r->connection;
2834 fc->error = 1;
2835
2836 if (stream->waiting) {
2837 r->blocked -= stream->waiting;
2838 stream->waiting = 0;
2839 ev = fc->write;
2840
2841 } else {
2842 ev = fc->read;
2843 }
2844
2845 stream = stream->index;
2846
2847 ev->eof = 1;
2848 ev->handler(ev);
2849 }
2850 }
2851
2852 sc->blocked = 0;
2853
2854 if (sc->processing) {
2855 return;
2856 }
2857
2858 ngx_http_close_connection(c);
2859}
2860
2861
2862static void
2863ngx_http_spdy_pool_cleanup(void *data)
2864{
2865 ngx_http_spdy_connection_t *sc = data;
2866
2867 if (sc->pool) {
2868 ngx_destroy_pool(sc->pool);
2869 }
2870}
2871
2872
2873static void *
2874ngx_http_spdy_zalloc(void *opaque, u_int items, u_int size)
2875{
2876 ngx_http_spdy_connection_t *sc = opaque;
2877
2878 return ngx_palloc(sc->connection->pool, items * size);
2879}
2880
2881
2882static void
2883ngx_http_spdy_zfree(void *opaque, void *address)
2884{
2885#if 0
2886 ngx_http_spdy_connection_t *sc = opaque;
2887
2888 ngx_log_debug1(NGX_LOG_DEBUG_HTTP, sc->connection->log, 0,
2889 "spdy zfree: %p", address);
2890#endif
2891}