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