1 /*
2 * libwebsockets - small server side websockets and web server implementation
3 *
4 * Copyright (C) 2010 - 2019 Andy Green <andy@warmcat.com>
5 *
6 * Permission is hereby granted, free of charge, to any person obtaining a copy
7 * of this software and associated documentation files (the "Software"), to
8 * deal in the Software without restriction, including without limitation the
9 * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
10 * sell copies of the Software, and to permit persons to whom the Software is
11 * furnished to do so, subject to the following conditions:
12 *
13 * The above copyright notice and this permission notice shall be included in
14 * all copies or substantial portions of the Software.
15 *
16 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
17 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
18 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
19 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
20 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
21 * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
22 * IN THE SOFTWARE.
23 */
24
25 #include "private-lib-core.h"
26
27 static const unsigned char lextable_h1[] = {
28 #include "lextable.h"
29 };
30
31 #define FAIL_CHAR 0x08
32
33 #if defined(LWS_WITH_CUSTOM_HEADERS)
34
35 #define UHO_NLEN 0
36 #define UHO_VLEN 2
37 #define UHO_LL 4
38 #define UHO_NAME 8
39
40 #endif
41
42 static struct allocated_headers *
_lws_create_ah(struct lws_context_per_thread * pt,ah_data_idx_t data_size)43 _lws_create_ah(struct lws_context_per_thread *pt, ah_data_idx_t data_size)
44 {
45 struct allocated_headers *ah = lws_zalloc(sizeof(*ah), "ah struct");
46
47 if (!ah)
48 return NULL;
49
50 ah->data = lws_malloc(data_size, "ah data");
51 if (!ah->data) {
52 lws_free(ah);
53
54 return NULL;
55 }
56 ah->next = pt->http.ah_list;
57 pt->http.ah_list = ah;
58 ah->data_length = data_size;
59 pt->http.ah_pool_length++;
60
61 lwsl_info("%s: created ah %p (size %d): pool length %u\n", __func__,
62 ah, (int)data_size, (unsigned int)pt->http.ah_pool_length);
63
64 return ah;
65 }
66
67 int
_lws_destroy_ah(struct lws_context_per_thread * pt,struct allocated_headers * ah)68 _lws_destroy_ah(struct lws_context_per_thread *pt, struct allocated_headers *ah)
69 {
70 lws_start_foreach_llp(struct allocated_headers **, a, pt->http.ah_list) {
71 if ((*a) == ah) {
72 *a = ah->next;
73 pt->http.ah_pool_length--;
74 lwsl_info("%s: freed ah %p : pool length %u\n",
75 __func__, ah,
76 (unsigned int)pt->http.ah_pool_length);
77 if (ah->data)
78 lws_free(ah->data);
79 lws_free(ah);
80
81 return 0;
82 }
83 } lws_end_foreach_llp(a, next);
84
85 return 1;
86 }
87
88 void
_lws_header_table_reset(struct allocated_headers * ah)89 _lws_header_table_reset(struct allocated_headers *ah)
90 {
91 /* init the ah to reflect no headers or data have appeared yet */
92 memset(ah->frag_index, 0, sizeof(ah->frag_index));
93 memset(ah->frags, 0, sizeof(ah->frags));
94 ah->nfrag = 0;
95 ah->pos = 0;
96 ah->http_response = 0;
97 ah->parser_state = WSI_TOKEN_NAME_PART;
98 ah->lextable_pos = 0;
99 ah->unk_pos = 0;
100 #if defined(LWS_WITH_CUSTOM_HEADERS)
101 ah->unk_ll_head = 0;
102 ah->unk_ll_tail = 0;
103 #endif
104 }
105
106 // doesn't scrub the ah rxbuffer by default, parent must do if needed
107
108 void
__lws_header_table_reset(struct lws * wsi,int autoservice)109 __lws_header_table_reset(struct lws *wsi, int autoservice)
110 {
111 struct allocated_headers *ah = wsi->http.ah;
112 struct lws_context_per_thread *pt;
113 struct lws_pollfd *pfd;
114
115 /* if we have the idea we're resetting 'our' ah, must be bound to one */
116 assert(ah);
117 /* ah also concurs with ownership */
118 assert(ah->wsi == wsi);
119
120 _lws_header_table_reset(ah);
121
122 /* since we will restart the ah, our new headers are not completed */
123 wsi->hdr_parsing_completed = 0;
124
125 /* while we hold the ah, keep a timeout on the wsi */
126 __lws_set_timeout(wsi, PENDING_TIMEOUT_HOLDING_AH,
127 wsi->vhost->timeout_secs_ah_idle);
128
129 time(&ah->assigned);
130
131 if (wsi->position_in_fds_table != LWS_NO_FDS_POS &&
132 lws_buflist_next_segment_len(&wsi->buflist, NULL) &&
133 autoservice) {
134 lwsl_debug("%s: service on readbuf ah\n", __func__);
135
136 pt = &wsi->context->pt[(int)wsi->tsi];
137 /*
138 * Unlike a normal connect, we have the headers already
139 * (or the first part of them anyway)
140 */
141 pfd = &pt->fds[wsi->position_in_fds_table];
142 pfd->revents |= LWS_POLLIN;
143 lwsl_err("%s: calling service\n", __func__);
144 lws_service_fd_tsi(wsi->context, pfd, wsi->tsi);
145 }
146 }
147
148 void
lws_header_table_reset(struct lws * wsi,int autoservice)149 lws_header_table_reset(struct lws *wsi, int autoservice)
150 {
151 struct lws_context_per_thread *pt = &wsi->context->pt[(int)wsi->tsi];
152
153 lws_pt_lock(pt, __func__);
154
155 __lws_header_table_reset(wsi, autoservice);
156
157 lws_pt_unlock(pt);
158 }
159
160 static void
_lws_header_ensure_we_are_on_waiting_list(struct lws * wsi)161 _lws_header_ensure_we_are_on_waiting_list(struct lws *wsi)
162 {
163 struct lws_context_per_thread *pt = &wsi->context->pt[(int)wsi->tsi];
164 struct lws_pollargs pa;
165 struct lws **pwsi = &pt->http.ah_wait_list;
166
167 while (*pwsi) {
168 if (*pwsi == wsi)
169 return;
170 pwsi = &(*pwsi)->http.ah_wait_list;
171 }
172
173 lwsl_info("%s: wsi: %p\n", __func__, wsi);
174 wsi->http.ah_wait_list = pt->http.ah_wait_list;
175 pt->http.ah_wait_list = wsi;
176 pt->http.ah_wait_list_length++;
177
178 /* we cannot accept input then */
179
180 _lws_change_pollfd(wsi, LWS_POLLIN, 0, &pa);
181 }
182
183 static int
__lws_remove_from_ah_waiting_list(struct lws * wsi)184 __lws_remove_from_ah_waiting_list(struct lws *wsi)
185 {
186 struct lws_context_per_thread *pt = &wsi->context->pt[(int)wsi->tsi];
187 struct lws **pwsi =&pt->http.ah_wait_list;
188
189 while (*pwsi) {
190 if (*pwsi == wsi) {
191 lwsl_info("%s: wsi %p\n", __func__, wsi);
192 /* point prev guy to our next */
193 *pwsi = wsi->http.ah_wait_list;
194 /* we shouldn't point anywhere now */
195 wsi->http.ah_wait_list = NULL;
196 pt->http.ah_wait_list_length--;
197
198 return 1;
199 }
200 pwsi = &(*pwsi)->http.ah_wait_list;
201 }
202
203 return 0;
204 }
205
206 int LWS_WARN_UNUSED_RESULT
lws_header_table_attach(struct lws * wsi,int autoservice)207 lws_header_table_attach(struct lws *wsi, int autoservice)
208 {
209 struct lws_context *context = wsi->context;
210 struct lws_context_per_thread *pt = &context->pt[(int)wsi->tsi];
211 struct lws_pollargs pa;
212 int n;
213
214 #if defined(LWS_ROLE_MQTT) && defined(LWS_WITH_CLIENT)
215 if (lwsi_role_mqtt(wsi))
216 goto connect_via_info2;
217 #endif
218
219 lwsl_info("%s: wsi %p: ah %p (tsi %d, count = %d) in\n", __func__,
220 (void *)wsi, (void *)wsi->http.ah, wsi->tsi,
221 pt->http.ah_count_in_use);
222
223 if (!lwsi_role_http(wsi)) {
224 lwsl_err("%s: bad role %s\n", __func__, wsi->role_ops->name);
225 assert(0);
226 return -1;
227 }
228
229 lws_pt_lock(pt, __func__);
230
231 /* if we are already bound to one, just clear it down */
232 if (wsi->http.ah) {
233 lwsl_info("%s: cleardown\n", __func__);
234 goto reset;
235 }
236
237 n = pt->http.ah_count_in_use == context->max_http_header_pool;
238 #if defined(LWS_WITH_PEER_LIMITS)
239 if (!n) {
240 n = lws_peer_confirm_ah_attach_ok(context, wsi->peer);
241 if (n)
242 lws_stats_bump(pt, LWSSTATS_C_PEER_LIMIT_AH_DENIED, 1);
243 }
244 #endif
245 if (n) {
246 /*
247 * Pool is either all busy, or we don't want to give this
248 * particular guy an ah right now...
249 *
250 * Make sure we are on the waiting list, and return that we
251 * weren't able to provide the ah
252 */
253 _lws_header_ensure_we_are_on_waiting_list(wsi);
254
255 goto bail;
256 }
257
258 __lws_remove_from_ah_waiting_list(wsi);
259
260 wsi->http.ah = _lws_create_ah(pt, context->max_http_header_data);
261 if (!wsi->http.ah) { /* we could not create an ah */
262 _lws_header_ensure_we_are_on_waiting_list(wsi);
263
264 goto bail;
265 }
266
267 wsi->http.ah->in_use = 1;
268 wsi->http.ah->wsi = wsi; /* mark our owner */
269 pt->http.ah_count_in_use++;
270
271 #if defined(LWS_WITH_PEER_LIMITS) && (defined(LWS_ROLE_H1) || \
272 defined(LWS_ROLE_H2))
273 lws_context_lock(context, "ah attach"); /* <========================= */
274 if (wsi->peer)
275 wsi->peer->http.count_ah++;
276 lws_context_unlock(context); /* ====================================> */
277 #endif
278
279 _lws_change_pollfd(wsi, 0, LWS_POLLIN, &pa);
280
281 lwsl_info("%s: did attach wsi %p: ah %p: count %d (on exit)\n", __func__,
282 (void *)wsi, (void *)wsi->http.ah, pt->http.ah_count_in_use);
283
284 reset:
285 __lws_header_table_reset(wsi, autoservice);
286
287 lws_pt_unlock(pt);
288
289 #if defined(LWS_WITH_CLIENT)
290 #if defined(LWS_ROLE_MQTT)
291 connect_via_info2:
292 #endif
293 if (lwsi_role_client(wsi) && lwsi_state(wsi) == LRS_UNCONNECTED)
294 if (!lws_http_client_connect_via_info2(wsi))
295 /* our client connect has failed, the wsi
296 * has been closed
297 */
298 return -1;
299 #endif
300
301 return 0;
302
303 bail:
304 lws_pt_unlock(pt);
305
306 return 1;
307 }
308
__lws_header_table_detach(struct lws * wsi,int autoservice)309 int __lws_header_table_detach(struct lws *wsi, int autoservice)
310 {
311 struct lws_context *context = wsi->context;
312 struct allocated_headers *ah = wsi->http.ah;
313 struct lws_context_per_thread *pt = &context->pt[(int)wsi->tsi];
314 struct lws_pollargs pa;
315 struct lws **pwsi, **pwsi_eligible;
316 time_t now;
317
318 __lws_remove_from_ah_waiting_list(wsi);
319
320 if (!ah)
321 return 0;
322
323 lwsl_info("%s: wsi %p: ah %p (tsi=%d, count = %d)\n", __func__,
324 (void *)wsi, (void *)ah, wsi->tsi,
325 pt->http.ah_count_in_use);
326
327 /* we did have an ah attached */
328 time(&now);
329 if (ah->assigned && now - ah->assigned > 3) {
330 /*
331 * we're detaching the ah, but it was held an
332 * unreasonably long time
333 */
334 lwsl_debug("%s: wsi %p: ah held %ds, role/state 0x%lx 0x%x,"
335 "\n", __func__, wsi, (int)(now - ah->assigned),
336 (unsigned long)lwsi_role(wsi), lwsi_state(wsi));
337 }
338
339 ah->assigned = 0;
340
341 /* if we think we're detaching one, there should be one in use */
342 assert(pt->http.ah_count_in_use > 0);
343 /* and this specific one should have been in use */
344 assert(ah->in_use);
345 memset(&wsi->http.ah, 0, sizeof(wsi->http.ah));
346
347 #if defined(LWS_WITH_PEER_LIMITS)
348 if (ah->wsi)
349 lws_peer_track_ah_detach(context, wsi->peer);
350 #endif
351 ah->wsi = NULL; /* no owner */
352 wsi->http.ah = NULL;
353
354 pwsi = &pt->http.ah_wait_list;
355
356 /* oh there is nobody on the waiting list... leave the ah unattached */
357 if (!*pwsi)
358 goto nobody_usable_waiting;
359
360 /*
361 * at least one wsi on the same tsi is waiting, give it to oldest guy
362 * who is allowed to take it (if any)
363 */
364 lwsl_info("pt wait list %p\n", *pwsi);
365 wsi = NULL;
366 pwsi_eligible = NULL;
367
368 while (*pwsi) {
369 #if defined(LWS_WITH_PEER_LIMITS)
370 /* are we willing to give this guy an ah? */
371 if (!lws_peer_confirm_ah_attach_ok(context, (*pwsi)->peer))
372 #endif
373 {
374 wsi = *pwsi;
375 pwsi_eligible = pwsi;
376 }
377 #if defined(LWS_WITH_PEER_LIMITS)
378 else
379 if (!(*pwsi)->http.ah_wait_list)
380 lws_stats_bump(pt,
381 LWSSTATS_C_PEER_LIMIT_AH_DENIED, 1);
382 #endif
383 pwsi = &(*pwsi)->http.ah_wait_list;
384 }
385
386 if (!wsi) /* everybody waiting already has too many ah... */
387 goto nobody_usable_waiting;
388
389 lwsl_info("%s: transferring ah to last eligible wsi in wait list "
390 "%p (wsistate 0x%lx)\n", __func__, wsi,
391 (unsigned long)wsi->wsistate);
392
393 wsi->http.ah = ah;
394 ah->wsi = wsi; /* new owner */
395
396 __lws_header_table_reset(wsi, autoservice);
397 #if defined(LWS_WITH_PEER_LIMITS) && (defined(LWS_ROLE_H1) || \
398 defined(LWS_ROLE_H2))
399 lws_context_lock(context, "ah detach"); /* <========================= */
400 if (wsi->peer)
401 wsi->peer->http.count_ah++;
402 lws_context_unlock(context); /* ====================================> */
403 #endif
404
405 /* clients acquire the ah and then insert themselves in fds table... */
406 if (wsi->position_in_fds_table != LWS_NO_FDS_POS) {
407 lwsl_info("%s: Enabling %p POLLIN\n", __func__, wsi);
408
409 /* he has been stuck waiting for an ah, but now his wait is
410 * over, let him progress */
411
412 _lws_change_pollfd(wsi, 0, LWS_POLLIN, &pa);
413 }
414
415 /* point prev guy to next guy in list instead */
416 *pwsi_eligible = wsi->http.ah_wait_list;
417 /* the guy who got one is out of the list */
418 wsi->http.ah_wait_list = NULL;
419 pt->http.ah_wait_list_length--;
420
421 #if defined(LWS_WITH_CLIENT)
422 if (lwsi_role_client(wsi) && lwsi_state(wsi) == LRS_UNCONNECTED) {
423 lws_pt_unlock(pt);
424
425 if (!lws_http_client_connect_via_info2(wsi)) {
426 /* our client connect has failed, the wsi
427 * has been closed
428 */
429
430 return -1;
431 }
432 return 0;
433 }
434 #endif
435
436 assert(!!pt->http.ah_wait_list_length ==
437 !!(lws_intptr_t)pt->http.ah_wait_list);
438 bail:
439 lwsl_info("%s: wsi %p: ah %p (tsi=%d, count = %d)\n", __func__,
440 (void *)wsi, (void *)ah, pt->tid, pt->http.ah_count_in_use);
441
442 return 0;
443
444 nobody_usable_waiting:
445 lwsl_info("%s: nobody usable waiting\n", __func__);
446 _lws_destroy_ah(pt, ah);
447 pt->http.ah_count_in_use--;
448
449 goto bail;
450 }
451
lws_header_table_detach(struct lws * wsi,int autoservice)452 int lws_header_table_detach(struct lws *wsi, int autoservice)
453 {
454 struct lws_context *context = wsi->context;
455 struct lws_context_per_thread *pt = &context->pt[(int)wsi->tsi];
456 int n;
457
458 lws_pt_lock(pt, __func__);
459 n = __lws_header_table_detach(wsi, autoservice);
460 lws_pt_unlock(pt);
461
462 return n;
463 }
464
465 int
lws_hdr_fragment_length(struct lws * wsi,enum lws_token_indexes h,int frag_idx)466 lws_hdr_fragment_length(struct lws *wsi, enum lws_token_indexes h, int frag_idx)
467 {
468 int n;
469
470 if (!wsi->http.ah)
471 return 0;
472
473 n = wsi->http.ah->frag_index[h];
474 if (!n)
475 return 0;
476 do {
477 if (!frag_idx)
478 return wsi->http.ah->frags[n].len;
479 n = wsi->http.ah->frags[n].nfrag;
480 } while (frag_idx-- && n);
481
482 return 0;
483 }
484
lws_hdr_total_length(struct lws * wsi,enum lws_token_indexes h)485 int lws_hdr_total_length(struct lws *wsi, enum lws_token_indexes h)
486 {
487 int n;
488 int len = 0;
489
490 if (!wsi->http.ah)
491 return 0;
492
493 n = wsi->http.ah->frag_index[h];
494 if (!n)
495 return 0;
496 do {
497 len += wsi->http.ah->frags[n].len;
498 n = wsi->http.ah->frags[n].nfrag;
499
500 if (n && h != WSI_TOKEN_HTTP_COOKIE)
501 ++len;
502
503 } while (n);
504
505 return len;
506 }
507
lws_hdr_copy_fragment(struct lws * wsi,char * dst,int len,enum lws_token_indexes h,int frag_idx)508 int lws_hdr_copy_fragment(struct lws *wsi, char *dst, int len,
509 enum lws_token_indexes h, int frag_idx)
510 {
511 int n = 0;
512 int f;
513
514 if (!wsi->http.ah)
515 return -1;
516
517 f = wsi->http.ah->frag_index[h];
518
519 if (!f)
520 return -1;
521
522 while (n < frag_idx) {
523 f = wsi->http.ah->frags[f].nfrag;
524 if (!f)
525 return -1;
526 n++;
527 }
528
529 if (wsi->http.ah->frags[f].len >= len)
530 return -1;
531
532 memcpy(dst, wsi->http.ah->data + wsi->http.ah->frags[f].offset,
533 wsi->http.ah->frags[f].len);
534 dst[wsi->http.ah->frags[f].len] = '\0';
535
536 return wsi->http.ah->frags[f].len;
537 }
538
lws_hdr_copy(struct lws * wsi,char * dst,int len,enum lws_token_indexes h)539 int lws_hdr_copy(struct lws *wsi, char *dst, int len,
540 enum lws_token_indexes h)
541 {
542 int toklen = lws_hdr_total_length(wsi, h);
543 int n;
544 int comma;
545
546 *dst = '\0';
547 if (!toklen)
548 return 0;
549
550 if (toklen >= len)
551 return -1;
552
553 if (!wsi->http.ah)
554 return -1;
555
556 n = wsi->http.ah->frag_index[h];
557 if (!n)
558 return 0;
559
560 do {
561 comma = (wsi->http.ah->frags[n].nfrag &&
562 h != WSI_TOKEN_HTTP_COOKIE) ? 1 : 0;
563
564 if (wsi->http.ah->frags[n].len + comma >= len)
565 return -1;
566 strncpy(dst, &wsi->http.ah->data[wsi->http.ah->frags[n].offset],
567 wsi->http.ah->frags[n].len);
568 dst += wsi->http.ah->frags[n].len;
569 len -= wsi->http.ah->frags[n].len;
570 n = wsi->http.ah->frags[n].nfrag;
571
572 if (comma)
573 *dst++ = ',';
574
575 } while (n);
576 *dst = '\0';
577
578 return toklen;
579 }
580
581 #if defined(LWS_WITH_CUSTOM_HEADERS)
582 int
lws_hdr_custom_length(struct lws * wsi,const char * name,int nlen)583 lws_hdr_custom_length(struct lws *wsi, const char *name, int nlen)
584 {
585 ah_data_idx_t ll;
586
587 if (!wsi->http.ah || wsi->mux_substream)
588 return -1;
589
590 ll = wsi->http.ah->unk_ll_head;
591 while (ll) {
592 if (ll >= wsi->http.ah->data_length)
593 return -1;
594 if (nlen == lws_ser_ru16be(
595 (uint8_t *)&wsi->http.ah->data[ll + UHO_NLEN]) &&
596 !strncmp(name, &wsi->http.ah->data[ll + UHO_NAME], nlen))
597 return lws_ser_ru16be(
598 (uint8_t *)&wsi->http.ah->data[ll + UHO_VLEN]);
599
600 ll = lws_ser_ru32be((uint8_t *)&wsi->http.ah->data[ll + UHO_LL]);
601 }
602
603 return -1;
604 }
605
606 int
lws_hdr_custom_copy(struct lws * wsi,char * dst,int len,const char * name,int nlen)607 lws_hdr_custom_copy(struct lws *wsi, char *dst, int len, const char *name,
608 int nlen)
609 {
610 ah_data_idx_t ll;
611 int n;
612
613 if (!wsi->http.ah || wsi->mux_substream)
614 return -1;
615
616 *dst = '\0';
617
618 ll = wsi->http.ah->unk_ll_head;
619 while (ll) {
620 if (ll >= wsi->http.ah->data_length)
621 return -1;
622 if (nlen == lws_ser_ru16be(
623 (uint8_t *)&wsi->http.ah->data[ll + UHO_NLEN]) &&
624 !strncmp(name, &wsi->http.ah->data[ll + UHO_NAME], nlen)) {
625 n = lws_ser_ru16be(
626 (uint8_t *)&wsi->http.ah->data[ll + UHO_VLEN]);
627 if (n + 1 > len)
628 return -1;
629 strncpy(dst, &wsi->http.ah->data[ll + UHO_NAME + nlen], n);
630 dst[n] = '\0';
631
632 return n;
633 }
634 ll = lws_ser_ru32be((uint8_t *)&wsi->http.ah->data[ll + UHO_LL]);
635 }
636
637 return -1;
638 }
639 #endif
640
lws_hdr_simple_ptr(struct lws * wsi,enum lws_token_indexes h)641 char *lws_hdr_simple_ptr(struct lws *wsi, enum lws_token_indexes h)
642 {
643 int n;
644
645 if (!wsi->http.ah)
646 return NULL;
647
648 n = wsi->http.ah->frag_index[h];
649 if (!n)
650 return NULL;
651
652 return wsi->http.ah->data + wsi->http.ah->frags[n].offset;
653 }
654
655 static int LWS_WARN_UNUSED_RESULT
lws_pos_in_bounds(struct lws * wsi)656 lws_pos_in_bounds(struct lws *wsi)
657 {
658 if (!wsi->http.ah)
659 return -1;
660
661 if (wsi->http.ah->pos <
662 (unsigned int)wsi->context->max_http_header_data)
663 return 0;
664
665 if ((int)wsi->http.ah->pos >= wsi->context->max_http_header_data - 1) {
666 lwsl_err("Ran out of header data space\n");
667 return 1;
668 }
669
670 /*
671 * with these tests everywhere, it should never be able to exceed
672 * the limit, only meet it
673 */
674 lwsl_err("%s: pos %ld, limit %ld\n", __func__,
675 (unsigned long)wsi->http.ah->pos,
676 (unsigned long)wsi->context->max_http_header_data);
677 assert(0);
678
679 return 1;
680 }
681
682 int LWS_WARN_UNUSED_RESULT
lws_hdr_simple_create(struct lws * wsi,enum lws_token_indexes h,const char * s)683 lws_hdr_simple_create(struct lws *wsi, enum lws_token_indexes h, const char *s)
684 {
685 if (!*s) {
686 /*
687 * If we get an empty string, then remove any entry for the
688 * header
689 */
690 wsi->http.ah->frag_index[h] = 0;
691
692 return 0;
693 }
694
695 wsi->http.ah->nfrag++;
696 if (wsi->http.ah->nfrag == LWS_ARRAY_SIZE(wsi->http.ah->frags)) {
697 lwsl_warn("More hdr frags than we can deal with, dropping\n");
698 return -1;
699 }
700
701 wsi->http.ah->frag_index[h] = wsi->http.ah->nfrag;
702
703 wsi->http.ah->frags[wsi->http.ah->nfrag].offset = wsi->http.ah->pos;
704 wsi->http.ah->frags[wsi->http.ah->nfrag].len = 0;
705 wsi->http.ah->frags[wsi->http.ah->nfrag].nfrag = 0;
706
707 do {
708 if (lws_pos_in_bounds(wsi))
709 return -1;
710
711 wsi->http.ah->data[wsi->http.ah->pos++] = *s;
712 if (*s)
713 wsi->http.ah->frags[wsi->http.ah->nfrag].len++;
714 } while (*s++);
715
716 return 0;
717 }
718
719 static int LWS_WARN_UNUSED_RESULT
issue_char(struct lws * wsi,unsigned char c)720 issue_char(struct lws *wsi, unsigned char c)
721 {
722 unsigned short frag_len;
723
724 if (lws_pos_in_bounds(wsi))
725 return -1;
726
727 frag_len = wsi->http.ah->frags[wsi->http.ah->nfrag].len;
728 /*
729 * If we haven't hit the token limit, just copy the character into
730 * the header
731 */
732 if (!wsi->http.ah->current_token_limit ||
733 frag_len < wsi->http.ah->current_token_limit) {
734 wsi->http.ah->data[wsi->http.ah->pos++] = c;
735 if (c)
736 wsi->http.ah->frags[wsi->http.ah->nfrag].len++;
737 return 0;
738 }
739
740 /* Insert a null character when we *hit* the limit: */
741 if (frag_len == wsi->http.ah->current_token_limit) {
742 if (lws_pos_in_bounds(wsi))
743 return -1;
744
745 wsi->http.ah->data[wsi->http.ah->pos++] = '\0';
746 lwsl_warn("header %li exceeds limit %ld\n",
747 (long)wsi->http.ah->parser_state,
748 (long)wsi->http.ah->current_token_limit);
749 }
750
751 return 1;
752 }
753
754 int
lws_parse_urldecode(struct lws * wsi,uint8_t * _c)755 lws_parse_urldecode(struct lws *wsi, uint8_t *_c)
756 {
757 struct allocated_headers *ah = wsi->http.ah;
758 unsigned int enc = 0;
759 uint8_t c = *_c;
760
761 // lwsl_notice("ah->ups %d\n", ah->ups);
762
763 /*
764 * PRIORITY 1
765 * special URI processing... convert %xx
766 */
767 switch (ah->ues) {
768 case URIES_IDLE:
769 if (c == '%') {
770 ah->ues = URIES_SEEN_PERCENT;
771 goto swallow;
772 }
773 break;
774 case URIES_SEEN_PERCENT:
775 if (char_to_hex(c) < 0)
776 /* illegal post-% char */
777 goto forbid;
778
779 ah->esc_stash = c;
780 ah->ues = URIES_SEEN_PERCENT_H1;
781 goto swallow;
782
783 case URIES_SEEN_PERCENT_H1:
784 if (char_to_hex(c) < 0)
785 /* illegal post-% char */
786 goto forbid;
787
788 *_c = (char_to_hex(ah->esc_stash) << 4) |
789 char_to_hex(c);
790 c = *_c;
791 enc = 1;
792 ah->ues = URIES_IDLE;
793 break;
794 }
795
796 /*
797 * PRIORITY 2
798 * special URI processing...
799 * convert /.. or /... or /../ etc to /
800 * convert /./ to /
801 * convert // or /// etc to /
802 * leave /.dir or whatever alone
803 */
804
805 switch (ah->ups) {
806 case URIPS_IDLE:
807 if (!c)
808 return -1;
809 /* genuine delimiter */
810 if ((c == '&' || c == ';') && !enc) {
811 if (issue_char(wsi, '\0') < 0)
812 return -1;
813 /* link to next fragment */
814 ah->frags[ah->nfrag].nfrag = ah->nfrag + 1;
815 ah->nfrag++;
816 if (ah->nfrag >= LWS_ARRAY_SIZE(ah->frags))
817 goto excessive;
818 /* start next fragment after the & */
819 ah->post_literal_equal = 0;
820 ah->frags[ah->nfrag].offset = ++ah->pos;
821 ah->frags[ah->nfrag].len = 0;
822 ah->frags[ah->nfrag].nfrag = 0;
823 goto swallow;
824 }
825 /* uriencoded = in the name part, disallow */
826 if (c == '=' && enc &&
827 ah->frag_index[WSI_TOKEN_HTTP_URI_ARGS] &&
828 !ah->post_literal_equal) {
829 c = '_';
830 *_c =c;
831 }
832
833 /* after the real =, we don't care how many = */
834 if (c == '=' && !enc)
835 ah->post_literal_equal = 1;
836
837 /* + to space */
838 if (c == '+' && !enc) {
839 c = ' ';
840 *_c = c;
841 }
842 /* issue the first / always */
843 if (c == '/' && !ah->frag_index[WSI_TOKEN_HTTP_URI_ARGS])
844 ah->ups = URIPS_SEEN_SLASH;
845 break;
846 case URIPS_SEEN_SLASH:
847 /* swallow subsequent slashes */
848 if (c == '/')
849 goto swallow;
850 /* track and swallow the first . after / */
851 if (c == '.') {
852 ah->ups = URIPS_SEEN_SLASH_DOT;
853 goto swallow;
854 }
855 ah->ups = URIPS_IDLE;
856 break;
857 case URIPS_SEEN_SLASH_DOT:
858 /* swallow second . */
859 if (c == '.') {
860 ah->ups = URIPS_SEEN_SLASH_DOT_DOT;
861 goto swallow;
862 }
863 /* change /./ to / */
864 if (c == '/') {
865 ah->ups = URIPS_SEEN_SLASH;
866 goto swallow;
867 }
868 /* it was like /.dir ... regurgitate the . */
869 ah->ups = URIPS_IDLE;
870 if (issue_char(wsi, '.') < 0)
871 return -1;
872 break;
873
874 case URIPS_SEEN_SLASH_DOT_DOT:
875
876 /* /../ or /..[End of URI] --> backup to last / */
877 if (c == '/' || c == '?') {
878 /*
879 * back up one dir level if possible
880 * safe against header fragmentation because
881 * the method URI can only be in 1 fragment
882 */
883 if (ah->frags[ah->nfrag].len > 2) {
884 ah->pos--;
885 ah->frags[ah->nfrag].len--;
886 do {
887 ah->pos--;
888 ah->frags[ah->nfrag].len--;
889 } while (ah->frags[ah->nfrag].len > 1 &&
890 ah->data[ah->pos] != '/');
891 }
892 ah->ups = URIPS_SEEN_SLASH;
893 if (ah->frags[ah->nfrag].len > 1)
894 break;
895 goto swallow;
896 }
897
898 /* /..[^/] ... regurgitate and allow */
899
900 if (issue_char(wsi, '.') < 0)
901 return -1;
902 if (issue_char(wsi, '.') < 0)
903 return -1;
904 ah->ups = URIPS_IDLE;
905 break;
906 }
907
908 if (c == '?' && !enc &&
909 !ah->frag_index[WSI_TOKEN_HTTP_URI_ARGS]) { /* start of URI args */
910 if (ah->ues != URIES_IDLE)
911 goto forbid;
912
913 /* seal off uri header */
914 if (issue_char(wsi, '\0') < 0)
915 return -1;
916
917 /* move to using WSI_TOKEN_HTTP_URI_ARGS */
918 ah->nfrag++;
919 if (ah->nfrag >= LWS_ARRAY_SIZE(ah->frags))
920 goto excessive;
921 ah->frags[ah->nfrag].offset = ++ah->pos;
922 ah->frags[ah->nfrag].len = 0;
923 ah->frags[ah->nfrag].nfrag = 0;
924
925 ah->post_literal_equal = 0;
926 ah->frag_index[WSI_TOKEN_HTTP_URI_ARGS] = ah->nfrag;
927 ah->ups = URIPS_IDLE;
928 goto swallow;
929 }
930
931 return LPUR_CONTINUE;
932
933 swallow:
934 return LPUR_SWALLOW;
935
936 forbid:
937 return LPUR_FORBID;
938
939 excessive:
940 return LPUR_EXCESSIVE;
941 }
942
943 static const unsigned char methods[] = {
944 WSI_TOKEN_GET_URI,
945 WSI_TOKEN_POST_URI,
946 #if defined(LWS_WITH_HTTP_UNCOMMON_HEADERS)
947 WSI_TOKEN_OPTIONS_URI,
948 WSI_TOKEN_PUT_URI,
949 WSI_TOKEN_PATCH_URI,
950 WSI_TOKEN_DELETE_URI,
951 #endif
952 WSI_TOKEN_CONNECT,
953 WSI_TOKEN_HEAD_URI,
954 };
955
956 /*
957 * possible returns:, -1 fail, 0 ok or 2, transition to raw
958 */
959
960 lws_parser_return_t LWS_WARN_UNUSED_RESULT
lws_parse(struct lws * wsi,unsigned char * buf,int * len)961 lws_parse(struct lws *wsi, unsigned char *buf, int *len)
962 {
963 struct allocated_headers *ah = wsi->http.ah;
964 struct lws_context *context = wsi->context;
965 unsigned int n, m;
966 unsigned char c;
967 int r, pos;
968
969 assert(wsi->http.ah);
970
971 do {
972 (*len)--;
973 c = *buf++;
974
975 switch (ah->parser_state) {
976 #if defined(LWS_WITH_CUSTOM_HEADERS)
977 case WSI_TOKEN_UNKNOWN_VALUE_PART:
978
979 if (c == '\r')
980 break;
981 if (c == '\n') {
982 lws_ser_wu16be((uint8_t *)&ah->data[ah->unk_pos + 2],
983 ah->pos - ah->unk_value_pos);
984 ah->parser_state = WSI_TOKEN_NAME_PART;
985 ah->unk_pos = 0;
986 ah->lextable_pos = 0;
987 break;
988 }
989
990 /* trim leading whitespace */
991 if (ah->pos != ah->unk_value_pos ||
992 (c != ' ' && c != '\t')) {
993
994 if (lws_pos_in_bounds(wsi))
995 return LPR_FAIL;
996
997 ah->data[ah->pos++] = c;
998 }
999 pos = ah->lextable_pos;
1000 break;
1001 #endif
1002 default:
1003
1004 lwsl_parser("WSI_TOK_(%d) '%c'\n", ah->parser_state, c);
1005
1006 /* collect into malloc'd buffers */
1007 /* optional initial space swallow */
1008 if (!ah->frags[ah->frag_index[ah->parser_state]].len &&
1009 c == ' ')
1010 break;
1011
1012 for (m = 0; m < LWS_ARRAY_SIZE(methods); m++)
1013 if (ah->parser_state == methods[m])
1014 break;
1015 if (m == LWS_ARRAY_SIZE(methods))
1016 /* it was not any of the methods */
1017 goto check_eol;
1018
1019 /* special URI processing... end at space */
1020
1021 if (c == ' ') {
1022 /* enforce starting with / */
1023 if (!ah->frags[ah->nfrag].len)
1024 if (issue_char(wsi, '/') < 0)
1025 return LPR_FAIL;
1026
1027 if (ah->ups == URIPS_SEEN_SLASH_DOT_DOT) {
1028 /*
1029 * back up one dir level if possible
1030 * safe against header fragmentation
1031 * because the method URI can only be
1032 * in 1 fragment
1033 */
1034 if (ah->frags[ah->nfrag].len > 2) {
1035 ah->pos--;
1036 ah->frags[ah->nfrag].len--;
1037 do {
1038 ah->pos--;
1039 ah->frags[ah->nfrag].len--;
1040 } while (ah->frags[ah->nfrag].len > 1 &&
1041 ah->data[ah->pos] != '/');
1042 }
1043 }
1044
1045 /* begin parsing HTTP version: */
1046 if (issue_char(wsi, '\0') < 0)
1047 return LPR_FAIL;
1048 ah->parser_state = WSI_TOKEN_HTTP;
1049 goto start_fragment;
1050 }
1051
1052 r = lws_parse_urldecode(wsi, &c);
1053 switch (r) {
1054 case LPUR_CONTINUE:
1055 break;
1056 case LPUR_SWALLOW:
1057 goto swallow;
1058 case LPUR_FORBID:
1059 goto forbid;
1060 case LPUR_EXCESSIVE:
1061 goto excessive;
1062 default:
1063 return LPR_FAIL;
1064 }
1065 check_eol:
1066 /* bail at EOL */
1067 if (ah->parser_state != WSI_TOKEN_CHALLENGE &&
1068 c == '\x0d') {
1069 if (ah->ues != URIES_IDLE)
1070 goto forbid;
1071
1072 c = '\0';
1073 ah->parser_state = WSI_TOKEN_SKIPPING_SAW_CR;
1074 lwsl_parser("*\n");
1075 }
1076
1077 n = issue_char(wsi, c);
1078 if ((int)n < 0)
1079 return LPR_FAIL;
1080 if (n > 0)
1081 ah->parser_state = WSI_TOKEN_SKIPPING;
1082
1083 swallow:
1084 /* per-protocol end of headers management */
1085
1086 if (ah->parser_state == WSI_TOKEN_CHALLENGE)
1087 goto set_parsing_complete;
1088 break;
1089
1090 /* collecting and checking a name part */
1091 case WSI_TOKEN_NAME_PART:
1092 lwsl_parser("WSI_TOKEN_NAME_PART '%c' 0x%02X "
1093 "(role=0x%lx) "
1094 "wsi->lextable_pos=%d\n", c, c,
1095 (unsigned long)lwsi_role(wsi),
1096 ah->lextable_pos);
1097
1098 if (c >= 'A' && c <= 'Z')
1099 c += 'a' - 'A';
1100 /*
1101 * ...in case it's an unknown header, speculatively
1102 * store it as the name comes in. If we recognize it as
1103 * a known header, we'll snip this.
1104 */
1105
1106 if (!wsi->mux_substream && !ah->unk_pos) {
1107 ah->unk_pos = ah->pos;
1108 #if defined(LWS_WITH_CUSTOM_HEADERS)
1109 /*
1110 * Prepare new unknown header linked-list entry
1111 *
1112 * - 16-bit BE: name part length
1113 * - 16-bit BE: value part length
1114 * - 32-bit BE: data offset of next, or 0
1115 */
1116 for (n = 0; n < 8; n++)
1117 if (!lws_pos_in_bounds(wsi))
1118 ah->data[ah->pos++] = 0;
1119 #endif
1120 }
1121
1122 if (lws_pos_in_bounds(wsi))
1123 return LPR_FAIL;
1124
1125 ah->data[ah->pos++] = c;
1126 pos = ah->lextable_pos;
1127
1128 #if defined(LWS_WITH_CUSTOM_HEADERS)
1129 if (!wsi->mux_substream && pos < 0 && c == ':') {
1130 #if defined(_DEBUG)
1131 char dotstar[64];
1132 int uhlen;
1133 #endif
1134
1135 /*
1136 * process unknown headers
1137 *
1138 * register us in the unknown hdr ll
1139 */
1140
1141 if (!ah->unk_ll_head)
1142 ah->unk_ll_head = ah->unk_pos;
1143
1144 if (ah->unk_ll_tail)
1145 lws_ser_wu32be(
1146 (uint8_t *)&ah->data[ah->unk_ll_tail + UHO_LL],
1147 ah->unk_pos);
1148
1149 ah->unk_ll_tail = ah->unk_pos;
1150
1151 #if defined(_DEBUG)
1152 uhlen = ah->pos - (ah->unk_pos + UHO_NAME);
1153 lws_strnncpy(dotstar,
1154 &ah->data[ah->unk_pos + UHO_NAME],
1155 uhlen, sizeof(dotstar));
1156 lwsl_debug("%s: unk header %d '%s'\n",
1157 __func__,
1158 ah->pos - (ah->unk_pos + UHO_NAME),
1159 dotstar);
1160 #endif
1161
1162 /* set the unknown header name part length */
1163
1164 lws_ser_wu16be((uint8_t *)&ah->data[ah->unk_pos],
1165 (ah->pos - ah->unk_pos) - UHO_NAME);
1166
1167 ah->unk_value_pos = ah->pos;
1168
1169 /*
1170 * collect whatever's coming for the unknown header
1171 * argument until the next CRLF
1172 */
1173 ah->parser_state = WSI_TOKEN_UNKNOWN_VALUE_PART;
1174 break;
1175 }
1176 #endif
1177 if (pos < 0)
1178 break;
1179
1180 while (1) {
1181 if (lextable_h1[pos] & (1 << 7)) {
1182 /* 1-byte, fail on mismatch */
1183 if ((lextable_h1[pos] & 0x7f) != c) {
1184 nope:
1185 ah->lextable_pos = -1;
1186 break;
1187 }
1188 /* fall thru */
1189 pos++;
1190 if (lextable_h1[pos] == FAIL_CHAR)
1191 goto nope;
1192
1193 ah->lextable_pos = pos;
1194 break;
1195 }
1196
1197 if (lextable_h1[pos] == FAIL_CHAR)
1198 goto nope;
1199
1200 /* b7 = 0, end or 3-byte */
1201 if (lextable_h1[pos] < FAIL_CHAR) {
1202 if (!wsi->mux_substream) {
1203 /*
1204 * We hit a terminal marker, so
1205 * we recognized this header...
1206 * drop the speculative name
1207 * part storage
1208 */
1209 ah->pos = ah->unk_pos;
1210 ah->unk_pos = 0;
1211 }
1212
1213 ah->lextable_pos = pos;
1214 break;
1215 }
1216
1217 if (lextable_h1[pos] == c) { /* goto */
1218 ah->lextable_pos = pos +
1219 (lextable_h1[pos + 1]) +
1220 (lextable_h1[pos + 2] << 8);
1221 break;
1222 }
1223
1224 /* fall thru goto */
1225 pos += 3;
1226 /* continue */
1227 }
1228
1229 /*
1230 * If it's h1, server needs to be on the look out for
1231 * unknown methods...
1232 */
1233 if (ah->lextable_pos < 0 && lwsi_role_h1(wsi) &&
1234 lwsi_role_server(wsi)) {
1235 /*
1236 * this is not a header we know about... did
1237 * we get a valid method (GET, POST etc)
1238 * already, or is this the bogus method?
1239 */
1240 for (m = 0; m < LWS_ARRAY_SIZE(methods); m++)
1241 if (ah->frag_index[methods[m]]) {
1242 /*
1243 * already had the method
1244 */
1245 #if !defined(LWS_WITH_CUSTOM_HEADERS)
1246 ah->parser_state = WSI_TOKEN_SKIPPING;
1247 #endif
1248 if (wsi->mux_substream)
1249 ah->parser_state = WSI_TOKEN_SKIPPING;
1250 break;
1251 }
1252
1253 if (m != LWS_ARRAY_SIZE(methods)) {
1254 #if defined(LWS_WITH_CUSTOM_HEADERS)
1255 /*
1256 * We have the method, this is just an
1257 * unknown header then
1258 */
1259 if (!wsi->mux_substream)
1260 goto unknown_hdr;
1261 else
1262 break;
1263 #else
1264 break;
1265 #endif
1266 }
1267 /*
1268 * ...it's an unknown http method from a client
1269 * in fact, it cannot be valid http.
1270 *
1271 * Are we set up to transition to another role
1272 * in these cases?
1273 */
1274 if (lws_check_opt(wsi->vhost->options,
1275 LWS_SERVER_OPTION_FALLBACK_TO_APPLY_LISTEN_ACCEPT_CONFIG)) {
1276 lwsl_notice("%s: http fail fallback\n",
1277 __func__);
1278 /* transition to other role */
1279 return LPR_DO_FALLBACK;
1280 }
1281
1282 lwsl_info("Unknown method - dropping\n");
1283 goto forbid;
1284 }
1285 if (ah->lextable_pos < 0) {
1286 #if defined(LWS_WITH_CUSTOM_HEADERS)
1287 if (!wsi->mux_substream)
1288 goto unknown_hdr;
1289 #endif
1290 /*
1291 * ...otherwise for a client, let him ignore
1292 * unknown headers coming from the server
1293 */
1294 ah->parser_state = WSI_TOKEN_SKIPPING;
1295 break;
1296 }
1297
1298 if (lextable_h1[ah->lextable_pos] < FAIL_CHAR) {
1299 /* terminal state */
1300
1301 n = ((unsigned int)lextable_h1[ah->lextable_pos] << 8) |
1302 lextable_h1[ah->lextable_pos + 1];
1303
1304 lwsl_parser("known hdr %d\n", n);
1305 for (m = 0; m < LWS_ARRAY_SIZE(methods); m++)
1306 if (n == methods[m] &&
1307 ah->frag_index[methods[m]]) {
1308 lwsl_warn("Duplicated method\n");
1309 return LPR_FAIL;
1310 }
1311
1312 /*
1313 * WSORIGIN is protocol equiv to ORIGIN,
1314 * JWebSocket likes to send it, map to ORIGIN
1315 */
1316 #if defined(LWS_ROLE_WS)
1317 if (n == WSI_TOKEN_SWORIGIN)
1318 n = WSI_TOKEN_ORIGIN;
1319 #endif
1320
1321 ah->parser_state = (enum lws_token_indexes)
1322 (WSI_TOKEN_GET_URI + n);
1323 ah->ups = URIPS_IDLE;
1324
1325 if (context->token_limits)
1326 ah->current_token_limit = context->
1327 token_limits->token_limit[
1328 ah->parser_state];
1329 else
1330 ah->current_token_limit =
1331 wsi->context->max_http_header_data;
1332
1333 if (ah->parser_state == WSI_TOKEN_CHALLENGE)
1334 goto set_parsing_complete;
1335
1336 goto start_fragment;
1337 }
1338 break;
1339
1340 #if defined(LWS_WITH_CUSTOM_HEADERS)
1341 unknown_hdr:
1342 //ah->parser_state = WSI_TOKEN_SKIPPING;
1343 //break;
1344 if (!wsi->mux_substream)
1345 break;
1346 #endif
1347
1348 start_fragment:
1349 ah->nfrag++;
1350 excessive:
1351 if (ah->nfrag == LWS_ARRAY_SIZE(ah->frags)) {
1352 lwsl_warn("More hdr frags than we can deal with\n");
1353 return LPR_FAIL;
1354 }
1355
1356 ah->frags[ah->nfrag].offset = ah->pos;
1357 ah->frags[ah->nfrag].len = 0;
1358 ah->frags[ah->nfrag].nfrag = 0;
1359 ah->frags[ah->nfrag].flags = 2;
1360
1361 n = ah->frag_index[ah->parser_state];
1362 if (!n) { /* first fragment */
1363 ah->frag_index[ah->parser_state] = ah->nfrag;
1364 ah->hdr_token_idx = ah->parser_state;
1365 break;
1366 }
1367 /* continuation */
1368 while (ah->frags[n].nfrag)
1369 n = ah->frags[n].nfrag;
1370 ah->frags[n].nfrag = ah->nfrag;
1371
1372 if (issue_char(wsi, ' ') < 0)
1373 return LPR_FAIL;
1374 break;
1375
1376 /* skipping arg part of a name we didn't recognize */
1377 case WSI_TOKEN_SKIPPING:
1378 lwsl_parser("WSI_TOKEN_SKIPPING '%c'\n", c);
1379
1380 if (c == '\x0d')
1381 ah->parser_state = WSI_TOKEN_SKIPPING_SAW_CR;
1382 break;
1383
1384 case WSI_TOKEN_SKIPPING_SAW_CR:
1385 lwsl_parser("WSI_TOKEN_SKIPPING_SAW_CR '%c'\n", c);
1386 if (ah->ues != URIES_IDLE)
1387 goto forbid;
1388 if (c == '\x0a') {
1389 ah->parser_state = WSI_TOKEN_NAME_PART;
1390 ah->unk_pos = ah->lextable_pos = 0;
1391 } else
1392 ah->parser_state = WSI_TOKEN_SKIPPING;
1393 break;
1394 /* we're done, ignore anything else */
1395
1396 case WSI_PARSING_COMPLETE:
1397 lwsl_parser("WSI_PARSING_COMPLETE '%c'\n", c);
1398 break;
1399 }
1400
1401 } while (*len);
1402
1403 return LPR_OK;
1404
1405 set_parsing_complete:
1406 if (ah->ues != URIES_IDLE)
1407 goto forbid;
1408
1409 if (lws_hdr_total_length(wsi, WSI_TOKEN_UPGRADE)) {
1410 #if defined(LWS_ROLE_WS)
1411 if (lws_hdr_total_length(wsi, WSI_TOKEN_VERSION))
1412 wsi->rx_frame_type = /* temp for ws version index */
1413 atoi(lws_hdr_simple_ptr(wsi, WSI_TOKEN_VERSION));
1414
1415 lwsl_parser("v%02d hdrs done\n", wsi->rx_frame_type);
1416 #endif
1417 }
1418 ah->parser_state = WSI_PARSING_COMPLETE;
1419 wsi->hdr_parsing_completed = 1;
1420
1421 return LPR_OK;
1422
1423 forbid:
1424 lwsl_info(" forbidding on uri sanitation\n");
1425 #if defined(LWS_WITH_SERVER)
1426 lws_return_http_status(wsi, HTTP_STATUS_FORBIDDEN, NULL);
1427 #endif
1428
1429 return LPR_FORBIDDEN;
1430 }
1431
1432