Mercurial > hg > nginx
annotate src/http/v3/ngx_http_v3_parse.c @ 8456:c9538aef3211 quic
HTTP/3: refactored dynamic table implementation.
Previously dynamic table was not functional because of zero limit on its size
set by default. Now the following changes enable it:
- new directives to set SETTINGS_QPACK_MAX_TABLE_CAPACITY and
SETTINGS_QPACK_BLOCKED_STREAMS
- send settings with SETTINGS_QPACK_MAX_TABLE_CAPACITY and
SETTINGS_QPACK_BLOCKED_STREAMS to the client
- send Insert Count Increment to the client
- send Header Acknowledgement to the client
- evict old dynamic table entries on overflow
- decode Required Insert Count from client
- block stream if Required Insert Count is not reached
author | Roman Arutyunyan <arut@nginx.com> |
---|---|
date | Thu, 02 Jul 2020 15:34:05 +0300 |
parents | b0e81f49d7c0 |
children | a7f64438aa3c |
rev | line source |
---|---|
8226 | 1 |
2 /* | |
3 * Copyright (C) Roman Arutyunyan | |
4 * Copyright (C) Nginx, Inc. | |
5 */ | |
6 | |
7 | |
8 #include <ngx_config.h> | |
9 #include <ngx_core.h> | |
10 #include <ngx_http.h> | |
11 | |
12 | |
8456
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
13 static ngx_int_t ngx_http_v3_parse_lookup(ngx_connection_t *c, |
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
14 ngx_uint_t dynamic, ngx_uint_t index, ngx_str_t *name, ngx_str_t *value); |
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
15 |
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
16 |
8226 | 17 ngx_int_t |
18 ngx_http_v3_parse_varlen_int(ngx_connection_t *c, | |
19 ngx_http_v3_parse_varlen_int_t *st, u_char ch) | |
20 { | |
21 enum { | |
22 sw_start = 0, | |
23 sw_length_2, | |
24 sw_length_3, | |
25 sw_length_4, | |
26 sw_length_5, | |
27 sw_length_6, | |
28 sw_length_7, | |
29 sw_length_8 | |
30 }; | |
31 | |
32 switch (st->state) { | |
33 | |
34 case sw_start: | |
35 | |
36 st->value = ch; | |
37 if (st->value & 0xc0) { | |
38 st->state = sw_length_2; | |
39 break; | |
40 } | |
41 | |
42 goto done; | |
43 | |
44 case sw_length_2: | |
45 | |
46 st->value = (st->value << 8) + ch; | |
47 if ((st->value & 0xc000) == 0x4000) { | |
48 st->value &= 0x3fff; | |
49 goto done; | |
50 } | |
51 | |
52 st->state = sw_length_3; | |
53 break; | |
54 | |
55 case sw_length_4: | |
56 | |
57 st->value = (st->value << 8) + ch; | |
58 if ((st->value & 0xc0000000) == 0x80000000) { | |
59 st->value &= 0x3fffffff; | |
60 goto done; | |
61 } | |
62 | |
63 st->state = sw_length_5; | |
64 break; | |
65 | |
66 case sw_length_3: | |
67 case sw_length_5: | |
68 case sw_length_6: | |
69 case sw_length_7: | |
70 | |
71 st->value = (st->value << 8) + ch; | |
72 st->state++; | |
73 break; | |
74 | |
75 case sw_length_8: | |
76 | |
77 st->value = (st->value << 8) + ch; | |
78 st->value &= 0x3fffffffffffffff; | |
79 goto done; | |
80 } | |
81 | |
82 return NGX_AGAIN; | |
83 | |
84 done: | |
85 | |
86 ngx_log_debug1(NGX_LOG_DEBUG_HTTP, c->log, 0, | |
87 "http3 parse varlen int %uL", st->value); | |
88 | |
89 st->state = sw_start; | |
90 return NGX_DONE; | |
91 } | |
92 | |
93 | |
94 ngx_int_t | |
95 ngx_http_v3_parse_prefix_int(ngx_connection_t *c, | |
96 ngx_http_v3_parse_prefix_int_t *st, ngx_uint_t prefix, u_char ch) | |
97 { | |
8455
b0e81f49d7c0
HTTP/3: fixed prefixed integer encoding and decoding.
Roman Arutyunyan <arut@nginx.com>
parents:
8454
diff
changeset
|
98 ngx_uint_t mask; |
8226 | 99 enum { |
100 sw_start = 0, | |
101 sw_value | |
102 }; | |
103 | |
104 switch (st->state) { | |
105 | |
106 case sw_start: | |
107 | |
8455
b0e81f49d7c0
HTTP/3: fixed prefixed integer encoding and decoding.
Roman Arutyunyan <arut@nginx.com>
parents:
8454
diff
changeset
|
108 mask = (1 << prefix) - 1; |
b0e81f49d7c0
HTTP/3: fixed prefixed integer encoding and decoding.
Roman Arutyunyan <arut@nginx.com>
parents:
8454
diff
changeset
|
109 st->value = ch & mask; |
8226 | 110 |
8455
b0e81f49d7c0
HTTP/3: fixed prefixed integer encoding and decoding.
Roman Arutyunyan <arut@nginx.com>
parents:
8454
diff
changeset
|
111 if (st->value != mask) { |
8226 | 112 goto done; |
113 } | |
114 | |
8455
b0e81f49d7c0
HTTP/3: fixed prefixed integer encoding and decoding.
Roman Arutyunyan <arut@nginx.com>
parents:
8454
diff
changeset
|
115 st->shift = 0; |
8226 | 116 st->state = sw_value; |
117 break; | |
118 | |
119 case sw_value: | |
120 | |
8455
b0e81f49d7c0
HTTP/3: fixed prefixed integer encoding and decoding.
Roman Arutyunyan <arut@nginx.com>
parents:
8454
diff
changeset
|
121 st->value += (ch & 0x7f) << st->shift; |
8226 | 122 if (ch & 0x80) { |
8455
b0e81f49d7c0
HTTP/3: fixed prefixed integer encoding and decoding.
Roman Arutyunyan <arut@nginx.com>
parents:
8454
diff
changeset
|
123 st->shift += 7; |
8226 | 124 break; |
125 } | |
126 | |
127 goto done; | |
128 } | |
129 | |
130 return NGX_AGAIN; | |
131 | |
132 done: | |
133 | |
134 ngx_log_debug1(NGX_LOG_DEBUG_HTTP, c->log, 0, | |
135 "http3 parse prefix int %uL", st->value); | |
136 | |
137 st->state = sw_start; | |
138 return NGX_DONE; | |
139 } | |
140 | |
141 | |
142 ngx_int_t | |
143 ngx_http_v3_parse_headers(ngx_connection_t *c, ngx_http_v3_parse_headers_t *st, | |
144 u_char ch) | |
145 { | |
146 ngx_int_t rc; | |
147 enum { | |
148 sw_start = 0, | |
149 sw_length, | |
150 sw_prefix, | |
8456
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
151 sw_verify, |
8226 | 152 sw_header_rep, |
153 sw_done | |
154 }; | |
155 | |
156 switch (st->state) { | |
157 | |
158 case sw_start: | |
159 | |
160 ngx_log_debug0(NGX_LOG_DEBUG_HTTP, c->log, 0, "http3 parse headers"); | |
161 | |
162 if (ch != NGX_HTTP_V3_FRAME_HEADERS) { | |
163 return NGX_ERROR; | |
164 } | |
165 | |
166 st->state = sw_length; | |
167 break; | |
168 | |
169 case sw_length: | |
170 | |
171 if (ngx_http_v3_parse_varlen_int(c, &st->vlint, ch) != NGX_DONE) { | |
172 break; | |
173 } | |
174 | |
175 st->length = st->vlint.value; | |
176 | |
177 ngx_log_debug1(NGX_LOG_DEBUG_HTTP, c->log, 0, | |
178 "http3 parse headers len:%ui", st->length); | |
179 | |
180 st->state = sw_prefix; | |
181 break; | |
182 | |
183 case sw_prefix: | |
184 | |
185 if (st->length-- == 0) { | |
186 return NGX_ERROR; | |
187 } | |
188 | |
189 rc = ngx_http_v3_parse_header_block_prefix(c, &st->prefix, ch); | |
190 | |
191 if (rc == NGX_ERROR) { | |
192 return NGX_ERROR; | |
193 } | |
194 | |
195 if (rc != NGX_DONE) { | |
196 break; | |
197 } | |
198 | |
199 if (st->length == 0) { | |
200 return NGX_ERROR; | |
201 } | |
202 | |
8456
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
203 st->state = sw_verify; |
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
204 break; |
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
205 |
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
206 case sw_verify: |
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
207 |
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
208 rc = ngx_http_v3_check_insert_count(c, st->prefix.insert_count); |
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
209 if (rc != NGX_OK) { |
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
210 return rc; |
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
211 } |
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
212 |
8226 | 213 st->state = sw_header_rep; |
8456
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
214 |
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
215 /* fall through */ |
8226 | 216 |
217 case sw_header_rep: | |
218 | |
219 rc = ngx_http_v3_parse_header_rep(c, &st->header_rep, st->prefix.base, | |
220 ch); | |
221 | |
222 if (rc == NGX_ERROR) { | |
223 return NGX_ERROR; | |
224 } | |
225 | |
226 if (--st->length == 0) { | |
227 if (rc != NGX_DONE) { | |
228 return NGX_ERROR; | |
229 } | |
230 | |
231 goto done; | |
232 } | |
233 | |
234 if (rc == NGX_DONE) { | |
235 return NGX_OK; | |
236 } | |
237 | |
238 break; | |
239 } | |
240 | |
241 return NGX_AGAIN; | |
242 | |
243 done: | |
244 | |
245 ngx_log_debug0(NGX_LOG_DEBUG_HTTP, c->log, 0, "http3 parse headers done"); | |
246 | |
8456
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
247 if (st->prefix.insert_count > 0) { |
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
248 if (ngx_http_v3_client_ack_header(c, c->qs->id) != NGX_OK) { |
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
249 return NGX_ERROR; |
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
250 } |
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
251 } |
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
252 |
8226 | 253 st->state = sw_start; |
254 return NGX_DONE; | |
255 } | |
256 | |
257 | |
258 ngx_int_t | |
259 ngx_http_v3_parse_header_block_prefix(ngx_connection_t *c, | |
260 ngx_http_v3_parse_header_block_prefix_t *st, u_char ch) | |
261 { | |
262 enum { | |
263 sw_start = 0, | |
264 sw_req_insert_count, | |
265 sw_delta_base, | |
266 sw_read_delta_base | |
267 }; | |
268 | |
269 switch (st->state) { | |
270 | |
271 case sw_start: | |
272 | |
273 ngx_log_debug0(NGX_LOG_DEBUG_HTTP, c->log, 0, | |
274 "http3 parse header block prefix"); | |
275 | |
276 st->state = sw_req_insert_count; | |
277 | |
278 /* fall through */ | |
279 | |
280 case sw_req_insert_count: | |
281 | |
282 if (ngx_http_v3_parse_prefix_int(c, &st->pint, 8, ch) != NGX_DONE) { | |
283 break; | |
284 } | |
285 | |
286 st->insert_count = st->pint.value; | |
287 st->state = sw_delta_base; | |
288 break; | |
289 | |
290 case sw_delta_base: | |
291 | |
292 st->sign = (ch & 0x80) ? 1 : 0; | |
293 st->state = sw_read_delta_base; | |
294 | |
295 /* fall through */ | |
296 | |
297 case sw_read_delta_base: | |
298 | |
299 if (ngx_http_v3_parse_prefix_int(c, &st->pint, 7, ch) != NGX_DONE) { | |
300 break; | |
301 } | |
302 | |
303 st->delta_base = st->pint.value; | |
304 goto done; | |
305 } | |
306 | |
307 return NGX_AGAIN; | |
308 | |
309 done: | |
310 | |
8456
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
311 if (ngx_http_v3_decode_insert_count(c, &st->insert_count) != NGX_OK) { |
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
312 return NGX_ERROR; |
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
313 } |
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
314 |
8226 | 315 if (st->sign) { |
316 st->base = st->insert_count - st->delta_base - 1; | |
317 } else { | |
318 st->base = st->insert_count + st->delta_base; | |
319 } | |
320 | |
321 ngx_log_debug4(NGX_LOG_DEBUG_HTTP, c->log, 0, | |
322 "http3 parse header block prefix done " | |
8456
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
323 "insert_count:%ui, sign:%ui, delta_base:%ui, base:%uL", |
8226 | 324 st->insert_count, st->sign, st->delta_base, st->base); |
325 | |
326 st->state = sw_start; | |
327 return NGX_DONE; | |
328 } | |
329 | |
330 | |
331 ngx_int_t | |
332 ngx_http_v3_parse_header_rep(ngx_connection_t *c, | |
333 ngx_http_v3_parse_header_rep_t *st, ngx_uint_t base, u_char ch) | |
334 { | |
335 ngx_int_t rc; | |
336 enum { | |
337 sw_start = 0, | |
338 sw_header_ri, | |
339 sw_header_lri, | |
340 sw_header_l, | |
341 sw_header_pbi, | |
342 sw_header_lpbi | |
343 }; | |
344 | |
345 if (st->state == sw_start) { | |
346 | |
347 ngx_log_debug0(NGX_LOG_DEBUG_HTTP, c->log, 0, | |
348 "http3 parse header representation"); | |
349 | |
350 ngx_memzero(&st->header, sizeof(ngx_http_v3_parse_header_t)); | |
351 | |
352 st->header.base = base; | |
353 | |
354 if (ch & 0x80) { | |
355 /* Indexed Header Field */ | |
356 | |
357 st->state = sw_header_ri; | |
358 | |
359 } else if (ch & 0x40) { | |
360 /* Literal Header Field With Name Reference */ | |
361 | |
362 st->state = sw_header_lri; | |
363 | |
364 } else if (ch & 0x20) { | |
365 /* Literal Header Field Without Name Reference */ | |
366 | |
367 st->state = sw_header_l; | |
368 | |
369 } else if (ch & 0x10) { | |
370 /* Indexed Header Field With Post-Base Index */ | |
371 | |
372 st->state = sw_header_pbi; | |
373 | |
374 } else { | |
375 /* Literal Header Field With Post-Base Name Reference */ | |
376 | |
377 st->state = sw_header_lpbi; | |
378 } | |
379 } | |
380 | |
381 switch (st->state) { | |
382 | |
383 case sw_header_ri: | |
384 rc = ngx_http_v3_parse_header_ri(c, &st->header, ch); | |
385 break; | |
386 | |
387 case sw_header_lri: | |
388 rc = ngx_http_v3_parse_header_lri(c, &st->header, ch); | |
389 break; | |
390 | |
391 case sw_header_l: | |
392 rc = ngx_http_v3_parse_header_l(c, &st->header, ch); | |
393 break; | |
394 | |
395 case sw_header_pbi: | |
396 rc = ngx_http_v3_parse_header_pbi(c, &st->header, ch); | |
397 break; | |
398 | |
399 case sw_header_lpbi: | |
400 rc = ngx_http_v3_parse_header_lpbi(c, &st->header, ch); | |
401 break; | |
402 | |
403 default: | |
404 rc = NGX_OK; | |
405 } | |
406 | |
407 if (rc == NGX_ERROR) { | |
408 return NGX_ERROR; | |
409 } | |
410 | |
411 if (rc == NGX_AGAIN) { | |
412 return NGX_AGAIN; | |
413 } | |
414 | |
415 /* rc == NGX_DONE */ | |
416 | |
417 ngx_log_debug0(NGX_LOG_DEBUG_HTTP, c->log, 0, | |
418 "http3 parse header representation done"); | |
419 | |
420 st->state = sw_start; | |
421 return NGX_DONE; | |
422 } | |
423 | |
424 | |
425 ngx_int_t | |
426 ngx_http_v3_parse_literal(ngx_connection_t *c, ngx_http_v3_parse_literal_t *st, | |
427 u_char ch) | |
428 { | |
8454
032cb35ce758
HTTP/3: http3_max_field_size directive to limit string size.
Roman Arutyunyan <arut@nginx.com>
parents:
8295
diff
changeset
|
429 ngx_uint_t n; |
032cb35ce758
HTTP/3: http3_max_field_size directive to limit string size.
Roman Arutyunyan <arut@nginx.com>
parents:
8295
diff
changeset
|
430 ngx_http_v3_srv_conf_t *v3cf; |
8226 | 431 enum { |
432 sw_start = 0, | |
433 sw_value | |
434 }; | |
435 | |
436 switch (st->state) { | |
437 | |
438 case sw_start: | |
439 | |
440 ngx_log_debug2(NGX_LOG_DEBUG_HTTP, c->log, 0, | |
441 "http3 parse literal huff:%ui, len:%ui", | |
442 st->huffman, st->length); | |
443 | |
444 n = st->length; | |
445 | |
8454
032cb35ce758
HTTP/3: http3_max_field_size directive to limit string size.
Roman Arutyunyan <arut@nginx.com>
parents:
8295
diff
changeset
|
446 v3cf = ngx_http_v3_get_module_srv_conf(c, ngx_http_v3_module); |
032cb35ce758
HTTP/3: http3_max_field_size directive to limit string size.
Roman Arutyunyan <arut@nginx.com>
parents:
8295
diff
changeset
|
447 |
032cb35ce758
HTTP/3: http3_max_field_size directive to limit string size.
Roman Arutyunyan <arut@nginx.com>
parents:
8295
diff
changeset
|
448 if (n > v3cf->max_field_size) { |
032cb35ce758
HTTP/3: http3_max_field_size directive to limit string size.
Roman Arutyunyan <arut@nginx.com>
parents:
8295
diff
changeset
|
449 ngx_log_error(NGX_LOG_ERR, c->log, 0, |
032cb35ce758
HTTP/3: http3_max_field_size directive to limit string size.
Roman Arutyunyan <arut@nginx.com>
parents:
8295
diff
changeset
|
450 "client exceeded http3_max_field_size limit"); |
032cb35ce758
HTTP/3: http3_max_field_size directive to limit string size.
Roman Arutyunyan <arut@nginx.com>
parents:
8295
diff
changeset
|
451 return NGX_ERROR; |
032cb35ce758
HTTP/3: http3_max_field_size directive to limit string size.
Roman Arutyunyan <arut@nginx.com>
parents:
8295
diff
changeset
|
452 } |
032cb35ce758
HTTP/3: http3_max_field_size directive to limit string size.
Roman Arutyunyan <arut@nginx.com>
parents:
8295
diff
changeset
|
453 |
8226 | 454 if (st->huffman) { |
455 n = n * 8 / 5; | |
456 st->huffstate = 0; | |
457 } | |
458 | |
459 st->last = ngx_pnalloc(c->pool, n + 1); | |
460 if (st->last == NULL) { | |
461 return NGX_ERROR; | |
462 } | |
463 | |
464 st->value.data = st->last; | |
465 st->state = sw_value; | |
466 | |
467 /* fall through */ | |
468 | |
469 case sw_value: | |
470 | |
471 if (st->huffman) { | |
472 if (ngx_http_v2_huff_decode(&st->huffstate, &ch, 1, &st->last, | |
473 st->length == 1, c->log) | |
474 != NGX_OK) | |
475 { | |
476 return NGX_ERROR; | |
477 } | |
478 | |
479 } else { | |
480 *st->last++ = ch; | |
481 } | |
482 | |
483 if (--st->length) { | |
484 break; | |
485 } | |
486 | |
487 st->value.len = st->last - st->value.data; | |
488 *st->last = '\0'; | |
489 goto done; | |
490 } | |
491 | |
492 return NGX_AGAIN; | |
493 | |
494 done: | |
495 | |
496 ngx_log_debug1(NGX_LOG_DEBUG_HTTP, c->log, 0, | |
497 "http3 parse literal done \"%V\"", &st->value); | |
498 | |
499 st->state = sw_start; | |
500 return NGX_DONE; | |
501 } | |
502 | |
503 | |
504 ngx_int_t | |
505 ngx_http_v3_parse_header_ri(ngx_connection_t *c, ngx_http_v3_parse_header_t *st, | |
506 u_char ch) | |
507 { | |
508 enum { | |
509 sw_start = 0, | |
510 sw_index | |
511 }; | |
512 | |
513 switch (st->state) { | |
514 | |
515 case sw_start: | |
516 | |
517 ngx_log_debug0(NGX_LOG_DEBUG_HTTP, c->log, 0, "http3 parse header ri"); | |
518 | |
519 st->dynamic = (ch & 0x40) ? 0 : 1; | |
520 st->state = sw_index; | |
521 | |
522 /* fall through */ | |
523 | |
524 case sw_index: | |
525 | |
526 if (ngx_http_v3_parse_prefix_int(c, &st->pint, 6, ch) != NGX_DONE) { | |
527 break; | |
528 } | |
529 | |
530 st->index = st->pint.value; | |
531 goto done; | |
532 } | |
533 | |
534 return NGX_AGAIN; | |
535 | |
536 done: | |
537 | |
538 ngx_log_debug2(NGX_LOG_DEBUG_HTTP, c->log, 0, | |
539 "http3 parse header ri done %s%ui]", | |
540 st->dynamic ? "dynamic[-" : "static[", st->index); | |
541 | |
542 if (st->dynamic) { | |
543 st->index = st->base - st->index - 1; | |
544 } | |
545 | |
8456
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
546 if (ngx_http_v3_parse_lookup(c, st->dynamic, st->index, &st->name, |
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
547 &st->value) |
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
548 != NGX_OK) |
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
549 { |
8226 | 550 return NGX_ERROR; |
551 } | |
552 | |
553 st->state = sw_start; | |
554 return NGX_DONE; | |
555 } | |
556 | |
557 | |
558 ngx_int_t | |
559 ngx_http_v3_parse_header_lri(ngx_connection_t *c, | |
560 ngx_http_v3_parse_header_t *st, u_char ch) | |
561 { | |
8456
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
562 ngx_int_t rc; |
8226 | 563 enum { |
564 sw_start = 0, | |
565 sw_index, | |
566 sw_value_len, | |
567 sw_read_value_len, | |
568 sw_value | |
569 }; | |
570 | |
571 switch (st->state) { | |
572 | |
573 case sw_start: | |
574 | |
575 ngx_log_debug0(NGX_LOG_DEBUG_HTTP, c->log, 0, "http3 parse header lri"); | |
576 | |
577 st->dynamic = (ch & 0x10) ? 0 : 1; | |
578 st->state = sw_index; | |
579 | |
580 /* fall through */ | |
581 | |
582 case sw_index: | |
583 | |
584 if (ngx_http_v3_parse_prefix_int(c, &st->pint, 4, ch) != NGX_DONE) { | |
585 break; | |
586 } | |
587 | |
588 st->index = st->pint.value; | |
589 st->state = sw_value_len; | |
590 break; | |
591 | |
592 case sw_value_len: | |
593 | |
594 st->literal.huffman = (ch & 0x80) ? 1 : 0; | |
595 st->state = sw_read_value_len; | |
596 | |
597 /* fall through */ | |
598 | |
599 case sw_read_value_len: | |
600 | |
601 if (ngx_http_v3_parse_prefix_int(c, &st->pint, 7, ch) != NGX_DONE) { | |
602 break; | |
603 } | |
604 | |
605 st->literal.length = st->pint.value; | |
606 if (st->literal.length == 0) { | |
607 goto done; | |
608 } | |
609 | |
610 st->state = sw_value; | |
611 break; | |
612 | |
613 case sw_value: | |
614 | |
615 rc = ngx_http_v3_parse_literal(c, &st->literal, ch); | |
616 | |
617 if (rc == NGX_ERROR) { | |
618 return NGX_ERROR; | |
619 } | |
620 | |
621 if (rc == NGX_DONE) { | |
622 st->value = st->literal.value; | |
623 goto done; | |
624 } | |
625 | |
626 break; | |
627 } | |
628 | |
629 return NGX_AGAIN; | |
630 | |
631 done: | |
632 | |
633 ngx_log_debug3(NGX_LOG_DEBUG_HTTP, c->log, 0, | |
634 "http3 parse header lri done %s%ui] \"%V\"", | |
635 st->dynamic ? "dynamic[-" : "static[", | |
636 st->index, &st->value); | |
637 | |
638 if (st->dynamic) { | |
639 st->index = st->base - st->index - 1; | |
640 } | |
641 | |
8456
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
642 if (ngx_http_v3_parse_lookup(c, st->dynamic, st->index, &st->name, NULL) |
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
643 != NGX_OK) |
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
644 { |
8226 | 645 return NGX_ERROR; |
646 } | |
647 | |
648 st->state = sw_start; | |
649 return NGX_DONE; | |
650 } | |
651 | |
652 | |
653 ngx_int_t | |
654 ngx_http_v3_parse_header_l(ngx_connection_t *c, | |
655 ngx_http_v3_parse_header_t *st, u_char ch) | |
656 { | |
657 ngx_int_t rc; | |
658 enum { | |
659 sw_start = 0, | |
660 sw_name_len, | |
661 sw_name, | |
662 sw_value_len, | |
663 sw_read_value_len, | |
664 sw_value | |
665 }; | |
666 | |
667 switch (st->state) { | |
668 | |
669 case sw_start: | |
670 | |
671 ngx_log_debug0(NGX_LOG_DEBUG_HTTP, c->log, 0, "http3 parse header l"); | |
672 | |
673 st->literal.huffman = (ch & 0x08) ? 1 : 0; | |
674 st->state = sw_name_len; | |
675 | |
676 /* fall through */ | |
677 | |
678 case sw_name_len: | |
679 | |
680 if (ngx_http_v3_parse_prefix_int(c, &st->pint, 3, ch) != NGX_DONE) { | |
681 break; | |
682 } | |
683 | |
684 st->literal.length = st->pint.value; | |
685 if (st->literal.length == 0) { | |
686 return NGX_ERROR; | |
687 } | |
688 | |
689 st->state = sw_name; | |
690 break; | |
691 | |
692 case sw_name: | |
693 | |
694 rc = ngx_http_v3_parse_literal(c, &st->literal, ch); | |
695 | |
696 if (rc == NGX_ERROR) { | |
697 return NGX_ERROR; | |
698 } | |
699 | |
700 if (rc == NGX_DONE) { | |
701 st->name = st->literal.value; | |
702 st->state = sw_value_len; | |
703 } | |
704 | |
705 break; | |
706 | |
707 case sw_value_len: | |
708 | |
709 st->literal.huffman = (ch & 0x80) ? 1 : 0; | |
710 st->state = sw_read_value_len; | |
711 | |
712 /* fall through */ | |
713 | |
714 case sw_read_value_len: | |
715 | |
716 if (ngx_http_v3_parse_prefix_int(c, &st->pint, 7, ch) != NGX_DONE) { | |
717 break; | |
718 } | |
719 | |
720 st->literal.length = st->pint.value; | |
721 if (st->literal.length == 0) { | |
722 goto done; | |
723 } | |
724 | |
725 st->state = sw_value; | |
726 break; | |
727 | |
728 case sw_value: | |
729 | |
730 rc = ngx_http_v3_parse_literal(c, &st->literal, ch); | |
731 | |
732 if (rc == NGX_ERROR) { | |
733 return NGX_ERROR; | |
734 } | |
735 | |
736 if (rc == NGX_DONE) { | |
737 st->value = st->literal.value; | |
738 goto done; | |
739 } | |
740 | |
741 break; | |
742 } | |
743 | |
744 return NGX_AGAIN; | |
745 | |
746 done: | |
747 | |
748 ngx_log_debug2(NGX_LOG_DEBUG_HTTP, c->log, 0, | |
749 "http3 parse header l done \"%V\" \"%V\"", | |
750 &st->name, &st->value); | |
751 | |
752 st->state = sw_start; | |
753 return NGX_DONE; | |
754 } | |
755 | |
756 | |
757 ngx_int_t | |
758 ngx_http_v3_parse_header_pbi(ngx_connection_t *c, | |
759 ngx_http_v3_parse_header_t *st, u_char ch) | |
760 { | |
761 enum { | |
762 sw_start = 0, | |
763 sw_index | |
764 }; | |
765 | |
766 switch (st->state) { | |
767 | |
768 case sw_start: | |
769 | |
770 ngx_log_debug0(NGX_LOG_DEBUG_HTTP, c->log, 0, "http3 parse header pbi"); | |
771 | |
772 st->state = sw_index; | |
773 | |
774 /* fall through */ | |
775 | |
776 case sw_index: | |
777 | |
778 if (ngx_http_v3_parse_prefix_int(c, &st->pint, 4, ch) != NGX_DONE) { | |
779 break; | |
780 } | |
781 | |
782 st->index = st->pint.value; | |
783 goto done; | |
784 } | |
785 | |
786 return NGX_AGAIN; | |
787 | |
788 done: | |
789 | |
790 ngx_log_debug1(NGX_LOG_DEBUG_HTTP, c->log, 0, | |
791 "http3 parse header pbi done dynamic[+%ui]", st->index); | |
792 | |
8456
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
793 if (ngx_http_v3_parse_lookup(c, 1, st->base + st->index, &st->name, |
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
794 &st->value) |
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
795 != NGX_OK) |
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
796 { |
8226 | 797 return NGX_ERROR; |
798 } | |
799 | |
800 st->state = sw_start; | |
801 return NGX_DONE; | |
802 } | |
803 | |
804 | |
805 ngx_int_t | |
806 ngx_http_v3_parse_header_lpbi(ngx_connection_t *c, | |
807 ngx_http_v3_parse_header_t *st, u_char ch) | |
808 { | |
8456
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
809 ngx_int_t rc; |
8226 | 810 enum { |
811 sw_start = 0, | |
812 sw_index, | |
813 sw_value_len, | |
814 sw_read_value_len, | |
815 sw_value | |
816 }; | |
817 | |
818 switch (st->state) { | |
819 | |
820 case sw_start: | |
821 | |
822 ngx_log_debug0(NGX_LOG_DEBUG_HTTP, c->log, 0, | |
823 "http3 parse header lpbi"); | |
824 | |
825 st->state = sw_index; | |
826 | |
827 /* fall through */ | |
828 | |
829 case sw_index: | |
830 | |
831 if (ngx_http_v3_parse_prefix_int(c, &st->pint, 3, ch) != NGX_DONE) { | |
832 break; | |
833 } | |
834 | |
835 st->index = st->pint.value; | |
836 st->state = sw_value_len; | |
837 break; | |
838 | |
839 case sw_value_len: | |
840 | |
841 st->literal.huffman = (ch & 0x80) ? 1 : 0; | |
842 st->state = sw_read_value_len; | |
843 | |
844 /* fall through */ | |
845 | |
846 case sw_read_value_len: | |
847 | |
848 if (ngx_http_v3_parse_prefix_int(c, &st->pint, 7, ch) != NGX_DONE) { | |
849 break; | |
850 } | |
851 | |
852 st->literal.length = st->pint.value; | |
853 if (st->literal.length == 0) { | |
854 goto done; | |
855 } | |
856 | |
857 st->state = sw_value; | |
858 break; | |
859 | |
860 case sw_value: | |
861 | |
862 rc = ngx_http_v3_parse_literal(c, &st->literal, ch); | |
863 | |
864 if (rc == NGX_ERROR) { | |
865 return NGX_ERROR; | |
866 } | |
867 | |
868 if (rc == NGX_DONE) { | |
869 st->value = st->literal.value; | |
870 goto done; | |
871 } | |
872 | |
873 break; | |
874 } | |
875 | |
876 return NGX_AGAIN; | |
877 | |
878 done: | |
879 | |
880 ngx_log_debug2(NGX_LOG_DEBUG_HTTP, c->log, 0, | |
881 "http3 parse header lpbi done dynamic[+%ui] \"%V\"", | |
882 st->index, &st->value); | |
883 | |
8456
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
884 if (ngx_http_v3_parse_lookup(c, 1, st->base + st->index, &st->name, NULL) |
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
885 != NGX_OK) |
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
886 { |
8226 | 887 return NGX_ERROR; |
888 } | |
889 | |
890 st->state = sw_start; | |
891 return NGX_DONE; | |
892 } | |
893 | |
894 | |
8456
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
895 static ngx_int_t |
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
896 ngx_http_v3_parse_lookup(ngx_connection_t *c, ngx_uint_t dynamic, |
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
897 ngx_uint_t index, ngx_str_t *name, ngx_str_t *value) |
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
898 { |
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
899 u_char *p; |
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
900 |
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
901 if (!dynamic) { |
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
902 return ngx_http_v3_lookup_static(c, index, name, value); |
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
903 } |
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
904 |
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
905 if (ngx_http_v3_lookup(c, index, name, value) != NGX_OK) { |
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
906 return NGX_ERROR; |
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
907 } |
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
908 |
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
909 if (name) { |
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
910 p = ngx_pnalloc(c->pool, name->len + 1); |
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
911 if (p == NULL) { |
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
912 return NGX_ERROR; |
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
913 } |
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
914 |
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
915 ngx_memcpy(p, name->data, name->len); |
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
916 p[name->len] = '\0'; |
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
917 name->data = p; |
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
918 } |
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
919 |
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
920 if (value) { |
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
921 p = ngx_pnalloc(c->pool, value->len + 1); |
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
922 if (p == NULL) { |
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
923 return NGX_ERROR; |
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
924 } |
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
925 |
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
926 ngx_memcpy(p, value->data, value->len); |
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
927 p[value->len] = '\0'; |
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
928 value->data = p; |
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
929 } |
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
930 |
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
931 return NGX_OK; |
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
932 } |
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
933 |
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
934 |
8226 | 935 ngx_int_t |
936 ngx_http_v3_parse_control(ngx_connection_t *c, void *data, u_char ch) | |
937 { | |
938 ngx_http_v3_parse_control_t *st = data; | |
939 | |
940 ngx_int_t rc; | |
941 enum { | |
942 sw_start = 0, | |
943 sw_type, | |
944 sw_length, | |
945 sw_settings, | |
946 sw_max_push_id, | |
947 sw_skip | |
948 }; | |
949 | |
950 switch (st->state) { | |
951 | |
952 case sw_start: | |
953 | |
954 ngx_log_debug0(NGX_LOG_DEBUG_HTTP, c->log, 0, "http3 parse control"); | |
955 | |
956 st->state = sw_type; | |
957 | |
958 /* fall through */ | |
959 | |
960 case sw_type: | |
961 | |
962 if (ngx_http_v3_parse_varlen_int(c, &st->vlint, ch) != NGX_DONE) { | |
963 break; | |
964 } | |
965 | |
966 st->type = st->vlint.value; | |
967 | |
968 ngx_log_debug1(NGX_LOG_DEBUG_HTTP, c->log, 0, | |
969 "http3 parse frame type:%ui", st->type); | |
970 | |
971 st->state = sw_length; | |
972 break; | |
973 | |
974 case sw_length: | |
975 | |
976 if (ngx_http_v3_parse_varlen_int(c, &st->vlint, ch) != NGX_DONE) { | |
977 break; | |
978 } | |
979 | |
980 ngx_log_debug1(NGX_LOG_DEBUG_HTTP, c->log, 0, | |
981 "http3 parse frame len:%uL", st->vlint.value); | |
982 | |
983 st->length = st->vlint.value; | |
984 if (st->length == 0) { | |
985 st->state = sw_type; | |
986 break; | |
987 } | |
988 | |
989 switch (st->type) { | |
990 | |
991 case NGX_HTTP_V3_FRAME_SETTINGS: | |
992 st->state = sw_settings; | |
993 break; | |
994 | |
995 case NGX_HTTP_V3_FRAME_MAX_PUSH_ID: | |
996 st->state = sw_max_push_id; | |
997 break; | |
998 | |
999 default: | |
1000 ngx_log_debug0(NGX_LOG_DEBUG_HTTP, c->log, 0, | |
1001 "http3 parse skip unknown frame"); | |
1002 st->state = sw_skip; | |
1003 } | |
1004 | |
1005 break; | |
1006 | |
1007 case sw_settings: | |
1008 | |
1009 rc = ngx_http_v3_parse_settings(c, &st->settings, ch); | |
1010 | |
1011 if (rc == NGX_ERROR) { | |
1012 return NGX_ERROR; | |
1013 } | |
1014 | |
1015 if (--st->length > 0) { | |
1016 break; | |
1017 } | |
1018 | |
1019 if (rc != NGX_DONE) { | |
1020 return NGX_ERROR; | |
1021 } | |
1022 | |
1023 st->state = sw_type; | |
1024 break; | |
1025 | |
1026 case sw_max_push_id: | |
1027 | |
1028 if (ngx_http_v3_parse_varlen_int(c, &st->vlint, ch) != NGX_DONE) { | |
1029 break; | |
1030 } | |
1031 | |
1032 ngx_log_debug1(NGX_LOG_DEBUG_HTTP, c->log, 0, | |
1033 "http3 parse MAX_PUSH_ID:%uL", st->vlint.value); | |
1034 | |
1035 st->state = sw_type; | |
1036 break; | |
1037 | |
1038 case sw_skip: | |
1039 | |
1040 if (--st->length == 0) { | |
1041 st->state = sw_type; | |
1042 } | |
1043 | |
1044 break; | |
1045 } | |
1046 | |
1047 return NGX_AGAIN; | |
1048 } | |
1049 | |
1050 | |
1051 ngx_int_t | |
1052 ngx_http_v3_parse_settings(ngx_connection_t *c, | |
1053 ngx_http_v3_parse_settings_t *st, u_char ch) | |
1054 { | |
1055 enum { | |
1056 sw_start = 0, | |
1057 sw_id, | |
1058 sw_value | |
1059 }; | |
1060 | |
1061 switch (st->state) { | |
1062 | |
1063 case sw_start: | |
1064 | |
1065 ngx_log_debug0(NGX_LOG_DEBUG_HTTP, c->log, 0, "http3 parse settings"); | |
1066 | |
1067 st->state = sw_id; | |
1068 | |
1069 /* fall through */ | |
1070 | |
1071 case sw_id: | |
1072 | |
1073 if (ngx_http_v3_parse_varlen_int(c, &st->vlint, ch) != NGX_DONE) { | |
1074 break; | |
1075 } | |
1076 | |
1077 st->id = st->vlint.value; | |
1078 st->state = sw_value; | |
1079 break; | |
1080 | |
1081 case sw_value: | |
1082 | |
1083 if (ngx_http_v3_parse_varlen_int(c, &st->vlint, ch) != NGX_DONE) { | |
1084 break; | |
1085 } | |
1086 | |
1087 if (ngx_http_v3_set_param(c, st->id, st->vlint.value) != NGX_OK) { | |
1088 return NGX_ERROR; | |
1089 } | |
1090 | |
1091 goto done; | |
1092 } | |
1093 | |
1094 return NGX_AGAIN; | |
1095 | |
1096 done: | |
1097 | |
1098 ngx_log_debug0(NGX_LOG_DEBUG_HTTP, c->log, 0, "http3 parse settings done"); | |
1099 | |
1100 st->state = sw_start; | |
1101 return NGX_DONE; | |
1102 } | |
1103 | |
1104 | |
1105 ngx_int_t | |
1106 ngx_http_v3_parse_encoder(ngx_connection_t *c, void *data, u_char ch) | |
1107 { | |
1108 ngx_http_v3_parse_encoder_t *st = data; | |
1109 | |
1110 ngx_int_t rc; | |
1111 enum { | |
1112 sw_start = 0, | |
1113 sw_inr, | |
1114 sw_iwnr, | |
1115 sw_capacity, | |
1116 sw_duplicate | |
1117 }; | |
1118 | |
1119 if (st->state == sw_start) { | |
1120 | |
1121 ngx_log_debug0(NGX_LOG_DEBUG_HTTP, c->log, 0, | |
1122 "http3 parse encoder instruction"); | |
1123 | |
1124 if (ch & 0x80) { | |
1125 /* Insert With Name Reference */ | |
1126 | |
1127 st->state = sw_inr; | |
1128 | |
1129 } else if (ch & 0x40) { | |
1130 /* Insert Without Name Reference */ | |
1131 | |
1132 st->state = sw_iwnr; | |
1133 | |
1134 } else if (ch & 0x20) { | |
1135 /* Set Dynamic Table Capacity */ | |
1136 | |
1137 st->state = sw_capacity; | |
1138 | |
1139 } else { | |
1140 /* Duplicate */ | |
1141 | |
1142 st->state = sw_duplicate; | |
1143 } | |
1144 } | |
1145 | |
1146 switch (st->state) { | |
1147 | |
1148 case sw_inr: | |
1149 | |
1150 rc = ngx_http_v3_parse_header_inr(c, &st->header, ch); | |
1151 | |
1152 if (rc == NGX_ERROR) { | |
1153 return NGX_ERROR; | |
1154 } | |
1155 | |
1156 if (rc != NGX_DONE) { | |
1157 break; | |
1158 } | |
1159 | |
1160 goto done; | |
1161 | |
1162 case sw_iwnr: | |
1163 | |
1164 rc = ngx_http_v3_parse_header_iwnr(c, &st->header, ch); | |
1165 | |
1166 if (rc == NGX_ERROR) { | |
1167 return NGX_ERROR; | |
1168 } | |
1169 | |
1170 if (rc != NGX_DONE) { | |
1171 break; | |
1172 } | |
1173 | |
1174 goto done; | |
1175 | |
1176 case sw_capacity: | |
1177 | |
1178 if (ngx_http_v3_parse_prefix_int(c, &st->pint, 5, ch) != NGX_DONE) { | |
1179 break; | |
1180 } | |
1181 | |
1182 if (ngx_http_v3_set_capacity(c, st->pint.value) != NGX_OK) { | |
1183 return NGX_ERROR; | |
1184 } | |
1185 | |
1186 goto done; | |
1187 | |
1188 case sw_duplicate: | |
1189 | |
1190 if (ngx_http_v3_parse_prefix_int(c, &st->pint, 5, ch) != NGX_DONE) { | |
1191 break; | |
1192 } | |
1193 | |
1194 if (ngx_http_v3_duplicate(c, st->pint.value) != NGX_OK) { | |
1195 return NGX_ERROR; | |
1196 } | |
1197 | |
1198 goto done; | |
1199 } | |
1200 | |
1201 return NGX_AGAIN; | |
1202 | |
1203 done: | |
1204 | |
1205 ngx_log_debug0(NGX_LOG_DEBUG_HTTP, c->log, 0, | |
1206 "http3 parse encoder instruction done"); | |
1207 | |
1208 st->state = sw_start; | |
8456
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
1209 return NGX_AGAIN; |
8226 | 1210 } |
1211 | |
1212 | |
1213 ngx_int_t | |
1214 ngx_http_v3_parse_header_inr(ngx_connection_t *c, | |
1215 ngx_http_v3_parse_header_t *st, u_char ch) | |
1216 { | |
1217 ngx_int_t rc; | |
1218 enum { | |
1219 sw_start = 0, | |
1220 sw_name_index, | |
1221 sw_value_len, | |
1222 sw_read_value_len, | |
1223 sw_value | |
1224 }; | |
1225 | |
1226 switch (st->state) { | |
1227 | |
1228 case sw_start: | |
1229 | |
1230 ngx_log_debug0(NGX_LOG_DEBUG_HTTP, c->log, 0, "http3 parse header inr"); | |
1231 | |
1232 st->dynamic = (ch & 0x40) ? 0 : 1; | |
1233 st->state = sw_name_index; | |
1234 | |
1235 /* fall through */ | |
1236 | |
1237 case sw_name_index: | |
1238 | |
1239 if (ngx_http_v3_parse_prefix_int(c, &st->pint, 6, ch) != NGX_DONE) { | |
1240 break; | |
1241 } | |
1242 | |
1243 st->index = st->pint.value; | |
1244 st->state = sw_value_len; | |
1245 break; | |
1246 | |
1247 case sw_value_len: | |
1248 | |
1249 st->literal.huffman = (ch & 0x80) ? 1 : 0; | |
1250 st->state = sw_read_value_len; | |
1251 | |
1252 /* fall through */ | |
1253 | |
1254 case sw_read_value_len: | |
1255 | |
1256 if (ngx_http_v3_parse_prefix_int(c, &st->pint, 7, ch) != NGX_DONE) { | |
1257 break; | |
1258 } | |
1259 | |
1260 st->literal.length = st->pint.value; | |
1261 if (st->literal.length == 0) { | |
1262 goto done; | |
1263 } | |
1264 | |
1265 st->state = sw_value; | |
1266 break; | |
1267 | |
1268 case sw_value: | |
1269 | |
1270 rc = ngx_http_v3_parse_literal(c, &st->literal, ch); | |
1271 | |
1272 if (rc == NGX_ERROR) { | |
1273 return NGX_ERROR; | |
1274 } | |
1275 | |
1276 if (rc == NGX_DONE) { | |
1277 st->value = st->literal.value; | |
1278 goto done; | |
1279 } | |
1280 | |
1281 break; | |
1282 } | |
1283 | |
1284 return NGX_AGAIN; | |
1285 | |
1286 done: | |
1287 | |
1288 ngx_log_debug3(NGX_LOG_DEBUG_HTTP, c->log, 0, | |
1289 "http3 parse header inr done %s[%ui] \"%V\"", | |
1290 st->dynamic ? "dynamic" : "static", | |
1291 st->index, &st->value); | |
1292 | |
1293 if (ngx_http_v3_ref_insert(c, st->dynamic, st->index, &st->value) != NGX_OK) | |
1294 { | |
1295 return NGX_ERROR; | |
1296 } | |
1297 | |
1298 st->state = sw_start; | |
1299 return NGX_DONE; | |
1300 } | |
1301 | |
1302 | |
1303 ngx_int_t | |
1304 ngx_http_v3_parse_header_iwnr(ngx_connection_t *c, | |
1305 ngx_http_v3_parse_header_t *st, u_char ch) | |
1306 { | |
1307 ngx_int_t rc; | |
1308 enum { | |
1309 sw_start = 0, | |
1310 sw_name_len, | |
1311 sw_name, | |
1312 sw_value_len, | |
1313 sw_read_value_len, | |
1314 sw_value | |
1315 }; | |
1316 | |
1317 switch (st->state) { | |
1318 | |
1319 case sw_start: | |
1320 | |
1321 ngx_log_debug0(NGX_LOG_DEBUG_HTTP, c->log, 0, | |
1322 "http3 parse header iwnr"); | |
1323 | |
1324 st->literal.huffman = (ch & 0x20) ? 1 : 0; | |
1325 st->state = sw_name_len; | |
1326 | |
1327 /* fall through */ | |
1328 | |
1329 case sw_name_len: | |
1330 | |
1331 if (ngx_http_v3_parse_prefix_int(c, &st->pint, 5, ch) != NGX_DONE) { | |
1332 break; | |
1333 } | |
1334 | |
1335 st->literal.length = st->pint.value; | |
1336 if (st->literal.length == 0) { | |
1337 return NGX_ERROR; | |
1338 } | |
1339 | |
1340 st->state = sw_name; | |
1341 break; | |
1342 | |
1343 case sw_name: | |
1344 | |
1345 rc = ngx_http_v3_parse_literal(c, &st->literal, ch); | |
1346 | |
1347 if (rc == NGX_ERROR) { | |
1348 return NGX_ERROR; | |
1349 } | |
1350 | |
1351 if (rc == NGX_DONE) { | |
1352 st->name = st->literal.value; | |
1353 st->state = sw_value_len; | |
1354 } | |
1355 | |
1356 break; | |
1357 | |
1358 case sw_value_len: | |
1359 | |
1360 st->literal.huffman = (ch & 0x80) ? 1 : 0; | |
1361 st->state = sw_read_value_len; | |
1362 | |
1363 /* fall through */ | |
1364 | |
1365 case sw_read_value_len: | |
1366 | |
1367 if (ngx_http_v3_parse_prefix_int(c, &st->pint, 7, ch) != NGX_DONE) { | |
1368 break; | |
1369 } | |
1370 | |
1371 st->literal.length = st->pint.value; | |
1372 if (st->literal.length == 0) { | |
1373 goto done; | |
1374 } | |
1375 | |
1376 st->state = sw_value; | |
1377 break; | |
1378 | |
1379 case sw_value: | |
1380 | |
1381 rc = ngx_http_v3_parse_literal(c, &st->literal, ch); | |
1382 | |
1383 if (rc == NGX_ERROR) { | |
1384 return NGX_ERROR; | |
1385 } | |
1386 | |
1387 if (rc == NGX_DONE) { | |
1388 st->value = st->literal.value; | |
1389 goto done; | |
1390 } | |
1391 | |
1392 break; | |
1393 } | |
1394 | |
1395 return NGX_AGAIN; | |
1396 | |
1397 done: | |
1398 | |
1399 ngx_log_debug2(NGX_LOG_DEBUG_HTTP, c->log, 0, | |
1400 "http3 parse header iwnr done \"%V\":\"%V\"", | |
1401 &st->name, &st->value); | |
1402 | |
1403 if (ngx_http_v3_insert(c, &st->name, &st->value) != NGX_OK) { | |
1404 return NGX_ERROR; | |
1405 } | |
1406 | |
1407 st->state = sw_start; | |
1408 return NGX_DONE; | |
1409 } | |
1410 | |
1411 | |
1412 ngx_int_t | |
1413 ngx_http_v3_parse_decoder(ngx_connection_t *c, void *data, u_char ch) | |
1414 { | |
1415 ngx_http_v3_parse_decoder_t *st = data; | |
1416 | |
1417 enum { | |
1418 sw_start = 0, | |
1419 sw_ack_header, | |
1420 sw_cancel_stream, | |
1421 sw_inc_insert_count | |
1422 }; | |
1423 | |
1424 if (st->state == sw_start) { | |
1425 | |
1426 ngx_log_debug0(NGX_LOG_DEBUG_HTTP, c->log, 0, | |
1427 "http3 parse decoder instruction"); | |
1428 | |
1429 if (ch & 0x80) { | |
1430 /* Header Acknowledgement */ | |
1431 | |
1432 st->state = sw_ack_header; | |
1433 | |
1434 } else if (ch & 0x40) { | |
1435 /* Stream Cancellation */ | |
1436 | |
1437 st->state = sw_cancel_stream; | |
1438 | |
1439 } else { | |
1440 /* Insert Count Increment */ | |
1441 | |
1442 st->state = sw_inc_insert_count; | |
1443 } | |
1444 } | |
1445 | |
1446 switch (st->state) { | |
1447 | |
1448 case sw_ack_header: | |
1449 | |
1450 if (ngx_http_v3_parse_prefix_int(c, &st->pint, 6, ch) != NGX_DONE) { | |
1451 break; | |
1452 } | |
1453 | |
1454 if (ngx_http_v3_ack_header(c, st->pint.value) != NGX_OK) { | |
1455 return NGX_ERROR; | |
1456 } | |
1457 | |
1458 goto done; | |
1459 | |
1460 case sw_cancel_stream: | |
1461 | |
1462 if (ngx_http_v3_parse_prefix_int(c, &st->pint, 6, ch) != NGX_DONE) { | |
1463 break; | |
1464 } | |
1465 | |
1466 if (ngx_http_v3_cancel_stream(c, st->pint.value) != NGX_OK) { | |
1467 return NGX_ERROR; | |
1468 } | |
1469 | |
1470 goto done; | |
1471 | |
1472 case sw_inc_insert_count: | |
1473 | |
1474 if (ngx_http_v3_parse_prefix_int(c, &st->pint, 6, ch) != NGX_DONE) { | |
1475 break; | |
1476 } | |
1477 | |
1478 if (ngx_http_v3_inc_insert_count(c, st->pint.value) != NGX_OK) { | |
1479 return NGX_ERROR; | |
1480 } | |
1481 | |
1482 goto done; | |
1483 } | |
1484 | |
1485 return NGX_AGAIN; | |
1486 | |
1487 done: | |
1488 | |
1489 ngx_log_debug0(NGX_LOG_DEBUG_HTTP, c->log, 0, | |
1490 "http3 parse decoder instruction done"); | |
1491 | |
1492 st->state = sw_start; | |
8456
c9538aef3211
HTTP/3: refactored dynamic table implementation.
Roman Arutyunyan <arut@nginx.com>
parents:
8455
diff
changeset
|
1493 return NGX_AGAIN; |
8226 | 1494 } |
8295
5649079a41f4
Parsing HTTP/3 request body.
Roman Arutyunyan <arut@nginx.com>
parents:
8227
diff
changeset
|
1495 |
5649079a41f4
Parsing HTTP/3 request body.
Roman Arutyunyan <arut@nginx.com>
parents:
8227
diff
changeset
|
1496 |
5649079a41f4
Parsing HTTP/3 request body.
Roman Arutyunyan <arut@nginx.com>
parents:
8227
diff
changeset
|
1497 ngx_int_t |
5649079a41f4
Parsing HTTP/3 request body.
Roman Arutyunyan <arut@nginx.com>
parents:
8227
diff
changeset
|
1498 ngx_http_v3_parse_data(ngx_connection_t *c, ngx_http_v3_parse_data_t *st, |
5649079a41f4
Parsing HTTP/3 request body.
Roman Arutyunyan <arut@nginx.com>
parents:
8227
diff
changeset
|
1499 u_char ch) |
5649079a41f4
Parsing HTTP/3 request body.
Roman Arutyunyan <arut@nginx.com>
parents:
8227
diff
changeset
|
1500 { |
5649079a41f4
Parsing HTTP/3 request body.
Roman Arutyunyan <arut@nginx.com>
parents:
8227
diff
changeset
|
1501 enum { |
5649079a41f4
Parsing HTTP/3 request body.
Roman Arutyunyan <arut@nginx.com>
parents:
8227
diff
changeset
|
1502 sw_start = 0, |
5649079a41f4
Parsing HTTP/3 request body.
Roman Arutyunyan <arut@nginx.com>
parents:
8227
diff
changeset
|
1503 sw_type, |
5649079a41f4
Parsing HTTP/3 request body.
Roman Arutyunyan <arut@nginx.com>
parents:
8227
diff
changeset
|
1504 sw_length |
5649079a41f4
Parsing HTTP/3 request body.
Roman Arutyunyan <arut@nginx.com>
parents:
8227
diff
changeset
|
1505 }; |
5649079a41f4
Parsing HTTP/3 request body.
Roman Arutyunyan <arut@nginx.com>
parents:
8227
diff
changeset
|
1506 |
5649079a41f4
Parsing HTTP/3 request body.
Roman Arutyunyan <arut@nginx.com>
parents:
8227
diff
changeset
|
1507 switch (st->state) { |
5649079a41f4
Parsing HTTP/3 request body.
Roman Arutyunyan <arut@nginx.com>
parents:
8227
diff
changeset
|
1508 |
5649079a41f4
Parsing HTTP/3 request body.
Roman Arutyunyan <arut@nginx.com>
parents:
8227
diff
changeset
|
1509 case sw_start: |
5649079a41f4
Parsing HTTP/3 request body.
Roman Arutyunyan <arut@nginx.com>
parents:
8227
diff
changeset
|
1510 |
5649079a41f4
Parsing HTTP/3 request body.
Roman Arutyunyan <arut@nginx.com>
parents:
8227
diff
changeset
|
1511 ngx_log_debug0(NGX_LOG_DEBUG_HTTP, c->log, 0, "http3 parse data"); |
5649079a41f4
Parsing HTTP/3 request body.
Roman Arutyunyan <arut@nginx.com>
parents:
8227
diff
changeset
|
1512 |
5649079a41f4
Parsing HTTP/3 request body.
Roman Arutyunyan <arut@nginx.com>
parents:
8227
diff
changeset
|
1513 st->state = sw_type; |
5649079a41f4
Parsing HTTP/3 request body.
Roman Arutyunyan <arut@nginx.com>
parents:
8227
diff
changeset
|
1514 |
5649079a41f4
Parsing HTTP/3 request body.
Roman Arutyunyan <arut@nginx.com>
parents:
8227
diff
changeset
|
1515 /* fall through */ |
5649079a41f4
Parsing HTTP/3 request body.
Roman Arutyunyan <arut@nginx.com>
parents:
8227
diff
changeset
|
1516 |
5649079a41f4
Parsing HTTP/3 request body.
Roman Arutyunyan <arut@nginx.com>
parents:
8227
diff
changeset
|
1517 case sw_type: |
5649079a41f4
Parsing HTTP/3 request body.
Roman Arutyunyan <arut@nginx.com>
parents:
8227
diff
changeset
|
1518 |
5649079a41f4
Parsing HTTP/3 request body.
Roman Arutyunyan <arut@nginx.com>
parents:
8227
diff
changeset
|
1519 if (ngx_http_v3_parse_varlen_int(c, &st->vlint, ch) != NGX_DONE) { |
5649079a41f4
Parsing HTTP/3 request body.
Roman Arutyunyan <arut@nginx.com>
parents:
8227
diff
changeset
|
1520 break; |
5649079a41f4
Parsing HTTP/3 request body.
Roman Arutyunyan <arut@nginx.com>
parents:
8227
diff
changeset
|
1521 } |
5649079a41f4
Parsing HTTP/3 request body.
Roman Arutyunyan <arut@nginx.com>
parents:
8227
diff
changeset
|
1522 |
5649079a41f4
Parsing HTTP/3 request body.
Roman Arutyunyan <arut@nginx.com>
parents:
8227
diff
changeset
|
1523 if (st->vlint.value != NGX_HTTP_V3_FRAME_DATA) { |
5649079a41f4
Parsing HTTP/3 request body.
Roman Arutyunyan <arut@nginx.com>
parents:
8227
diff
changeset
|
1524 return NGX_ERROR; |
5649079a41f4
Parsing HTTP/3 request body.
Roman Arutyunyan <arut@nginx.com>
parents:
8227
diff
changeset
|
1525 } |
5649079a41f4
Parsing HTTP/3 request body.
Roman Arutyunyan <arut@nginx.com>
parents:
8227
diff
changeset
|
1526 |
5649079a41f4
Parsing HTTP/3 request body.
Roman Arutyunyan <arut@nginx.com>
parents:
8227
diff
changeset
|
1527 st->state = sw_length; |
5649079a41f4
Parsing HTTP/3 request body.
Roman Arutyunyan <arut@nginx.com>
parents:
8227
diff
changeset
|
1528 break; |
5649079a41f4
Parsing HTTP/3 request body.
Roman Arutyunyan <arut@nginx.com>
parents:
8227
diff
changeset
|
1529 |
5649079a41f4
Parsing HTTP/3 request body.
Roman Arutyunyan <arut@nginx.com>
parents:
8227
diff
changeset
|
1530 case sw_length: |
5649079a41f4
Parsing HTTP/3 request body.
Roman Arutyunyan <arut@nginx.com>
parents:
8227
diff
changeset
|
1531 |
5649079a41f4
Parsing HTTP/3 request body.
Roman Arutyunyan <arut@nginx.com>
parents:
8227
diff
changeset
|
1532 if (ngx_http_v3_parse_varlen_int(c, &st->vlint, ch) != NGX_DONE) { |
5649079a41f4
Parsing HTTP/3 request body.
Roman Arutyunyan <arut@nginx.com>
parents:
8227
diff
changeset
|
1533 break; |
5649079a41f4
Parsing HTTP/3 request body.
Roman Arutyunyan <arut@nginx.com>
parents:
8227
diff
changeset
|
1534 } |
5649079a41f4
Parsing HTTP/3 request body.
Roman Arutyunyan <arut@nginx.com>
parents:
8227
diff
changeset
|
1535 |
5649079a41f4
Parsing HTTP/3 request body.
Roman Arutyunyan <arut@nginx.com>
parents:
8227
diff
changeset
|
1536 st->length = st->vlint.value; |
5649079a41f4
Parsing HTTP/3 request body.
Roman Arutyunyan <arut@nginx.com>
parents:
8227
diff
changeset
|
1537 |
5649079a41f4
Parsing HTTP/3 request body.
Roman Arutyunyan <arut@nginx.com>
parents:
8227
diff
changeset
|
1538 ngx_log_debug1(NGX_LOG_DEBUG_HTTP, c->log, 0, |
5649079a41f4
Parsing HTTP/3 request body.
Roman Arutyunyan <arut@nginx.com>
parents:
8227
diff
changeset
|
1539 "http3 parse data frame len:%ui", st->length); |
5649079a41f4
Parsing HTTP/3 request body.
Roman Arutyunyan <arut@nginx.com>
parents:
8227
diff
changeset
|
1540 |
5649079a41f4
Parsing HTTP/3 request body.
Roman Arutyunyan <arut@nginx.com>
parents:
8227
diff
changeset
|
1541 goto done; |
5649079a41f4
Parsing HTTP/3 request body.
Roman Arutyunyan <arut@nginx.com>
parents:
8227
diff
changeset
|
1542 } |
5649079a41f4
Parsing HTTP/3 request body.
Roman Arutyunyan <arut@nginx.com>
parents:
8227
diff
changeset
|
1543 |
5649079a41f4
Parsing HTTP/3 request body.
Roman Arutyunyan <arut@nginx.com>
parents:
8227
diff
changeset
|
1544 return NGX_AGAIN; |
5649079a41f4
Parsing HTTP/3 request body.
Roman Arutyunyan <arut@nginx.com>
parents:
8227
diff
changeset
|
1545 |
5649079a41f4
Parsing HTTP/3 request body.
Roman Arutyunyan <arut@nginx.com>
parents:
8227
diff
changeset
|
1546 done: |
5649079a41f4
Parsing HTTP/3 request body.
Roman Arutyunyan <arut@nginx.com>
parents:
8227
diff
changeset
|
1547 |
5649079a41f4
Parsing HTTP/3 request body.
Roman Arutyunyan <arut@nginx.com>
parents:
8227
diff
changeset
|
1548 ngx_log_debug0(NGX_LOG_DEBUG_HTTP, c->log, 0, "http3 parse data done"); |
5649079a41f4
Parsing HTTP/3 request body.
Roman Arutyunyan <arut@nginx.com>
parents:
8227
diff
changeset
|
1549 |
5649079a41f4
Parsing HTTP/3 request body.
Roman Arutyunyan <arut@nginx.com>
parents:
8227
diff
changeset
|
1550 st->state = sw_start; |
5649079a41f4
Parsing HTTP/3 request body.
Roman Arutyunyan <arut@nginx.com>
parents:
8227
diff
changeset
|
1551 return NGX_DONE; |
5649079a41f4
Parsing HTTP/3 request body.
Roman Arutyunyan <arut@nginx.com>
parents:
8227
diff
changeset
|
1552 } |