Imported Upstream version 1.0.0
[platform/upstream/nghttp2.git] / lib / nghttp2_helper.c
1 /*
2  * nghttp2 - HTTP/2 C Library
3  *
4  * Copyright (c) 2012 Tatsuhiro Tsujikawa
5  *
6  * Permission is hereby granted, free of charge, to any person obtaining
7  * a copy of this software and associated documentation files (the
8  * "Software"), to deal in the Software without restriction, including
9  * without limitation the rights to use, copy, modify, merge, publish,
10  * distribute, sublicense, and/or sell copies of the Software, and to
11  * permit persons to whom the Software is furnished to do so, subject to
12  * the following conditions:
13  *
14  * The above copyright notice and this permission notice shall be
15  * included in all copies or substantial portions of the Software.
16  *
17  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
18  * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
19  * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
20  * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
21  * LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
22  * OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
23  * WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
24  */
25 #include "nghttp2_helper.h"
26
27 #include <assert.h>
28 #include <string.h>
29
30 #include "nghttp2_net.h"
31
32 void nghttp2_put_uint16be(uint8_t *buf, uint16_t n) {
33   uint16_t x = htons(n);
34   memcpy(buf, &x, sizeof(uint16_t));
35 }
36
37 void nghttp2_put_uint32be(uint8_t *buf, uint32_t n) {
38   uint32_t x = htonl(n);
39   memcpy(buf, &x, sizeof(uint32_t));
40 }
41
42 uint16_t nghttp2_get_uint16(const uint8_t *data) {
43   uint16_t n;
44   memcpy(&n, data, sizeof(uint16_t));
45   return ntohs(n);
46 }
47
48 uint32_t nghttp2_get_uint32(const uint8_t *data) {
49   uint32_t n;
50   memcpy(&n, data, sizeof(uint32_t));
51   return ntohl(n);
52 }
53
54 void *nghttp2_memdup(const void *src, size_t n, nghttp2_mem *mem) {
55   void *dest;
56
57   if (n == 0) {
58     return NULL;
59   }
60
61   dest = nghttp2_mem_malloc(mem, n);
62   if (dest == NULL) {
63     return NULL;
64   }
65   memcpy(dest, src, n);
66   return dest;
67 }
68
69 /* Generated by gendowncasetbl.py */
70 static const int DOWNCASE_TBL[] = {
71     0 /* NUL  */,   1 /* SOH  */,   2 /* STX  */,   3 /* ETX  */,
72     4 /* EOT  */,   5 /* ENQ  */,   6 /* ACK  */,   7 /* BEL  */,
73     8 /* BS   */,   9 /* HT   */,   10 /* LF   */,  11 /* VT   */,
74     12 /* FF   */,  13 /* CR   */,  14 /* SO   */,  15 /* SI   */,
75     16 /* DLE  */,  17 /* DC1  */,  18 /* DC2  */,  19 /* DC3  */,
76     20 /* DC4  */,  21 /* NAK  */,  22 /* SYN  */,  23 /* ETB  */,
77     24 /* CAN  */,  25 /* EM   */,  26 /* SUB  */,  27 /* ESC  */,
78     28 /* FS   */,  29 /* GS   */,  30 /* RS   */,  31 /* US   */,
79     32 /* SPC  */,  33 /* !    */,  34 /* "    */,  35 /* #    */,
80     36 /* $    */,  37 /* %    */,  38 /* &    */,  39 /* '    */,
81     40 /* (    */,  41 /* )    */,  42 /* *    */,  43 /* +    */,
82     44 /* ,    */,  45 /* -    */,  46 /* .    */,  47 /* /    */,
83     48 /* 0    */,  49 /* 1    */,  50 /* 2    */,  51 /* 3    */,
84     52 /* 4    */,  53 /* 5    */,  54 /* 6    */,  55 /* 7    */,
85     56 /* 8    */,  57 /* 9    */,  58 /* :    */,  59 /* ;    */,
86     60 /* <    */,  61 /* =    */,  62 /* >    */,  63 /* ?    */,
87     64 /* @    */,  97 /* A    */,  98 /* B    */,  99 /* C    */,
88     100 /* D    */, 101 /* E    */, 102 /* F    */, 103 /* G    */,
89     104 /* H    */, 105 /* I    */, 106 /* J    */, 107 /* K    */,
90     108 /* L    */, 109 /* M    */, 110 /* N    */, 111 /* O    */,
91     112 /* P    */, 113 /* Q    */, 114 /* R    */, 115 /* S    */,
92     116 /* T    */, 117 /* U    */, 118 /* V    */, 119 /* W    */,
93     120 /* X    */, 121 /* Y    */, 122 /* Z    */, 91 /* [    */,
94     92 /* \    */,  93 /* ]    */,  94 /* ^    */,  95 /* _    */,
95     96 /* `    */,  97 /* a    */,  98 /* b    */,  99 /* c    */,
96     100 /* d    */, 101 /* e    */, 102 /* f    */, 103 /* g    */,
97     104 /* h    */, 105 /* i    */, 106 /* j    */, 107 /* k    */,
98     108 /* l    */, 109 /* m    */, 110 /* n    */, 111 /* o    */,
99     112 /* p    */, 113 /* q    */, 114 /* r    */, 115 /* s    */,
100     116 /* t    */, 117 /* u    */, 118 /* v    */, 119 /* w    */,
101     120 /* x    */, 121 /* y    */, 122 /* z    */, 123 /* {    */,
102     124 /* |    */, 125 /* }    */, 126 /* ~    */, 127 /* DEL  */,
103     128 /* 0x80 */, 129 /* 0x81 */, 130 /* 0x82 */, 131 /* 0x83 */,
104     132 /* 0x84 */, 133 /* 0x85 */, 134 /* 0x86 */, 135 /* 0x87 */,
105     136 /* 0x88 */, 137 /* 0x89 */, 138 /* 0x8a */, 139 /* 0x8b */,
106     140 /* 0x8c */, 141 /* 0x8d */, 142 /* 0x8e */, 143 /* 0x8f */,
107     144 /* 0x90 */, 145 /* 0x91 */, 146 /* 0x92 */, 147 /* 0x93 */,
108     148 /* 0x94 */, 149 /* 0x95 */, 150 /* 0x96 */, 151 /* 0x97 */,
109     152 /* 0x98 */, 153 /* 0x99 */, 154 /* 0x9a */, 155 /* 0x9b */,
110     156 /* 0x9c */, 157 /* 0x9d */, 158 /* 0x9e */, 159 /* 0x9f */,
111     160 /* 0xa0 */, 161 /* 0xa1 */, 162 /* 0xa2 */, 163 /* 0xa3 */,
112     164 /* 0xa4 */, 165 /* 0xa5 */, 166 /* 0xa6 */, 167 /* 0xa7 */,
113     168 /* 0xa8 */, 169 /* 0xa9 */, 170 /* 0xaa */, 171 /* 0xab */,
114     172 /* 0xac */, 173 /* 0xad */, 174 /* 0xae */, 175 /* 0xaf */,
115     176 /* 0xb0 */, 177 /* 0xb1 */, 178 /* 0xb2 */, 179 /* 0xb3 */,
116     180 /* 0xb4 */, 181 /* 0xb5 */, 182 /* 0xb6 */, 183 /* 0xb7 */,
117     184 /* 0xb8 */, 185 /* 0xb9 */, 186 /* 0xba */, 187 /* 0xbb */,
118     188 /* 0xbc */, 189 /* 0xbd */, 190 /* 0xbe */, 191 /* 0xbf */,
119     192 /* 0xc0 */, 193 /* 0xc1 */, 194 /* 0xc2 */, 195 /* 0xc3 */,
120     196 /* 0xc4 */, 197 /* 0xc5 */, 198 /* 0xc6 */, 199 /* 0xc7 */,
121     200 /* 0xc8 */, 201 /* 0xc9 */, 202 /* 0xca */, 203 /* 0xcb */,
122     204 /* 0xcc */, 205 /* 0xcd */, 206 /* 0xce */, 207 /* 0xcf */,
123     208 /* 0xd0 */, 209 /* 0xd1 */, 210 /* 0xd2 */, 211 /* 0xd3 */,
124     212 /* 0xd4 */, 213 /* 0xd5 */, 214 /* 0xd6 */, 215 /* 0xd7 */,
125     216 /* 0xd8 */, 217 /* 0xd9 */, 218 /* 0xda */, 219 /* 0xdb */,
126     220 /* 0xdc */, 221 /* 0xdd */, 222 /* 0xde */, 223 /* 0xdf */,
127     224 /* 0xe0 */, 225 /* 0xe1 */, 226 /* 0xe2 */, 227 /* 0xe3 */,
128     228 /* 0xe4 */, 229 /* 0xe5 */, 230 /* 0xe6 */, 231 /* 0xe7 */,
129     232 /* 0xe8 */, 233 /* 0xe9 */, 234 /* 0xea */, 235 /* 0xeb */,
130     236 /* 0xec */, 237 /* 0xed */, 238 /* 0xee */, 239 /* 0xef */,
131     240 /* 0xf0 */, 241 /* 0xf1 */, 242 /* 0xf2 */, 243 /* 0xf3 */,
132     244 /* 0xf4 */, 245 /* 0xf5 */, 246 /* 0xf6 */, 247 /* 0xf7 */,
133     248 /* 0xf8 */, 249 /* 0xf9 */, 250 /* 0xfa */, 251 /* 0xfb */,
134     252 /* 0xfc */, 253 /* 0xfd */, 254 /* 0xfe */, 255 /* 0xff */,
135 };
136
137 void nghttp2_downcase(uint8_t *s, size_t len) {
138   size_t i;
139   for (i = 0; i < len; ++i) {
140     s[i] = DOWNCASE_TBL[s[i]];
141   }
142 }
143
144 /*
145  *   local_window_size
146  *   ^  *
147  *   |  *    recv_window_size
148  *   |  *  * ^
149  *   |  *  * |
150  *  0+++++++++
151  *   |  *  *   \
152  *   |  *  *   | This rage is hidden in flow control.  But it must be
153  *   v  *  *   / kept in order to restore it when window size is enlarged.
154  *   recv_reduction
155  *   (+ for negative direction)
156  *
157  *   recv_window_size could be negative if we decrease
158  *   local_window_size more than recv_window_size:
159  *
160  *   local_window_size
161  *   ^  *
162  *   |  *
163  *   |  *
164  *   0++++++++
165  *   |  *    ^ recv_window_size (negative)
166  *   |  *    |
167  *   v  *  *
168  *   recv_reduction
169  */
170 int nghttp2_adjust_local_window_size(int32_t *local_window_size_ptr,
171                                      int32_t *recv_window_size_ptr,
172                                      int32_t *recv_reduction_ptr,
173                                      int32_t *delta_ptr) {
174   if (*delta_ptr > 0) {
175     int32_t recv_reduction_delta;
176     int32_t delta;
177     int32_t new_recv_window_size =
178         nghttp2_max(0, *recv_window_size_ptr) - *delta_ptr;
179
180     if (new_recv_window_size >= 0) {
181       *recv_window_size_ptr = new_recv_window_size;
182       return 0;
183     }
184
185     delta = -new_recv_window_size;
186
187     /* The delta size is strictly more than received bytes. Increase
188        local_window_size by that difference |delta|. */
189     if (*local_window_size_ptr > NGHTTP2_MAX_WINDOW_SIZE - delta) {
190       return NGHTTP2_ERR_FLOW_CONTROL;
191     }
192     *local_window_size_ptr += delta;
193     /* If there is recv_reduction due to earlier window_size
194        reduction, we have to adjust it too. */
195     recv_reduction_delta = nghttp2_min(*recv_reduction_ptr, delta);
196     *recv_reduction_ptr -= recv_reduction_delta;
197     if (*recv_window_size_ptr < 0) {
198       *recv_window_size_ptr += recv_reduction_delta;
199     } else {
200       /* If *recv_window_size_ptr > 0, then those bytes are going to
201          be returned to the remote peer (by WINDOW_UPDATE with the
202          adjusted *delta_ptr), so it is effectively 0 now.  We set to
203          *recv_reduction_delta, because caller does not take into
204          account it in *delta_ptr. */
205       *recv_window_size_ptr = recv_reduction_delta;
206     }
207     /* recv_reduction_delta must be paied from *delta_ptr, since it
208        was added in window size reduction (see below). */
209     *delta_ptr -= recv_reduction_delta;
210
211     return 0;
212   }
213
214   if (*local_window_size_ptr + *delta_ptr < 0 ||
215       *recv_window_size_ptr < INT32_MIN - *delta_ptr ||
216       *recv_reduction_ptr > INT32_MAX + *delta_ptr) {
217     return NGHTTP2_ERR_FLOW_CONTROL;
218   }
219   /* Decreasing local window size. Note that we achieve this without
220      noticing to the remote peer. To do this, we cut
221      recv_window_size by -delta. This means that we don't send
222      WINDOW_UPDATE for -delta bytes. */
223   *local_window_size_ptr += *delta_ptr;
224   *recv_window_size_ptr += *delta_ptr;
225   *recv_reduction_ptr -= *delta_ptr;
226   *delta_ptr = 0;
227
228   return 0;
229 }
230
231 int nghttp2_should_send_window_update(int32_t local_window_size,
232                                       int32_t recv_window_size) {
233   return recv_window_size >= local_window_size / 2;
234 }
235
236 const char *nghttp2_strerror(int error_code) {
237   switch (error_code) {
238   case 0:
239     return "Success";
240   case NGHTTP2_ERR_INVALID_ARGUMENT:
241     return "Invalid argument";
242   case NGHTTP2_ERR_BUFFER_ERROR:
243     return "Out of buffer space";
244   case NGHTTP2_ERR_UNSUPPORTED_VERSION:
245     return "Unsupported SPDY version";
246   case NGHTTP2_ERR_WOULDBLOCK:
247     return "Operation would block";
248   case NGHTTP2_ERR_PROTO:
249     return "Protocol error";
250   case NGHTTP2_ERR_INVALID_FRAME:
251     return "Invalid frame octets";
252   case NGHTTP2_ERR_EOF:
253     return "EOF";
254   case NGHTTP2_ERR_DEFERRED:
255     return "Data transfer deferred";
256   case NGHTTP2_ERR_STREAM_ID_NOT_AVAILABLE:
257     return "No more Stream ID available";
258   case NGHTTP2_ERR_STREAM_CLOSED:
259     return "Stream was already closed or invalid";
260   case NGHTTP2_ERR_STREAM_CLOSING:
261     return "Stream is closing";
262   case NGHTTP2_ERR_STREAM_SHUT_WR:
263     return "The transmission is not allowed for this stream";
264   case NGHTTP2_ERR_INVALID_STREAM_ID:
265     return "Stream ID is invalid";
266   case NGHTTP2_ERR_INVALID_STREAM_STATE:
267     return "Invalid stream state";
268   case NGHTTP2_ERR_DEFERRED_DATA_EXIST:
269     return "Another DATA frame has already been deferred";
270   case NGHTTP2_ERR_START_STREAM_NOT_ALLOWED:
271     return "request HEADERS is not allowed";
272   case NGHTTP2_ERR_GOAWAY_ALREADY_SENT:
273     return "GOAWAY has already been sent";
274   case NGHTTP2_ERR_INVALID_HEADER_BLOCK:
275     return "Invalid header block";
276   case NGHTTP2_ERR_INVALID_STATE:
277     return "Invalid state";
278   case NGHTTP2_ERR_TEMPORAL_CALLBACK_FAILURE:
279     return "The user callback function failed due to the temporal error";
280   case NGHTTP2_ERR_FRAME_SIZE_ERROR:
281     return "The length of the frame is invalid";
282   case NGHTTP2_ERR_HEADER_COMP:
283     return "Header compression/decompression error";
284   case NGHTTP2_ERR_FLOW_CONTROL:
285     return "Flow control error";
286   case NGHTTP2_ERR_INSUFF_BUFSIZE:
287     return "Insufficient buffer size given to function";
288   case NGHTTP2_ERR_PAUSE:
289     return "Callback was paused by the application";
290   case NGHTTP2_ERR_TOO_MANY_INFLIGHT_SETTINGS:
291     return "Too many inflight SETTINGS";
292   case NGHTTP2_ERR_PUSH_DISABLED:
293     return "Server push is disabled by peer";
294   case NGHTTP2_ERR_DATA_EXIST:
295     return "DATA frame already exists";
296   case NGHTTP2_ERR_SESSION_CLOSING:
297     return "The current session is closing";
298   case NGHTTP2_ERR_HTTP_HEADER:
299     return "Invalid HTTP header field was received";
300   case NGHTTP2_ERR_HTTP_MESSAGING:
301     return "Violation in HTTP messaging rule";
302   case NGHTTP2_ERR_REFUSED_STREAM:
303     return "Stream was refused";
304   case NGHTTP2_ERR_INTERNAL:
305     return "Internal error";
306   case NGHTTP2_ERR_NOMEM:
307     return "Out of memory";
308   case NGHTTP2_ERR_CALLBACK_FAILURE:
309     return "The user callback function failed";
310   case NGHTTP2_ERR_BAD_CLIENT_MAGIC:
311     return "Received bad clinet magic byte string";
312   default:
313     return "Unknown error code";
314   }
315 }
316
317 /* Generated by gennmchartbl.py */
318 static int VALID_HD_NAME_CHARS[] = {
319     0 /* NUL  */, 0 /* SOH  */, 0 /* STX  */, 0 /* ETX  */, 0 /* EOT  */,
320     0 /* ENQ  */, 0 /* ACK  */, 0 /* BEL  */, 0 /* BS   */, 0 /* HT   */,
321     0 /* LF   */, 0 /* VT   */, 0 /* FF   */, 0 /* CR   */, 0 /* SO   */,
322     0 /* SI   */, 0 /* DLE  */, 0 /* DC1  */, 0 /* DC2  */, 0 /* DC3  */,
323     0 /* DC4  */, 0 /* NAK  */, 0 /* SYN  */, 0 /* ETB  */, 0 /* CAN  */,
324     0 /* EM   */, 0 /* SUB  */, 0 /* ESC  */, 0 /* FS   */, 0 /* GS   */,
325     0 /* RS   */, 0 /* US   */, 0 /* SPC  */, 1 /* !    */, 0 /* "    */,
326     1 /* #    */, 1 /* $    */, 1 /* %    */, 1 /* &    */, 1 /* '    */,
327     0 /* (    */, 0 /* )    */, 1 /* *    */, 1 /* +    */, 0 /* ,    */,
328     1 /* -    */, 1 /* .    */, 0 /* /    */, 1 /* 0    */, 1 /* 1    */,
329     1 /* 2    */, 1 /* 3    */, 1 /* 4    */, 1 /* 5    */, 1 /* 6    */,
330     1 /* 7    */, 1 /* 8    */, 1 /* 9    */, 0 /* :    */, 0 /* ;    */,
331     0 /* <    */, 0 /* =    */, 0 /* >    */, 0 /* ?    */, 0 /* @    */,
332     0 /* A    */, 0 /* B    */, 0 /* C    */, 0 /* D    */, 0 /* E    */,
333     0 /* F    */, 0 /* G    */, 0 /* H    */, 0 /* I    */, 0 /* J    */,
334     0 /* K    */, 0 /* L    */, 0 /* M    */, 0 /* N    */, 0 /* O    */,
335     0 /* P    */, 0 /* Q    */, 0 /* R    */, 0 /* S    */, 0 /* T    */,
336     0 /* U    */, 0 /* V    */, 0 /* W    */, 0 /* X    */, 0 /* Y    */,
337     0 /* Z    */, 0 /* [    */, 0 /* \    */, 0 /* ]    */, 1 /* ^    */,
338     1 /* _    */, 1 /* `    */, 1 /* a    */, 1 /* b    */, 1 /* c    */,
339     1 /* d    */, 1 /* e    */, 1 /* f    */, 1 /* g    */, 1 /* h    */,
340     1 /* i    */, 1 /* j    */, 1 /* k    */, 1 /* l    */, 1 /* m    */,
341     1 /* n    */, 1 /* o    */, 1 /* p    */, 1 /* q    */, 1 /* r    */,
342     1 /* s    */, 1 /* t    */, 1 /* u    */, 1 /* v    */, 1 /* w    */,
343     1 /* x    */, 1 /* y    */, 1 /* z    */, 0 /* {    */, 1 /* |    */,
344     0 /* }    */, 1 /* ~    */, 0 /* DEL  */, 0 /* 0x80 */, 0 /* 0x81 */,
345     0 /* 0x82 */, 0 /* 0x83 */, 0 /* 0x84 */, 0 /* 0x85 */, 0 /* 0x86 */,
346     0 /* 0x87 */, 0 /* 0x88 */, 0 /* 0x89 */, 0 /* 0x8a */, 0 /* 0x8b */,
347     0 /* 0x8c */, 0 /* 0x8d */, 0 /* 0x8e */, 0 /* 0x8f */, 0 /* 0x90 */,
348     0 /* 0x91 */, 0 /* 0x92 */, 0 /* 0x93 */, 0 /* 0x94 */, 0 /* 0x95 */,
349     0 /* 0x96 */, 0 /* 0x97 */, 0 /* 0x98 */, 0 /* 0x99 */, 0 /* 0x9a */,
350     0 /* 0x9b */, 0 /* 0x9c */, 0 /* 0x9d */, 0 /* 0x9e */, 0 /* 0x9f */,
351     0 /* 0xa0 */, 0 /* 0xa1 */, 0 /* 0xa2 */, 0 /* 0xa3 */, 0 /* 0xa4 */,
352     0 /* 0xa5 */, 0 /* 0xa6 */, 0 /* 0xa7 */, 0 /* 0xa8 */, 0 /* 0xa9 */,
353     0 /* 0xaa */, 0 /* 0xab */, 0 /* 0xac */, 0 /* 0xad */, 0 /* 0xae */,
354     0 /* 0xaf */, 0 /* 0xb0 */, 0 /* 0xb1 */, 0 /* 0xb2 */, 0 /* 0xb3 */,
355     0 /* 0xb4 */, 0 /* 0xb5 */, 0 /* 0xb6 */, 0 /* 0xb7 */, 0 /* 0xb8 */,
356     0 /* 0xb9 */, 0 /* 0xba */, 0 /* 0xbb */, 0 /* 0xbc */, 0 /* 0xbd */,
357     0 /* 0xbe */, 0 /* 0xbf */, 0 /* 0xc0 */, 0 /* 0xc1 */, 0 /* 0xc2 */,
358     0 /* 0xc3 */, 0 /* 0xc4 */, 0 /* 0xc5 */, 0 /* 0xc6 */, 0 /* 0xc7 */,
359     0 /* 0xc8 */, 0 /* 0xc9 */, 0 /* 0xca */, 0 /* 0xcb */, 0 /* 0xcc */,
360     0 /* 0xcd */, 0 /* 0xce */, 0 /* 0xcf */, 0 /* 0xd0 */, 0 /* 0xd1 */,
361     0 /* 0xd2 */, 0 /* 0xd3 */, 0 /* 0xd4 */, 0 /* 0xd5 */, 0 /* 0xd6 */,
362     0 /* 0xd7 */, 0 /* 0xd8 */, 0 /* 0xd9 */, 0 /* 0xda */, 0 /* 0xdb */,
363     0 /* 0xdc */, 0 /* 0xdd */, 0 /* 0xde */, 0 /* 0xdf */, 0 /* 0xe0 */,
364     0 /* 0xe1 */, 0 /* 0xe2 */, 0 /* 0xe3 */, 0 /* 0xe4 */, 0 /* 0xe5 */,
365     0 /* 0xe6 */, 0 /* 0xe7 */, 0 /* 0xe8 */, 0 /* 0xe9 */, 0 /* 0xea */,
366     0 /* 0xeb */, 0 /* 0xec */, 0 /* 0xed */, 0 /* 0xee */, 0 /* 0xef */,
367     0 /* 0xf0 */, 0 /* 0xf1 */, 0 /* 0xf2 */, 0 /* 0xf3 */, 0 /* 0xf4 */,
368     0 /* 0xf5 */, 0 /* 0xf6 */, 0 /* 0xf7 */, 0 /* 0xf8 */, 0 /* 0xf9 */,
369     0 /* 0xfa */, 0 /* 0xfb */, 0 /* 0xfc */, 0 /* 0xfd */, 0 /* 0xfe */,
370     0 /* 0xff */
371 };
372
373 int nghttp2_check_header_name(const uint8_t *name, size_t len) {
374   const uint8_t *last;
375   if (len == 0) {
376     return 0;
377   }
378   if (*name == ':') {
379     if (len == 1) {
380       return 0;
381     }
382     ++name;
383     --len;
384   }
385   for (last = name + len; name != last; ++name) {
386     if (!VALID_HD_NAME_CHARS[*name]) {
387       return 0;
388     }
389   }
390   return 1;
391 }
392
393 /* Generated by genvchartbl.py */
394 static int VALID_HD_VALUE_CHARS[] = {
395     0 /* NUL  */, 0 /* SOH  */, 0 /* STX  */, 0 /* ETX  */, 0 /* EOT  */,
396     0 /* ENQ  */, 0 /* ACK  */, 0 /* BEL  */, 0 /* BS   */, 1 /* HT   */,
397     0 /* LF   */, 0 /* VT   */, 0 /* FF   */, 0 /* CR   */, 0 /* SO   */,
398     0 /* SI   */, 0 /* DLE  */, 0 /* DC1  */, 0 /* DC2  */, 0 /* DC3  */,
399     0 /* DC4  */, 0 /* NAK  */, 0 /* SYN  */, 0 /* ETB  */, 0 /* CAN  */,
400     0 /* EM   */, 0 /* SUB  */, 0 /* ESC  */, 0 /* FS   */, 0 /* GS   */,
401     0 /* RS   */, 0 /* US   */, 1 /* SPC  */, 1 /* !    */, 1 /* "    */,
402     1 /* #    */, 1 /* $    */, 1 /* %    */, 1 /* &    */, 1 /* '    */,
403     1 /* (    */, 1 /* )    */, 1 /* *    */, 1 /* +    */, 1 /* ,    */,
404     1 /* -    */, 1 /* .    */, 1 /* /    */, 1 /* 0    */, 1 /* 1    */,
405     1 /* 2    */, 1 /* 3    */, 1 /* 4    */, 1 /* 5    */, 1 /* 6    */,
406     1 /* 7    */, 1 /* 8    */, 1 /* 9    */, 1 /* :    */, 1 /* ;    */,
407     1 /* <    */, 1 /* =    */, 1 /* >    */, 1 /* ?    */, 1 /* @    */,
408     1 /* A    */, 1 /* B    */, 1 /* C    */, 1 /* D    */, 1 /* E    */,
409     1 /* F    */, 1 /* G    */, 1 /* H    */, 1 /* I    */, 1 /* J    */,
410     1 /* K    */, 1 /* L    */, 1 /* M    */, 1 /* N    */, 1 /* O    */,
411     1 /* P    */, 1 /* Q    */, 1 /* R    */, 1 /* S    */, 1 /* T    */,
412     1 /* U    */, 1 /* V    */, 1 /* W    */, 1 /* X    */, 1 /* Y    */,
413     1 /* Z    */, 1 /* [    */, 1 /* \    */, 1 /* ]    */, 1 /* ^    */,
414     1 /* _    */, 1 /* `    */, 1 /* a    */, 1 /* b    */, 1 /* c    */,
415     1 /* d    */, 1 /* e    */, 1 /* f    */, 1 /* g    */, 1 /* h    */,
416     1 /* i    */, 1 /* j    */, 1 /* k    */, 1 /* l    */, 1 /* m    */,
417     1 /* n    */, 1 /* o    */, 1 /* p    */, 1 /* q    */, 1 /* r    */,
418     1 /* s    */, 1 /* t    */, 1 /* u    */, 1 /* v    */, 1 /* w    */,
419     1 /* x    */, 1 /* y    */, 1 /* z    */, 1 /* {    */, 1 /* |    */,
420     1 /* }    */, 1 /* ~    */, 0 /* DEL  */, 1 /* 0x80 */, 1 /* 0x81 */,
421     1 /* 0x82 */, 1 /* 0x83 */, 1 /* 0x84 */, 1 /* 0x85 */, 1 /* 0x86 */,
422     1 /* 0x87 */, 1 /* 0x88 */, 1 /* 0x89 */, 1 /* 0x8a */, 1 /* 0x8b */,
423     1 /* 0x8c */, 1 /* 0x8d */, 1 /* 0x8e */, 1 /* 0x8f */, 1 /* 0x90 */,
424     1 /* 0x91 */, 1 /* 0x92 */, 1 /* 0x93 */, 1 /* 0x94 */, 1 /* 0x95 */,
425     1 /* 0x96 */, 1 /* 0x97 */, 1 /* 0x98 */, 1 /* 0x99 */, 1 /* 0x9a */,
426     1 /* 0x9b */, 1 /* 0x9c */, 1 /* 0x9d */, 1 /* 0x9e */, 1 /* 0x9f */,
427     1 /* 0xa0 */, 1 /* 0xa1 */, 1 /* 0xa2 */, 1 /* 0xa3 */, 1 /* 0xa4 */,
428     1 /* 0xa5 */, 1 /* 0xa6 */, 1 /* 0xa7 */, 1 /* 0xa8 */, 1 /* 0xa9 */,
429     1 /* 0xaa */, 1 /* 0xab */, 1 /* 0xac */, 1 /* 0xad */, 1 /* 0xae */,
430     1 /* 0xaf */, 1 /* 0xb0 */, 1 /* 0xb1 */, 1 /* 0xb2 */, 1 /* 0xb3 */,
431     1 /* 0xb4 */, 1 /* 0xb5 */, 1 /* 0xb6 */, 1 /* 0xb7 */, 1 /* 0xb8 */,
432     1 /* 0xb9 */, 1 /* 0xba */, 1 /* 0xbb */, 1 /* 0xbc */, 1 /* 0xbd */,
433     1 /* 0xbe */, 1 /* 0xbf */, 1 /* 0xc0 */, 1 /* 0xc1 */, 1 /* 0xc2 */,
434     1 /* 0xc3 */, 1 /* 0xc4 */, 1 /* 0xc5 */, 1 /* 0xc6 */, 1 /* 0xc7 */,
435     1 /* 0xc8 */, 1 /* 0xc9 */, 1 /* 0xca */, 1 /* 0xcb */, 1 /* 0xcc */,
436     1 /* 0xcd */, 1 /* 0xce */, 1 /* 0xcf */, 1 /* 0xd0 */, 1 /* 0xd1 */,
437     1 /* 0xd2 */, 1 /* 0xd3 */, 1 /* 0xd4 */, 1 /* 0xd5 */, 1 /* 0xd6 */,
438     1 /* 0xd7 */, 1 /* 0xd8 */, 1 /* 0xd9 */, 1 /* 0xda */, 1 /* 0xdb */,
439     1 /* 0xdc */, 1 /* 0xdd */, 1 /* 0xde */, 1 /* 0xdf */, 1 /* 0xe0 */,
440     1 /* 0xe1 */, 1 /* 0xe2 */, 1 /* 0xe3 */, 1 /* 0xe4 */, 1 /* 0xe5 */,
441     1 /* 0xe6 */, 1 /* 0xe7 */, 1 /* 0xe8 */, 1 /* 0xe9 */, 1 /* 0xea */,
442     1 /* 0xeb */, 1 /* 0xec */, 1 /* 0xed */, 1 /* 0xee */, 1 /* 0xef */,
443     1 /* 0xf0 */, 1 /* 0xf1 */, 1 /* 0xf2 */, 1 /* 0xf3 */, 1 /* 0xf4 */,
444     1 /* 0xf5 */, 1 /* 0xf6 */, 1 /* 0xf7 */, 1 /* 0xf8 */, 1 /* 0xf9 */,
445     1 /* 0xfa */, 1 /* 0xfb */, 1 /* 0xfc */, 1 /* 0xfd */, 1 /* 0xfe */,
446     1 /* 0xff */
447 };
448
449 int nghttp2_check_header_value(const uint8_t *value, size_t len) {
450   const uint8_t *last;
451   for (last = value + len; value != last; ++value) {
452     if (!VALID_HD_VALUE_CHARS[*value]) {
453       return 0;
454     }
455   }
456   return 1;
457 }
458
459 uint8_t *nghttp2_cpymem(uint8_t *dest, const void *src, size_t len) {
460   memcpy(dest, src, len);
461
462   return dest + len;
463 }