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